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] + ';';
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2023-01-10 10:04:01 +01:00
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2023-02-23 15:18:53 +01:00
|
|
|
static updateCorpusFollowerRole(corpusId, followerId, roleName) {
|
2023-02-15 16:17:25 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
2023-02-23 15:18:53 +01:00
|
|
|
fetch(`/corpora/${corpusId}/followers/${followerId}/role`, {method: 'POST', headers: {Accept: 'application/json', 'Content-Type': 'application/json'}, body: JSON.stringify({role: roleName})})
|
2023-02-15 16:17:25 +01:00
|
|
|
.then(
|
|
|
|
(response) => {
|
2023-02-20 16:15:17 +01:00
|
|
|
if (response.ok) {
|
2023-02-23 15:18:53 +01:00
|
|
|
app.flash('Role updated', 'corpus');
|
2023-02-20 16:15:17 +01:00
|
|
|
resolve(response);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
app.flash(`${response.statusText}`, 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
2023-02-15 16:17:25 +01:00
|
|
|
},
|
|
|
|
(response) => {
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-02-09 16:22:37 +01:00
|
|
|
static enableCorpusIsPublicRequest(userId, corpusId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let corpus;
|
|
|
|
try {
|
|
|
|
corpus = app.data.users[userId].corpora[corpusId];
|
|
|
|
} catch (error) {
|
|
|
|
corpus = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
let modalElement = Utils.HTMLToElement(
|
|
|
|
`
|
|
|
|
<div class="modal">
|
|
|
|
<div class="modal-content">
|
|
|
|
<h4>Hier könnte eine Warnung stehen</h4>
|
|
|
|
<p></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">Confirm</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) => {
|
|
|
|
let corpusTitle = corpus?.title;
|
2023-02-20 10:40:33 +01:00
|
|
|
fetch(`/corpora/${corpusId}/is_public/enable`, {method: 'POST', headers: {Accept: 'application/json'}})
|
2023-02-09 16:22:37 +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(`Corpus "${corpusTitle}" is public now`, 'corpus');
|
|
|
|
resolve(response);
|
|
|
|
},
|
|
|
|
(response) => {
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
modal.open();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static disableCorpusIsPublicRequest(userId, corpusId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let corpus;
|
|
|
|
try {
|
|
|
|
corpus = app.data.users[userId].corpora[corpusId];
|
|
|
|
} catch (error) {
|
|
|
|
corpus = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
let corpusTitle = corpus?.title;
|
2023-02-20 10:40:33 +01:00
|
|
|
fetch(`/corpora/${corpusId}/is_public/disable`, {method: 'POST', headers: {Accept: 'application/json'}})
|
2023-02-09 16:22:37 +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(`Corpus "${corpusTitle}" is private now`, 'corpus');
|
|
|
|
resolve(response);
|
|
|
|
},
|
|
|
|
(response) => {
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-09-02 13:07:30 +02:00
|
|
|
static buildCorpusRequest(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
|
|
|
|
2023-01-09 10:03:52 +01:00
|
|
|
fetch(`/corpora/${corpusId}/build`, {method: 'POST', 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);}
|
|
|
|
if (response.status === 409) {app.flash('Conflict', 'error'); reject(response);}
|
2023-01-09 10:03:52 +01:00
|
|
|
app.flash(`Corpus "${corpus?.title}" marked for building`, '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);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-02-20 16:15:17 +01:00
|
|
|
static unfollowCorpusRequest(corpusId, followerId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
fetch(`/corpora/${corpusId}/followers/${followerId}/unfollow`, {method: 'POST', headers: {Accept: 'application/json'}})
|
|
|
|
.then(
|
|
|
|
(response) => {
|
|
|
|
if (response.ok) {
|
|
|
|
app.flash(`User unfollowed from Corpus`, 'corpus');
|
|
|
|
resolve(response);
|
|
|
|
} else {
|
|
|
|
app.flash(`${response.statusText}`, 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
(response) => {
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2023-01-10 10:04:01 +01:00
|
|
|
let modalElement = Utils.HTMLToElement(
|
2022-09-02 13:07:30 +02:00
|
|
|
`
|
|
|
|
<div class="modal">
|
|
|
|
<div class="modal-content">
|
2022-11-16 14:15:52 +01:00
|
|
|
<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
|
|
|
|
2023-01-10 10:04:01 +01: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();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static deleteSpaCyNLPPipelineModelRequest(userId, spaCyNLPPipelineModelId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-09 10:03:52 +01:00
|
|
|
let spaCyNLPPipelineModel;
|
|
|
|
try {
|
|
|
|
spaCyNLPPipelineModel = app.data.users[userId].spacy_nlp_pipeline_models[spaCyNLPPipelineModelId];
|
|
|
|
} catch (error) {
|
|
|
|
spaCyNLPPipelineModel = {};
|
|
|
|
}
|
|
|
|
|
2023-01-10 10:04:01 +01:00
|
|
|
let modalElement = Utils.HTMLToElement(
|
2022-11-15 15:11:16 +01:00
|
|
|
`
|
|
|
|
<div class="modal">
|
|
|
|
<div class="modal-content">
|
|
|
|
<h4>Confirm SpaCy NLP Pipeline Model deletion</h4>
|
2023-01-09 10:03:52 +01:00
|
|
|
<p>Do you really want to delete the SpaCy NLP Pipeline Model <b>${spaCyNLPPipelineModel?.title}</b>? All files will be permanently deleted!</p>
|
2022-11-15 15:11:16 +01: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 spaCyNLPPipelineModelTitle = spaCyNLPPipelineModel?.title;
|
2022-11-15 15:11:16 +01:00
|
|
|
fetch(`/contributions/spacy-nlp-pipeline-models/${spaCyNLPPipelineModelId}`, {method: 'DELETE'})
|
|
|
|
.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(`SpaCy NLP Pipeline Model "${spaCyNLPPipelineModelTitle}" marked for deletion`);
|
|
|
|
resolve(response);
|
2022-11-15 15:11:16 +01: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();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static deleteTesseractOCRPipelineModelRequest(userId, tesseractOCRPipelineModelId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-09 10:03:52 +01:00
|
|
|
let tesseractOCRPipelineModel;
|
|
|
|
try {
|
|
|
|
tesseractOCRPipelineModel = app.data.users[userId].tesseract_ocr_pipeline_models[tesseractOCRPipelineModelId];
|
|
|
|
} catch (error) {
|
|
|
|
tesseractOCRPipelineModel = {};
|
|
|
|
}
|
|
|
|
|
2023-01-10 10:04:01 +01:00
|
|
|
let modalElement = Utils.HTMLToElement(
|
2022-11-15 15:11:16 +01:00
|
|
|
`
|
|
|
|
<div class="modal">
|
|
|
|
<div class="modal-content">
|
|
|
|
<h4>Confirm Tesseract OCR Pipeline Model deletion</h4>
|
2023-01-09 10:03:52 +01:00
|
|
|
<p>Do you really want to delete the Tesseract OCR Pipeline Model <b>${tesseractOCRPipelineModel?.title}</b>? All files will be permanently deleted!</p>
|
2022-11-15 15:11:16 +01: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 tesseractOCRPipelineModelTitle = tesseractOCRPipelineModel?.title;
|
2022-11-15 15:11:16 +01:00
|
|
|
fetch(`/contributions/tesseract-ocr-pipeline-models/${tesseractOCRPipelineModelId}`, {method: 'DELETE'})
|
|
|
|
.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(`Tesseract OCR Pipeline Model "${tesseractOCRPipelineModelTitle}" marked for deletion`);
|
|
|
|
resolve(response);
|
2022-11-15 15:11:16 +01: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();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-12-19 16:25:10 +01:00
|
|
|
static deleteProfileAvatarRequest(userId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-10 10:04:01 +01:00
|
|
|
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();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-09-02 13:07:30 +02:00
|
|
|
static deleteJobRequest(userId, jobId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-09 10:03:52 +01:00
|
|
|
let job;
|
|
|
|
try {
|
|
|
|
job = app.data.users[userId].jobs[jobId];
|
|
|
|
} catch (error) {
|
|
|
|
job = {};
|
|
|
|
}
|
2022-09-02 13:07:30 +02:00
|
|
|
|
2023-01-10 10:04:01 +01:00
|
|
|
let modalElement = Utils.HTMLToElement(
|
2022-09-02 13:07:30 +02:00
|
|
|
`
|
|
|
|
<div class="modal">
|
|
|
|
<div class="modal-content">
|
2022-11-16 14:15:52 +01:00
|
|
|
<h4>Confirm Job deletion</h4>
|
2023-01-09 10:03:52 +01:00
|
|
|
<p>Do you really want to delete the Job <b>${job?.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 jobTitle = job?.title;
|
|
|
|
fetch(`/jobs/${jobId}`, {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(`Job "${jobTitle}" marked for deletion`, 'job');
|
|
|
|
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 getJobLogRequest(userId, jobId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-09 10:03:52 +01:00
|
|
|
fetch(`/jobs/${jobId}/log`, {method: 'GET', headers: {Accept: 'application/json, text/plain'}})
|
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);}
|
|
|
|
return response.text();
|
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);
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.then(
|
|
|
|
(text) => {
|
2023-01-10 10:04:01 +01:00
|
|
|
let modalElement = Utils.HTMLToElement(
|
2022-09-02 13:07:30 +02:00
|
|
|
`
|
|
|
|
<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();
|
2022-11-21 15:41:07 +01:00
|
|
|
resolve(text);
|
2022-09-02 13:07:30 +02:00
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static restartJobRequest(userId, jobId) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-09 10:03:52 +01:00
|
|
|
let job;
|
|
|
|
try {
|
|
|
|
job = app.data.users[userId].jobs[jobId];
|
|
|
|
} catch (error) {
|
|
|
|
job = {};
|
|
|
|
}
|
2022-09-02 13:07:30 +02:00
|
|
|
|
2023-01-10 10:04:01 +01:00
|
|
|
let modalElement = Utils.HTMLToElement(
|
2022-09-02 13:07:30 +02:00
|
|
|
`
|
|
|
|
<div class="modal">
|
|
|
|
<div class="modal-content">
|
2022-11-16 14:15:52 +01:00
|
|
|
<h4>Confirm Job restart</h4>
|
2023-01-09 10:03:52 +01:00
|
|
|
<p>Do you really want to restart the Job <b>${job?.title}</b>? All Job Results 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">Restart</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 jobTitle = job?.title;
|
|
|
|
fetch(`/jobs/${jobId}/restart`, {method: 'POST', 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);}
|
|
|
|
if (response.status === 409) {app.flash('Conflict', 'error'); reject(response);}
|
|
|
|
app.flash(`Job "${jobTitle}" restarted.`, 'job');
|
|
|
|
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 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
|
|
|
|
2023-01-10 10:04:01 +01:00
|
|
|
let modalElement = Utils.HTMLToElement(
|
2022-09-02 13:07:30 +02:00
|
|
|
`
|
|
|
|
<div class="modal">
|
|
|
|
<div class="modal-content">
|
2022-11-16 14:15:52 +01:00
|
|
|
<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();
|
|
|
|
});
|
|
|
|
}
|
2022-11-17 12:11:21 +01:00
|
|
|
|
2023-01-11 13:29:47 +01:00
|
|
|
static tesseractOCRPipelineModelToggleIsPublicRequest(userId, tesseractOCRPipelineModelId, is_public) {
|
2022-11-17 12:11:21 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-09 10:03:52 +01:00
|
|
|
let tesseractOCRPipelineModel;
|
|
|
|
try {
|
|
|
|
tesseractOCRPipelineModel = app.data.users[userId].tesseract_ocr_pipeline_models[tesseractOCRPipelineModelId];
|
|
|
|
} catch (error) {
|
|
|
|
tesseractOCRPipelineModel = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
fetch(`/contributions/tesseract-ocr-pipeline-models/${tesseractOCRPipelineModelId}/toggle-public-status`, {method: 'POST', headers: {Accept: 'application/json'}})
|
2023-01-11 13:29:47 +01:00
|
|
|
.then(
|
|
|
|
(response) => {
|
|
|
|
if (response.status === 403) {
|
|
|
|
app.flash('Forbidden', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
resolve(response);
|
|
|
|
},
|
|
|
|
(response) => {
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
);
|
2022-11-17 12:11:21 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-01-11 13:29:47 +01:00
|
|
|
static spaCyNLPPipelineModelToggleIsPublicRequest(userId, spaCyNLPPipelineModelId) {
|
2022-11-17 12:11:21 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
2023-01-09 10:03:52 +01:00
|
|
|
let spaCyNLPPipelineModel;
|
|
|
|
try {
|
|
|
|
spaCyNLPPipelineModel = app.data.users[userId].spacy_nlp_pipeline_models[spaCyNLPPipelineModelId];
|
|
|
|
} catch (error) {
|
|
|
|
spaCyNLPPipelineModel = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
fetch(`/contributions/spacy-nlp-pipeline-models/${spaCyNLPPipelineModelId}/toggle-public-status`, {method: 'POST', headers: {Accept: 'application/json'}})
|
2023-01-11 13:29:47 +01:00
|
|
|
.then(
|
|
|
|
(response) => {
|
|
|
|
if (response.status === 403) {
|
|
|
|
app.flash('Forbidden', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
resolve(response);
|
|
|
|
},
|
|
|
|
(response) => {
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
);
|
2022-11-17 12:11:21 +01:00
|
|
|
});
|
|
|
|
}
|
2023-02-21 16:18:04 +01:00
|
|
|
|
2023-02-24 15:22:26 +01:00
|
|
|
static generateCorpusShareLinkRequest(corpusId, role, expiration) {
|
2023-02-21 16:18:04 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
2023-02-24 15:22:26 +01:00
|
|
|
const data = {role: role, expiration: expiration};
|
2023-02-21 16:18:04 +01:00
|
|
|
fetch(`/corpora/${corpusId}/generate-corpus-share-link`, {method: 'POST', headers: {Accept: 'text/plain'}, body: JSON.stringify(data)})
|
|
|
|
.then(
|
|
|
|
(response) => {
|
|
|
|
if (!response.ok) {
|
|
|
|
app.flash(`Something went wrong: ${response.status} ${response.statusText}`, 'error');
|
|
|
|
reject(response);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
return response.text();
|
|
|
|
},
|
|
|
|
(response) => {
|
|
|
|
// Something went wrong during the HTTP request
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(response);
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.then(
|
|
|
|
(corpusShareLink) => {resolve(corpusShareLink);},
|
|
|
|
(error) => {
|
|
|
|
// Something went wrong during ReadableStream processing
|
|
|
|
app.flash('Something went wrong', 'error');
|
|
|
|
reject(error);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
2022-09-02 13:07:30 +02:00
|
|
|
}
|