2021-06-17 12:39:06 +01:00
|
|
|
import time from './time';
|
|
|
|
import Setting from './models/Setting';
|
2023-07-27 16:05:56 +01:00
|
|
|
import Logger from '@joplin/utils/Logger';
|
2021-06-17 12:39:06 +01:00
|
|
|
|
|
|
|
interface Task {
|
|
|
|
id: string;
|
2023-06-30 10:30:29 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2021-06-17 12:39:06 +01:00
|
|
|
callback: Function;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface TaskResult {
|
|
|
|
id: string;
|
|
|
|
result: any;
|
|
|
|
error?: Error;
|
|
|
|
}
|
|
|
|
|
|
|
|
export default class TaskQueue {
|
|
|
|
|
|
|
|
private waitingTasks_: Task[] = [];
|
|
|
|
private processingTasks_: Record<string, Task> = {};
|
|
|
|
private processingQueue_ = false;
|
|
|
|
private stopping_ = false;
|
|
|
|
private results_: Record<string, TaskResult> = {};
|
|
|
|
private name_: string;
|
|
|
|
private logger_: Logger;
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public constructor(name: string, logger: Logger = null) {
|
2019-06-07 23:20:08 +01:00
|
|
|
this.name_ = name;
|
2021-06-17 12:39:06 +01:00
|
|
|
this.logger_ = logger ? logger : new Logger();
|
2019-06-07 23:20:08 +01:00
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public concurrency() {
|
2019-06-07 23:20:08 +01:00
|
|
|
return Setting.value('sync.maxConcurrentConnections');
|
|
|
|
}
|
|
|
|
|
2023-06-30 10:30:29 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2023-03-06 14:22:01 +00:00
|
|
|
public push(id: string, callback: Function) {
|
2019-06-07 23:20:08 +01:00
|
|
|
if (this.stopping_) throw new Error('Cannot push task when queue is stopping');
|
|
|
|
|
|
|
|
this.waitingTasks_.push({
|
|
|
|
id: id,
|
|
|
|
callback: callback,
|
|
|
|
});
|
|
|
|
this.processQueue_();
|
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
private processQueue_() {
|
2019-06-07 23:20:08 +01:00
|
|
|
if (this.processingQueue_ || this.stopping_) return;
|
|
|
|
|
|
|
|
this.processingQueue_ = true;
|
|
|
|
|
2021-06-17 12:39:06 +01:00
|
|
|
const completeTask = (task: Task, result: any, error: Error) => {
|
2019-06-07 23:20:08 +01:00
|
|
|
delete this.processingTasks_[task.id];
|
|
|
|
|
2021-06-17 12:39:06 +01:00
|
|
|
const r: TaskResult = {
|
2019-06-07 23:20:08 +01:00
|
|
|
id: task.id,
|
|
|
|
result: result,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (error) r.error = error;
|
|
|
|
|
|
|
|
this.results_[task.id] = r;
|
|
|
|
|
|
|
|
this.processQueue_();
|
2019-07-29 15:43:53 +02:00
|
|
|
};
|
2019-06-07 23:20:08 +01:00
|
|
|
|
|
|
|
while (this.waitingTasks_.length > 0 && Object.keys(this.processingTasks_).length < this.concurrency()) {
|
|
|
|
if (this.stopping_) break;
|
|
|
|
|
|
|
|
const task = this.waitingTasks_.splice(0, 1)[0];
|
|
|
|
this.processingTasks_[task.id] = task;
|
|
|
|
|
2019-07-29 15:43:53 +02:00
|
|
|
task
|
|
|
|
.callback()
|
2022-09-30 17:23:14 +01:00
|
|
|
// eslint-disable-next-line promise/prefer-await-to-then -- Old code before rule was applied
|
2021-06-17 12:39:06 +01:00
|
|
|
.then((result: any) => {
|
2019-07-29 15:43:53 +02:00
|
|
|
completeTask(task, result, null);
|
|
|
|
})
|
2022-09-30 17:23:14 +01:00
|
|
|
// eslint-disable-next-line promise/prefer-await-to-then -- Old code before rule was applied
|
2021-06-17 12:39:06 +01:00
|
|
|
.catch((error: Error) => {
|
2019-07-29 15:43:53 +02:00
|
|
|
if (!error) error = new Error('Unknown error');
|
|
|
|
completeTask(task, null, error);
|
|
|
|
});
|
2019-06-07 23:20:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
this.processingQueue_ = false;
|
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public isWaiting(taskId: string) {
|
2020-05-21 09:14:33 +01:00
|
|
|
return this.waitingTasks_.find(task => task.id === taskId);
|
2019-06-07 23:20:08 +01:00
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public isProcessing(taskId: string) {
|
2019-06-07 23:20:08 +01:00
|
|
|
return taskId in this.processingTasks_;
|
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public isDone(taskId: string) {
|
2019-06-07 23:20:08 +01:00
|
|
|
return taskId in this.results_;
|
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public async waitForAll() {
|
2021-06-17 12:39:06 +01:00
|
|
|
return new Promise((resolve) => {
|
|
|
|
const checkIID = setInterval(() => {
|
|
|
|
if (this.waitingTasks_.length) return;
|
|
|
|
if (this.processingTasks_.length) return;
|
|
|
|
clearInterval(checkIID);
|
|
|
|
resolve(null);
|
|
|
|
}, 100);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public taskExists(taskId: string) {
|
2021-06-17 12:39:06 +01:00
|
|
|
return this.isWaiting(taskId) || this.isProcessing(taskId) || this.isDone(taskId);
|
|
|
|
}
|
2019-06-07 23:20:08 +01:00
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public taskResult(taskId: string) {
|
2021-06-17 12:39:06 +01:00
|
|
|
if (!this.taskExists(taskId)) throw new Error(`No such task: ${taskId}`);
|
|
|
|
return this.results_[taskId];
|
|
|
|
}
|
2019-06-07 23:20:08 +01:00
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public async waitForResult(taskId: string) {
|
2021-06-17 12:39:06 +01:00
|
|
|
if (!this.taskExists(taskId)) throw new Error(`No such task: ${taskId}`);
|
|
|
|
|
|
|
|
while (true) {
|
2019-06-07 23:20:08 +01:00
|
|
|
const task = this.results_[taskId];
|
|
|
|
if (task) return task;
|
|
|
|
await time.sleep(0.1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public async stop() {
|
2019-06-07 23:20:08 +01:00
|
|
|
this.stopping_ = true;
|
|
|
|
|
2019-09-19 22:51:18 +01:00
|
|
|
this.logger_.info(`TaskQueue.stop: ${this.name_}: waiting for tasks to complete: ${Object.keys(this.processingTasks_).length}`);
|
2019-06-07 23:20:08 +01:00
|
|
|
|
|
|
|
// In general it's not a big issue if some tasks are still running because
|
|
|
|
// it won't call anything unexpected in caller code, since the caller has
|
|
|
|
// to explicitely retrieve the results
|
|
|
|
const startTime = Date.now();
|
|
|
|
while (Object.keys(this.processingTasks_).length) {
|
|
|
|
await time.sleep(0.1);
|
|
|
|
if (Date.now() - startTime >= 30000) {
|
2019-09-19 22:51:18 +01:00
|
|
|
this.logger_.warn(`TaskQueue.stop: ${this.name_}: timed out waiting for task to complete`);
|
2019-06-07 23:20:08 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-19 22:51:18 +01:00
|
|
|
this.logger_.info(`TaskQueue.stop: ${this.name_}: Done, waited for ${Date.now() - startTime}`);
|
2019-06-07 23:20:08 +01:00
|
|
|
}
|
|
|
|
|
2023-03-06 14:22:01 +00:00
|
|
|
public isStopping() {
|
2019-06-07 23:20:08 +01:00
|
|
|
return this.stopping_;
|
|
|
|
}
|
|
|
|
}
|