Switch to poolifier worker threads pool implementation.
[e-mobility-charging-stations-simulator.git] / src / worker / WorkerSet.ts
CommitLineData
a4624c96 1import { Worker, threadId } from 'worker_threads';
c045d9a9
JB
2import { WorkerData, WorkerEvents, WorkerSetElement } from '../types/Worker';
3
6013bc53
JB
4import Constants from '../utils/Constants';
5import Utils from '../utils/Utils';
144cabe0 6import Wrk from './Wrk';
6013bc53
JB
7
8export default class WorkerSet extends Wrk {
9 public maxElementsPerWorker: number;
c045d9a9 10 private workers: Set<WorkerSetElement>;
6013bc53
JB
11
12 /**
13 * Create a new `WorkerSet`.
14 *
15 * @param {string} workerScript
16 * @param {number} maxElementsPerWorker
17 */
18 constructor(workerScript: string, maxElementsPerWorker = 1) {
19 super(workerScript);
c045d9a9 20 this.workers = new Set<WorkerSetElement>();
6013bc53 21 this.maxElementsPerWorker = maxElementsPerWorker;
6013bc53
JB
22 }
23
24 get size(): number {
25 return this.workers.size;
26 }
27
28 /**
29 *
30 * @return {Promise<void>}
31 * @public
32 */
33 public async addElement(elementData: WorkerData): Promise<void> {
34 if (!this.workers) {
a4624c96 35 throw Error('Cannot add a WorkerSet element: workers\' set does not exist');
6013bc53 36 }
c045d9a9 37 if (this.getLastWorkerSetElement().numberOfWorkerElements >= this.maxElementsPerWorker) {
a4624c96 38 this.startWorker();
6013bc53 39 // Start worker sequentially to optimize memory at startup
a4624c96 40 await Utils.sleep(Constants.START_WORKER_DELAY);
6013bc53 41 }
3e1416d8 42 this.getLastWorker().postMessage({ id: WorkerEvents.START_WORKER_ELEMENT, workerData: elementData });
c045d9a9 43 this.getLastWorkerSetElement().numberOfWorkerElements++;
6013bc53
JB
44 }
45
46 /**
47 *
48 * @return {Promise<void>}
49 * @public
50 */
51 public async start(): Promise<void> {
a4624c96 52 this.startWorker();
6013bc53
JB
53 // Start worker sequentially to optimize memory at startup
54 await Utils.sleep(Constants.START_WORKER_DELAY);
55 }
56
57 /**
58 *
59 * @return {Promise}
60 * @private
61 */
a4624c96
JB
62 private startWorker(): void {
63 const worker = new Worker(this.workerScript);
64 worker.on('message', () => { });
65 worker.on('error', () => { });
66 worker.on('exit', (code) => {
67 if (code !== 0) {
68 console.error(`Worker ${threadId} stopped with exit code ${code}`);
69 }
6013bc53 70 });
a4624c96 71 this.workers.add({ worker, numberOfWorkerElements: 0 });
6013bc53
JB
72 }
73
c045d9a9
JB
74 private getLastWorkerSetElement(): WorkerSetElement {
75 let workerSetElement: WorkerSetElement;
6013bc53 76 // eslint-disable-next-line no-empty
c045d9a9
JB
77 for (workerSetElement of this.workers) { }
78 return workerSetElement;
79 }
80
81 private getLastWorker(): Worker {
82 return this.getLastWorkerSetElement().worker;
6013bc53
JB
83 }
84}