import { dirname, extname, join, parse } from 'node:path'
import process, { exit } from 'node:process'
import { fileURLToPath } from 'node:url'
+import { isMainThread } from 'node:worker_threads'
import type { Worker } from 'worker_threads'
import chalk from 'chalk'
ChargingStationWorkerMessageEvents,
ConfigurationSection,
ProcedureName,
- type StationTemplateUrl,
type Statistics,
type StorageConfiguration,
type UIServerConfiguration,
isAsyncFunction,
isNotEmptyArray,
logPrefix,
- logger
+ logger,
+ max
} from '../utils/index.js'
import { type WorkerAbstract, WorkerFactory } from '../worker/index.js'
private workerImplementation?: WorkerAbstract<ChargingStationWorkerData>
private readonly uiServer?: AbstractUIServer
private storage?: Storage
- private readonly chargingStationsByTemplate: Map<string, { configured: number, started: number }>
+ private readonly chargingStationsByTemplate: Map<
+ string,
+ { configured: number, started: number, lastIndex: number }
+ >
+
private readonly version: string = version
private initializedCounters: boolean
private started: boolean
this.stopping = false
this.chargingStationsByTemplate = new Map<
string,
- {
- configured: number
- started: number
- }
+ { configured: number, started: number, lastIndex: number }
>()
this.uiServer = UIServerFactory.getUIServerImplementation(
Configuration.getConfigurationSection<UIServerConfiguration>(ConfigurationSection.uiServer)
this.initializedCounters = false
this.initializeCounters()
Configuration.configurationChangeCallback = async () => {
- await Bootstrap.getInstance().restart(false)
+ if (isMainThread) {
+ await Bootstrap.getInstance().restart()
+ }
}
}
)
}
+ public getLastIndex (templateName: string): number {
+ return this.chargingStationsByTemplate.get(templateName)?.lastIndex ?? 0
+ }
+
private get numberOfStartedChargingStations (): number {
return [...this.chargingStationsByTemplate.values()].reduce(
(accumulator, value) => accumulator + value.started,
this.chargingStationsByTemplate.get(parse(stationTemplateUrl.file).name)
?.configured ?? stationTemplateUrl.numberOfStations
for (let index = 1; index <= nbStations; index++) {
- await this.startChargingStation(index, stationTemplateUrl)
+ await this.addChargingStation(index, stationTemplateUrl.file)
}
} catch (error) {
console.error(
chalk.green(
`Charging stations simulator ${
this.version
- } started with ${this.numberOfConfiguredChargingStations} charging station(s) from ${this.numberOfChargingStationTemplates} configured charging station template(s) and ${
+ } started with ${this.numberOfConfiguredChargingStations} configured charging station(s) from ${this.numberOfChargingStationTemplates} charging station template(s) and ${
Configuration.workerDynamicPoolInUse() ? `${workerConfiguration.poolMinSize}/` : ''
}${this.workerImplementation?.size}${
Configuration.workerPoolInUse() ? `/${workerConfiguration.poolMaxSize}` : ''
}
}
- public async stop (stopChargingStations = true): Promise<void> {
+ public async stop (): Promise<void> {
if (this.started) {
if (!this.stopping) {
this.stopping = true
- if (stopChargingStations) {
- await this.uiServer?.sendInternalRequest(
- this.uiServer.buildProtocolRequest(
- generateUUID(),
- ProcedureName.STOP_CHARGING_STATION,
- Constants.EMPTY_FROZEN_OBJECT
- )
+ await this.uiServer?.sendInternalRequest(
+ this.uiServer.buildProtocolRequest(
+ generateUUID(),
+ ProcedureName.STOP_CHARGING_STATION,
+ Constants.EMPTY_FROZEN_OBJECT
)
- try {
- await this.waitChargingStationsStopped()
- } catch (error) {
- console.error(chalk.red('Error while waiting for charging stations to stop: '), error)
- }
+ )
+ try {
+ await this.waitChargingStationsStopped()
+ } catch (error) {
+ console.error(chalk.red('Error while waiting for charging stations to stop: '), error)
}
await this.workerImplementation?.stop()
delete this.workerImplementation
}
}
- public async restart (stopChargingStations?: boolean): Promise<void> {
- await this.stop(stopChargingStations)
+ private async restart (): Promise<void> {
+ await this.stop()
Configuration.getConfigurationSection<UIServerConfiguration>(ConfigurationSection.uiServer)
.enabled === false && this.uiServer?.stop()
this.initializedCounters = false
}
private initializeWorkerImplementation (workerConfiguration: WorkerConfiguration): void {
+ if (!isMainThread) {
+ return
+ }
let elementsPerWorker: number | undefined
switch (workerConfiguration.elementsPerWorker) {
case 'auto':
data.stationInfo.chargingStationId
} (hashId: ${data.stationInfo.hashId}) started (${
this.numberOfStartedChargingStations
- } started from ${this.numberOfConfiguredChargingStations})`
+ } started from ${this.numberOfConfiguredChargingStations} configured charging station(s))`
)
}
data.stationInfo.chargingStationId
} (hashId: ${data.stationInfo.hashId}) stopped (${
this.numberOfStartedChargingStations
- } started from ${this.numberOfConfiguredChargingStations})`
+ } started from ${this.numberOfConfiguredChargingStations} configured charging station(s))`
)
}
const templateName = parse(stationTemplateUrl.file).name
this.chargingStationsByTemplate.set(templateName, {
configured: stationTemplateUrl.numberOfStations,
- started: 0
+ started: 0,
+ lastIndex: 0
})
this.uiServer?.chargingStationTemplates.add(templateName)
}
)
exit(exitCodes.missingChargingStationsConfiguration)
}
- if (this.numberOfConfiguredChargingStations === 0) {
+ if (
+ this.numberOfConfiguredChargingStations === 0 &&
+ Configuration.getConfigurationSection<UIServerConfiguration>(ConfigurationSection.uiServer)
+ .enabled === true
+ ) {
console.error(
chalk.red(
- "'stationTemplateUrls' has no charging station enabled, please check your configuration"
+ "'stationTemplateUrls' has no charging station enabled and UI server is disabled, please check your configuration"
)
)
exit(exitCodes.noChargingStationTemplates)
}
}
- private async startChargingStation (
- index: number,
- stationTemplateUrl: StationTemplateUrl
- ): Promise<void> {
+ public async addChargingStation (index: number, stationTemplateFile: string): Promise<void> {
await this.workerImplementation?.addElement({
index,
templateFile: join(
dirname(fileURLToPath(import.meta.url)),
'assets',
'station-templates',
- stationTemplateUrl.file
+ stationTemplateFile
)
})
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
+ this.chargingStationsByTemplate.get(parse(stationTemplateFile).name)!.lastIndex = max(
+ index,
+ this.chargingStationsByTemplate.get(parse(stationTemplateFile).name)?.lastIndex ?? -Infinity
+ )
}
private gracefulShutdown (): void {
this.requestHandlers = new Map<ProcedureName, ProtocolRequestHandler>([
[ProcedureName.LIST_TEMPLATES, this.handleListTemplates.bind(this)],
[ProcedureName.LIST_CHARGING_STATIONS, this.handleListChargingStations.bind(this)],
+ [ProcedureName.ADD_CHARGING_STATIONS, this.handleAddChargingStations.bind(this)],
[ProcedureName.START_SIMULATOR, this.handleStartSimulator.bind(this)],
[ProcedureName.STOP_SIMULATOR, this.handleStopSimulator.bind(this)]
])
errorMessage: (error as OCPPError).message,
errorStack: (error as OCPPError).stack,
errorDetails: (error as OCPPError).details
- }
+ } satisfies ResponsePayload
}
if (responsePayload != null) {
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
} satisfies ResponsePayload
}
+ private async handleAddChargingStations (
+ messageId?: string,
+ procedureName?: ProcedureName,
+ requestPayload?: RequestPayload
+ ): Promise<ResponsePayload> {
+ const { template, numberOfStations } = requestPayload as {
+ template: string
+ numberOfStations: number
+ }
+ if (!this.uiServer.chargingStationTemplates.has(template)) {
+ return {
+ status: ResponseStatus.FAILURE,
+ errorMessage: `Template '${template}' not found`
+ } satisfies ResponsePayload
+ }
+ for (let i = 0; i < numberOfStations; i++) {
+ try {
+ await Bootstrap.getInstance().addChargingStation(
+ Bootstrap.getInstance().getLastIndex(template) + 1,
+ `${template}.json`
+ )
+ } catch (error) {
+ return {
+ status: ResponseStatus.FAILURE,
+ errorMessage: (error as Error).message,
+ errorStack: (error as Error).stack
+ } satisfies ResponsePayload
+ }
+ }
+ return {
+ status: ResponseStatus.SUCCESS
+ }
+ }
+
private async handleStartSimulator (): Promise<ResponsePayload> {
try {
await Bootstrap.getInstance().start()