1 import Configuration from
'../utils/Configuration';
2 import { StationWorkerData
} from
'../types/Worker';
3 import Utils from
'../utils/Utils';
4 import WorkerAbstract from
'../worker/WorkerAbstract';
5 import WorkerFactory from
'../worker/WorkerFactory';
6 import { isMainThread
} from
'worker_threads';
7 import path from
'path';
9 export default class Bootstrap
{
10 private static instance
: Bootstrap
;
11 private started
: boolean;
12 private workerScript
: string;
13 private workerImplementationInstance
: WorkerAbstract
;
15 private constructor() {
17 this.workerScript
= path
.join(path
.resolve(__dirname
, '../'), 'charging-station', 'StationWorker.js');
18 Configuration
.setConfigurationChangeCallback(async () => this.restart());
21 public static getInstance(): Bootstrap
{
22 if (!Bootstrap
.instance
) {
23 Bootstrap
.instance
= new Bootstrap();
25 return Bootstrap
.instance
;
28 public async start(): Promise
<void> {
29 if (isMainThread
&& !this.started
) {
31 let numStationsTotal
= 0;
32 await this.getWorkerImplementationInstance().start();
33 // Start ChargingStation object in worker thread
34 if (Configuration
.getStationTemplateURLs()) {
35 for (const stationURL
of Configuration
.getStationTemplateURLs()) {
37 const nbStations
= stationURL
.numberOfStations
? stationURL
.numberOfStations
: 0;
38 for (let index
= 1; index
<= nbStations
; index
++) {
39 const workerData
: StationWorkerData
= {
41 templateFile
: path
.join(path
.resolve(__dirname
, '../'), 'assets', 'station-templates', path
.basename(stationURL
.file
))
43 await this.getWorkerImplementationInstance().addElement(workerData
);
47 console
.error('Charging station start with template file ' + stationURL
.file
+ ' error ', error
);
51 console
.log('No stationTemplateURLs defined in configuration, exiting');
53 if (numStationsTotal
=== 0) {
54 console
.log('No charging station template enabled in configuration, exiting');
56 console
.log(`Charging station simulator started with ${numStationsTotal.toString()} charging station(s) and ${Utils.workerDynamicPoolInUse() ? `${Configuration.getWorkerPoolMinSize().toString()}/` : ''}${this.getWorkerImplementationInstance().size}${Utils.workerPoolInUse() ? `/${Configuration.getWorkerPoolMaxSize().toString()}
` : ''} worker(s) concurrently running in '${Configuration.getWorkerProcess()}' mode${this.getWorkerImplementationInstance().maxElementsPerWorker ? ` (${this.getWorkerImplementationInstance().maxElementsPerWorker} charging station(s) per worker)` : ''}`);
60 console.error('Bootstrap start error ', error);
65 public async stop(): Promise<void> {
66 if (isMainThread && this.started) {
67 await this.getWorkerImplementationInstance().stop();
68 // Nullify to force worker implementation instance creation
69 this.workerImplementationInstance = null;
74 public async restart(): Promise<void> {
79 private getWorkerImplementationInstance(): WorkerAbstract {
80 if (!this.workerImplementationInstance) {
81 this.workerImplementationInstance = WorkerFactory.getWorkerImplementation<StationWorkerData>(this.workerScript, Configuration.getWorkerProcess(), {
82 poolMaxSize: Configuration.getWorkerPoolMaxSize(),
83 poolMinSize: Configuration.getWorkerPoolMinSize(),
84 elementsPerWorker: Configuration.getChargingStationsPerWorker()
87 return this.workerImplementationInstance;