2021-07-16 11:41:58 -03:00
|
|
|
/*
|
2021-12-25 14:39:34 -08:00
|
|
|
* Copyright [2021] [wisemapping]
|
2021-07-16 11:41:58 -03:00
|
|
|
*
|
|
|
|
* Licensed under WiseMapping Public License, Version 1.0 (the "License").
|
|
|
|
* It is basically the Apache License, Version 2.0 (the "License") plus the
|
|
|
|
* "powered by wisemapping" text requirement on every single page;
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the license at
|
|
|
|
*
|
|
|
|
* http://www.wisemapping.org/license
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2021-12-03 16:11:17 -08:00
|
|
|
import { $assert } from '@wisemapping/core-js';
|
2021-12-14 17:54:11 -08:00
|
|
|
import { $msg } from './Messages';
|
2022-02-21 02:50:03 +00:00
|
|
|
import PersistenceManager, { PersistenceError } from './PersistenceManager';
|
2021-07-16 11:41:58 -03:00
|
|
|
|
2021-12-14 21:42:52 -08:00
|
|
|
class RESTPersistenceManager extends PersistenceManager {
|
2022-01-30 18:33:26 -08:00
|
|
|
private documentUrl: string;
|
2022-01-31 10:12:05 -03:00
|
|
|
|
2022-01-30 18:33:26 -08:00
|
|
|
private revertUrl: string;
|
2022-01-31 10:12:05 -03:00
|
|
|
|
2022-01-30 18:33:26 -08:00
|
|
|
private lockUrl: string;
|
2022-01-31 10:12:05 -03:00
|
|
|
|
2022-01-30 18:33:26 -08:00
|
|
|
private onSave: boolean;
|
2022-01-31 10:12:05 -03:00
|
|
|
|
2022-01-30 18:33:26 -08:00
|
|
|
private clearTimeout;
|
|
|
|
|
2022-07-13 01:45:36 +00:00
|
|
|
constructor(options: { documentUrl: string; revertUrl: string; lockUrl: string }) {
|
2021-10-04 17:05:34 -07:00
|
|
|
$assert(options.documentUrl, 'documentUrl can not be null');
|
|
|
|
$assert(options.revertUrl, 'revertUrl can not be null');
|
|
|
|
$assert(options.lockUrl, 'lockUrl can not be null');
|
2021-12-14 21:42:52 -08:00
|
|
|
super();
|
2021-10-04 17:05:34 -07:00
|
|
|
|
|
|
|
this.documentUrl = options.documentUrl;
|
|
|
|
this.revertUrl = options.revertUrl;
|
|
|
|
this.lockUrl = options.lockUrl;
|
2021-12-04 15:39:20 -08:00
|
|
|
}
|
2021-10-04 17:05:34 -07:00
|
|
|
|
2022-02-13 18:06:50 -08:00
|
|
|
saveMapXml(mapId: string, mapXml: Document, pref: string, saveHistory: boolean, events): void {
|
2021-10-04 17:05:34 -07:00
|
|
|
const data = {
|
|
|
|
id: mapId,
|
2022-02-13 18:06:50 -08:00
|
|
|
xml: new XMLSerializer().serializeToString(mapXml),
|
2021-10-04 17:05:34 -07:00
|
|
|
properties: pref,
|
|
|
|
};
|
|
|
|
|
2022-03-23 23:33:44 -03:00
|
|
|
const query = `minor=${!saveHistory}`;
|
2021-10-04 17:05:34 -07:00
|
|
|
|
2022-01-30 18:33:26 -08:00
|
|
|
if (!this.onSave) {
|
2021-10-04 17:05:34 -07:00
|
|
|
// Mark save in process and fire a event unlocking the save ...
|
2022-01-30 18:33:26 -08:00
|
|
|
this.onSave = true;
|
|
|
|
this.clearTimeout = setTimeout(() => {
|
|
|
|
this.clearTimeout = null;
|
|
|
|
this.onSave = false;
|
2021-12-14 21:42:52 -08:00
|
|
|
}, 10000);
|
2021-10-04 17:05:34 -07:00
|
|
|
|
2022-01-30 18:33:26 -08:00
|
|
|
const persistence = this;
|
2022-11-29 21:48:53 -08:00
|
|
|
const crfs = this.getCSRFToken();
|
2022-07-13 01:45:36 +00:00
|
|
|
fetch(`${this.documentUrl.replace('{id}', mapId)}?${query}`, {
|
|
|
|
method: 'PUT',
|
|
|
|
// Blob helps to resuce the memory on large payload.
|
|
|
|
body: new Blob([JSON.stringify(data)], { type: 'text/plain' }),
|
2022-11-29 21:48:53 -08:00
|
|
|
headers: crfs
|
|
|
|
? {
|
|
|
|
'Content-Type': 'application/json; charset=utf-8',
|
|
|
|
Accept: 'application/json',
|
|
|
|
'X-CSRF-Token': crfs,
|
|
|
|
}
|
|
|
|
: {
|
|
|
|
'Content-Type': 'application/json; charset=utf-8',
|
|
|
|
Accept: 'application/json',
|
|
|
|
},
|
2022-07-13 01:45:36 +00:00
|
|
|
})
|
|
|
|
.then(async (response: Response) => {
|
|
|
|
if (response.ok) {
|
|
|
|
events.onSuccess();
|
2022-02-13 18:06:50 -08:00
|
|
|
} else {
|
2022-07-13 01:45:36 +00:00
|
|
|
console.log(`Saving error: ${response.status}`);
|
2022-11-29 21:48:53 -08:00
|
|
|
let userMsg: PersistenceError | null = null;
|
2022-07-13 01:45:36 +00:00
|
|
|
if (response.status === 405) {
|
|
|
|
userMsg = {
|
|
|
|
severity: 'SEVERE',
|
|
|
|
message: $msg('SESSION_EXPIRED'),
|
|
|
|
errorType: 'session-expired',
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
const responseText = await response.text();
|
|
|
|
const contentType = response.headers['Content-Type'];
|
|
|
|
if (contentType != null && contentType.indexOf('application/json') !== -1) {
|
2022-11-29 21:48:53 -08:00
|
|
|
let serverMsg: null | { globalSeverity: string } = null;
|
2022-07-13 01:45:36 +00:00
|
|
|
try {
|
|
|
|
serverMsg = JSON.parse(responseText);
|
2022-11-29 21:48:53 -08:00
|
|
|
serverMsg = serverMsg && serverMsg.globalSeverity ? serverMsg : null;
|
2022-07-13 01:45:36 +00:00
|
|
|
} catch (e) {
|
|
|
|
// Message could not be decoded ...
|
|
|
|
}
|
|
|
|
userMsg = persistence._buildError(serverMsg);
|
2022-02-13 18:06:50 -08:00
|
|
|
}
|
|
|
|
}
|
2022-11-29 21:48:53 -08:00
|
|
|
if (userMsg) {
|
|
|
|
this.triggerError(userMsg);
|
|
|
|
events.onError(userMsg);
|
|
|
|
}
|
2022-07-13 01:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Clear event timeout ...
|
|
|
|
if (persistence.clearTimeout) {
|
|
|
|
clearTimeout(persistence.clearTimeout);
|
2021-10-04 17:05:34 -07:00
|
|
|
}
|
2022-07-13 01:45:36 +00:00
|
|
|
persistence.onSave = false;
|
|
|
|
})
|
|
|
|
.catch(() => {
|
|
|
|
const userMsg: PersistenceError = {
|
|
|
|
severity: 'SEVERE',
|
|
|
|
message: $msg('SAVE_COULD_NOT_BE_COMPLETED'),
|
|
|
|
errorType: 'generic',
|
|
|
|
};
|
2022-02-21 02:50:03 +00:00
|
|
|
this.triggerError(userMsg);
|
2021-10-04 17:05:34 -07:00
|
|
|
events.onError(userMsg);
|
2022-07-13 01:45:36 +00:00
|
|
|
|
|
|
|
// Clear event timeout ...
|
|
|
|
if (persistence.clearTimeout) {
|
|
|
|
clearTimeout(persistence.clearTimeout);
|
|
|
|
}
|
|
|
|
persistence.onSave = false;
|
|
|
|
});
|
2021-10-04 17:05:34 -07:00
|
|
|
}
|
2021-12-04 15:39:20 -08:00
|
|
|
}
|
2021-10-04 17:05:34 -07:00
|
|
|
|
2022-11-29 21:48:53 -08:00
|
|
|
discardChanges(mapId: string): void {
|
|
|
|
const csrfToken = this.getCSRFToken();
|
|
|
|
|
2022-07-13 01:45:36 +00:00
|
|
|
fetch(this.revertUrl.replace('{id}', mapId), {
|
|
|
|
method: 'POST',
|
2022-11-29 21:48:53 -08:00
|
|
|
headers: csrfToken
|
|
|
|
? {
|
|
|
|
'Content-Type': 'application/json; charset=utf-8',
|
|
|
|
Accept: 'application/json',
|
|
|
|
'X-CSRF-Token': csrfToken,
|
|
|
|
}
|
|
|
|
: {
|
|
|
|
'Content-Type': 'application/json; charset=utf-8',
|
|
|
|
Accept: 'application/json',
|
|
|
|
},
|
2022-07-13 01:45:36 +00:00
|
|
|
});
|
2021-12-04 15:39:20 -08:00
|
|
|
}
|
2021-10-04 17:05:34 -07:00
|
|
|
|
2022-02-18 10:00:26 -08:00
|
|
|
unlockMap(mapId: string): void {
|
2022-11-29 21:48:53 -08:00
|
|
|
const csrfToken = this.getCSRFToken();
|
2022-07-13 01:45:36 +00:00
|
|
|
fetch(this.lockUrl.replace('{id}', mapId), {
|
|
|
|
method: 'PUT',
|
2022-11-29 21:48:53 -08:00
|
|
|
headers: csrfToken
|
|
|
|
? {
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
'X-CSRF-Token': csrfToken,
|
|
|
|
}
|
|
|
|
: {
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
},
|
2022-07-13 01:45:36 +00:00
|
|
|
body: 'false',
|
|
|
|
});
|
2021-12-04 15:39:20 -08:00
|
|
|
}
|
2021-10-04 17:05:34 -07:00
|
|
|
|
2022-01-30 18:33:26 -08:00
|
|
|
private _buildError(jsonSeverResponse) {
|
2021-10-04 17:05:34 -07:00
|
|
|
let message = jsonSeverResponse ? jsonSeverResponse.globalErrors[0] : null;
|
|
|
|
let severity = jsonSeverResponse ? jsonSeverResponse.globalSeverity : null;
|
|
|
|
|
|
|
|
if (!message) {
|
|
|
|
message = $msg('SAVE_COULD_NOT_BE_COMPLETED');
|
|
|
|
}
|
2021-07-16 11:41:58 -03:00
|
|
|
|
2021-10-04 17:05:34 -07:00
|
|
|
if (!severity) {
|
|
|
|
severity = 'INFO';
|
|
|
|
}
|
|
|
|
return { severity, message };
|
2021-12-04 15:39:20 -08:00
|
|
|
}
|
2021-10-04 17:05:34 -07:00
|
|
|
|
2022-11-15 22:45:59 -08:00
|
|
|
loadMapDom(mapId: string): Promise<Document> {
|
|
|
|
const url = `${this.documentUrl.replace('{id}', mapId)}/xml`;
|
2022-11-29 21:48:53 -08:00
|
|
|
const csrfToken = this.getCSRFToken();
|
2022-11-15 22:45:59 -08:00
|
|
|
return fetch(url, {
|
2021-10-04 17:05:34 -07:00
|
|
|
method: 'get',
|
2022-11-29 21:48:53 -08:00
|
|
|
headers: csrfToken
|
|
|
|
? {
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
Accept: 'application/xml',
|
|
|
|
'X-CSRF-Token': csrfToken,
|
|
|
|
}
|
|
|
|
: {
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
Accept: 'application/xml',
|
|
|
|
},
|
2022-11-15 22:45:59 -08:00
|
|
|
})
|
|
|
|
.then((response: Response) => {
|
|
|
|
if (!response.ok) {
|
|
|
|
console.error(`load error: ${response.status}`);
|
|
|
|
throw new Error(`load error: ${response.status}, ${response.statusText}`);
|
|
|
|
}
|
|
|
|
return response.text();
|
|
|
|
})
|
|
|
|
.then((xmlStr) => new DOMParser().parseFromString(xmlStr, 'text/xml'));
|
2021-12-04 15:39:20 -08:00
|
|
|
}
|
|
|
|
}
|
2021-07-16 11:41:58 -03:00
|
|
|
|
|
|
|
export default RESTPersistenceManager;
|