nopaque/app/static/js/Utils.js

421 lines
15 KiB
JavaScript
Raw Normal View History

2022-09-02 13:07:30 +02:00
class Utils {
2023-01-12 15:40:28 +01:00
static escape(text) {
// https://codereview.stackexchange.com/a/126722
var table = {
'<': 'lt',
'>': 'gt',
'"': 'quot',
'\'': 'apos',
'&': 'amp',
'\r': '#10',
'\n': '#13'
};
return text.toString().replace(/[<>"'\r\n&]/g, (chr) => {
return '&' + table[chr] + ';';
});
};
static HTMLToElement(HTMLString) {
let templateElement = document.createElement('template');
templateElement.innerHTML = HTMLString.trim();
return templateElement.content.firstChild;
2022-09-02 13:07:30 +02:00
}
2023-01-09 10:03:52 +01:00
static generateElementId(prefix='', suffix='') {
2023-01-04 20:00:37 +01:00
for (let i = 0; true; i++) {
2023-01-09 10:03:52 +01:00
if (document.querySelector(`#${prefix}${i}${suffix}`) !== null) {continue;}
return `${prefix}${i}${suffix}`;
2023-01-04 20:00:37 +01:00
}
}
2023-01-09 08:45:47 +01:00
static isObject(object) {
return object !== null && typeof object === 'object' && !Array.isArray(object);
}
static mergeObjectsDeep(...objects) {
let mergedObject = {};
if (objects.length === 0) {
return mergedObject;
}
2023-01-09 10:03:52 +01:00
if (!Utils.isObject(objects[0])) {throw 'Cannot merge non-object';}
2023-01-09 08:45:47 +01:00
if (objects.length === 1) {
return Utils.mergeObjectsDeep(mergedObject, objects[0]);
}
2023-01-09 10:03:52 +01:00
if (!Utils.isObject(objects[1])) {throw 'Cannot merge non-object';}
2023-01-09 08:45:47 +01:00
for (let key in objects[0]) {
if (objects[0].hasOwnProperty(key)) {
if (objects[1].hasOwnProperty(key)) {
if (Utils.isObject(objects[0][key]) && Utils.isObject(objects[1][key])) {
mergedObject[key] = Utils.mergeObjectsDeep(objects[0][key], objects[1][key]);
} else {
mergedObject[key] = objects[1][key];
}
} else {
mergedObject[key] = objects[0][key];
}
}
}
for (let key in objects[1]) {
if (objects[1].hasOwnProperty(key)) {
if (!objects[0].hasOwnProperty(key)) {
mergedObject[key] = objects[1][key];
}
}
}
if (objects.length === 2) {
return mergedObject;
}
return Utils.mergeObjectsDeep(mergedObject, ...objects.slice(2));
}
2022-09-02 13:07:30 +02:00
static deleteCorpusRequest(userId, corpusId) {
return new Promise((resolve, reject) => {
2023-01-09 10:03:52 +01:00
let corpus;
try {
corpus = app.data.users[userId].corpora[corpusId];
} catch (error) {
corpus = {};
}
2022-09-02 13:07:30 +02:00
let modalElement = Utils.HTMLToElement(
2022-09-02 13:07:30 +02:00
`
<div class="modal">
<div class="modal-content">
<h4>Confirm Corpus deletion</h4>
2023-01-09 10:03:52 +01:00
<p>Do you really want to delete the Corpus <b>${corpus?.title}</b>? All files will be permanently deleted!</p>
2022-09-02 13:07:30 +02:00
</div>
<div class="modal-footer">
<a class="action-button btn modal-close waves-effect waves-light" data-action="cancel">Cancel</a>
<a class="action-button btn modal-close red waves-effect waves-light" data-action="confirm">Delete</a>
</div>
</div>
`
);
document.querySelector('#modals').appendChild(modalElement);
let modal = M.Modal.init(
modalElement,
{
dismissible: false,
onCloseEnd: () => {
modal.destroy();
modalElement.remove();
}
}
);
let confirmElement = modalElement.querySelector('.action-button[data-action="confirm"]');
confirmElement.addEventListener('click', (event) => {
2023-01-09 10:03:52 +01:00
let corpusTitle = corpus?.title;
fetch(`/corpora/${corpusId}`, {method: 'DELETE', headers: {Accept: 'application/json'}})
2022-09-02 13:07:30 +02:00
.then(
(response) => {
2022-11-21 15:41:07 +01:00
if (response.status === 403) {app.flash('Forbidden', 'error'); reject(response);}
if (response.status === 404) {app.flash('Not Found', 'error'); reject(response);}
2023-01-09 10:03:52 +01:00
app.flash(`Corpus "${corpusTitle}" marked for deletion`, 'corpus');
2022-11-21 15:41:07 +01:00
resolve(response);
2022-09-02 13:07:30 +02:00
},
(response) => {
2022-11-21 14:08:49 +01:00
app.flash('Something went wrong', 'error');
2022-09-02 13:07:30 +02:00
reject(response);
}
);
});
modal.open();
});
}
static deleteCorpusFileRequest(userId, corpusId, corpusFileId) {
return new Promise((resolve, reject) => {
2023-01-09 10:03:52 +01:00
let corpusFile;
try {
corpusFile = app.data.users[userId].corpora[corpusId].files[corpusFileId];
} catch (error) {
corpusFile = {};
}
2022-09-02 13:07:30 +02:00
let modalElement = Utils.HTMLToElement(
2022-09-02 13:07:30 +02:00
`
<div class="modal">
<div class="modal-content">
2022-11-15 15:11:16 +01:00
<h4>Confirm Corpus File deletion</h4>
2023-01-09 10:03:52 +01:00
<p>Do you really want to delete the Corpus File <b>${corpusFile?.title}</b>? All files will be permanently deleted!</p>
2022-09-02 13:07:30 +02:00
</div>
<div class="modal-footer">
<a class="action-button btn modal-close waves-effect waves-light" data-action="cancel">Cancel</a>
<a class="action-button btn modal-close red waves-effect waves-light" data-action="confirm">Delete</a>
</div>
</div>
`
);
document.querySelector('#modals').appendChild(modalElement);
let modal = M.Modal.init(
modalElement,
{
dismissible: false,
onCloseEnd: () => {
modal.destroy();
modalElement.remove();
}
}
);
let confirmElement = modalElement.querySelector('.action-button[data-action="confirm"]');
confirmElement.addEventListener('click', (event) => {
2023-01-09 10:03:52 +01:00
let corpusFileTitle = corpusFile?.title;
2022-09-02 13:07:30 +02:00
fetch(`/corpora/${corpusId}/files/${corpusFileId}`, {method: 'DELETE', headers: {Accept: 'application/json'}})
.then(
(response) => {
2022-11-21 15:41:07 +01:00
if (response.status === 403) {app.flash('Forbidden', 'error'); reject(response);}
if (response.status === 404) {app.flash('Not Found', 'error'); reject(response);}
app.flash(`Corpus File "${corpusFileTitle}" deleted`, 'corpus');
resolve(response);
2022-09-02 13:07:30 +02:00
},
(response) => {
2022-11-21 14:08:49 +01:00
app.flash('Something went wrong', 'error');
2022-11-15 15:11:16 +01:00
reject(response);
}
);
});
modal.open();
});
}
2022-12-19 16:25:10 +01:00
static deleteProfileAvatarRequest(userId) {
return new Promise((resolve, reject) => {
let modalElement = Utils.HTMLToElement(
2022-12-19 16:25:10 +01:00
`
<div class="modal">
<div class="modal-content">
<h4>Confirm Avatar deletion</h4>
<p>Do you really want to delete your Avatar?</p>
</div>
<div class="modal-footer">
<a class="action-button btn modal-close waves-effect waves-light" data-action="cancel">Cancel</a>
<a class="action-button btn modal-close red waves-effect waves-light" data-action="confirm">Delete</a>
</div>
</div>
`
);
document.querySelector('#modals').appendChild(modalElement);
let modal = M.Modal.init(
modalElement,
{
dismissible: false,
onCloseEnd: () => {
modal.destroy();
modalElement.remove();
}
}
);
let confirmElement = modalElement.querySelector('.action-button[data-action="confirm"]');
confirmElement.addEventListener('click', (event) => {
2022-12-23 18:08:33 +01:00
fetch(`/users/${userId}/avatar`, {method: 'DELETE', headers: {Accept: 'application/json'}})
2022-12-19 16:25:10 +01:00
.then(
(response) => {
if (response.status === 403) {app.flash('Forbidden', 'error'); reject(response);}
if (response.status === 404) {app.flash('Not Found', 'error'); reject(response);}
app.flash(`Avatar marked for deletion`);
resolve(response);
},
(response) => {
app.flash('Something went wrong', 'error');
reject(response);
}
);
});
modal.open();
});
}
2023-03-10 08:47:03 +01:00
// static deleteJobRequest(userId, jobId) {
// return new Promise((resolve, reject) => {
// let job;
// try {
// job = app.data.users[userId].jobs[jobId];
// } catch (error) {
// job = {};
// }
2022-09-02 13:07:30 +02:00
2023-03-10 08:47:03 +01:00
// let confirmElement = modalElement.querySelector('.action-button[data-action="confirm"]');
// confirmElement.addEventListener('click', (event) => {
// let jobTitle = job?.title;
// fetch(`/jobs/${jobId}`, {method: 'DELETE', headers: {Accept: 'application/json'}})
// .then(
// (response) => {
// if (response.status === 403) {app.flash('Forbidden', 'error'); reject(response);}
// if (response.status === 404) {app.flash('Not Found', 'error'); reject(response);}
// app.flash(`Job "${jobTitle}" marked for deletion`, 'job');
// resolve(response);
// },
// (response) => {
// app.flash('Something went wrong', 'error');
// reject(response);
// }
// );
// });
// modal.open();
// });
// }
2022-09-02 13:07:30 +02:00
2023-03-10 08:47:03 +01:00
// static getJobLogRequest(userId, jobId) {
// return new Promise((resolve, reject) => {
// fetch(`/jobs/${jobId}/log`, {method: 'GET', headers: {Accept: 'application/json, text/plain'}})
// .then(
// (response) => {
// if (response.status === 403) {app.flash('Forbidden', 'error'); reject(response);}
// if (response.status === 404) {app.flash('Not Found', 'error'); reject(response);}
// return response.text();
// },
// (response) => {
// app.flash('Something went wrong', 'error');
// reject(response);
// }
// )
// .then(
// (text) => {
// let modalElement = Utils.HTMLToElement(
// `
// <div class="modal">
// <div class="modal-content">
// <h4>Job logs</h4>
// <pre><code>${text}</code></pre>
// </div>
// <div class="modal-footer">
// <a class="btn modal-close waves-effect waves-light">Close</a>
// </div>
// </div>
// `
// );
// document.querySelector('#modals').appendChild(modalElement);
// let modal = M.Modal.init(
// modalElement,
// {
// onCloseEnd: () => {
// modal.destroy();
// modalElement.remove();
// }
// }
// );
// modal.open();
// resolve(text);
// }
// );
// });
// }
2022-09-02 13:07:30 +02:00
2023-03-10 08:47:03 +01:00
// static restartJobRequest(userId, jobId) {
// return new Promise((resolve, reject) => {
// let job;
// try {
// job = app.data.users[userId].jobs[jobId];
// } catch (error) {
// job = {};
// }
2022-09-02 13:07:30 +02:00
2023-03-10 08:47:03 +01:00
// let modalElement = Utils.HTMLToElement(
// `
// <div class="modal">
// <div class="modal-content">
// <h4>Confirm Job restart</h4>
// <p>Do you really want to restart the Job <b>${job?.title}</b>? All Job Results will be permanently deleted.</p>
// </div>
// <div class="modal-footer">
// <a class="action-button btn modal-close waves-effect waves-light" data-action="cancel">Cancel</a>
// <a class="action-button btn modal-close red waves-effect waves-light" data-action="confirm">Restart</a>
// </div>
// </div>
// `
// );
// document.querySelector('#modals').appendChild(modalElement);
// let modal = M.Modal.init(
// modalElement,
// {
// dismissible: false,
// onCloseEnd: () => {
// modal.destroy();
// modalElement.remove();
// }
// }
// );
2022-09-02 13:07:30 +02:00
2023-03-10 08:47:03 +01:00
// let confirmElement = modalElement.querySelector('.action-button[data-action="confirm"]');
// confirmElement.addEventListener('click', (event) => {
// let jobTitle = job?.title;
// fetch(`/jobs/${jobId}/restart`, {method: 'POST', headers: {Accept: 'application/json'}})
// .then(
// (response) => {
// if (response.status === 403) {app.flash('Forbidden', 'error'); reject(response);}
// if (response.status === 404) {app.flash('Not Found', 'error'); reject(response);}
// if (response.status === 409) {app.flash('Conflict', 'error'); reject(response);}
// app.flash(`Job "${jobTitle}" restarted.`, 'job');
// resolve(response);
// },
// (response) => {
// app.flash('Something went wrong', 'error');
// reject(response);
// }
// );
// });
// modal.open();
// });
// }
2022-09-02 13:07:30 +02:00
static deleteUserRequest(userId) {
return new Promise((resolve, reject) => {
2023-01-09 10:03:52 +01:00
let user;
try {
user = app.data.users[userId];
} catch (error) {
user = {};
}
2022-09-02 13:07:30 +02:00
let modalElement = Utils.HTMLToElement(
2022-09-02 13:07:30 +02:00
`
<div class="modal">
<div class="modal-content">
<h4>Confirm User deletion</h4>
2023-01-09 10:03:52 +01:00
<p>Do you really want to delete the User <b>${user?.username}</b>? All files will be permanently deleted!</p>
2022-09-02 13:07:30 +02:00
</div>
<div class="modal-footer">
<a class="action-button btn modal-close waves-effect waves-light" data-action="cancel">Cancel</a>
<a class="action-button btn modal-close red waves-effect waves-light" data-action="confirm">Delete</a>
</div>
</div>
`
);
document.querySelector('#modals').appendChild(modalElement);
let modal = M.Modal.init(
modalElement,
{
dismissible: false,
onCloseEnd: () => {
modal.destroy();
modalElement.remove();
}
}
);
let confirmElement = modalElement.querySelector('.action-button[data-action="confirm"]');
confirmElement.addEventListener('click', (event) => {
2023-01-09 10:03:52 +01:00
let userName = user?.username;
fetch(`/users/${userId}`, {method: 'DELETE', headers: {Accept: 'application/json'}})
2022-09-02 13:07:30 +02:00
.then(
(response) => {
2022-11-21 15:41:07 +01:00
if (response.status === 403) {app.flash('Forbidden', 'error'); reject(response);}
if (response.status === 404) {app.flash('Not Found', 'error'); reject(response);}
app.flash(`User "${userName}" marked for deletion`);
resolve(response);
2022-09-02 13:07:30 +02:00
},
(response) => {
2022-11-21 14:08:49 +01:00
app.flash('Something went wrong', 'error');
2022-09-02 13:07:30 +02:00
reject(response);
}
);
});
modal.open();
});
}
}