refactor(simulator): remove unneeded type casting
[e-mobility-charging-stations-simulator.git] / src / charging-station / ocpp / 2.0 / OCPP20RequestService.ts
1 // Partial Copyright Jerome Benoit. 2021-2023. All Rights Reserved.
2
3 import type { JSONSchemaType } from 'ajv';
4
5 import type { ChargingStation } from '../../../charging-station';
6 import { OCPPError } from '../../../exception';
7 import {
8 ErrorType,
9 type JsonObject,
10 type JsonType,
11 type OCPP20BootNotificationRequest,
12 type OCPP20HeartbeatRequest,
13 OCPP20RequestCommand,
14 type OCPP20StatusNotificationRequest,
15 OCPPVersion,
16 type RequestParams,
17 } from '../../../types';
18 import { Utils } from '../../../utils';
19 import {
20 OCPP20ServiceUtils,
21 OCPPConstants,
22 OCPPRequestService,
23 type OCPPResponseService,
24 } from '../internal';
25
26 const moduleName = 'OCPP20RequestService';
27
28 export class OCPP20RequestService extends OCPPRequestService {
29 protected jsonSchemas: Map<OCPP20RequestCommand, JSONSchemaType<JsonObject>>;
30
31 public constructor(ocppResponseService: OCPPResponseService) {
32 if (new.target?.name === moduleName) {
33 throw new TypeError(`Cannot construct ${new.target?.name} instances directly`);
34 }
35 super(OCPPVersion.VERSION_20, ocppResponseService);
36 this.jsonSchemas = new Map<OCPP20RequestCommand, JSONSchemaType<JsonObject>>([
37 [
38 OCPP20RequestCommand.BOOT_NOTIFICATION,
39 OCPP20ServiceUtils.parseJsonSchemaFile<OCPP20BootNotificationRequest>(
40 '../../../assets/json-schemas/ocpp/2.0/BootNotificationRequest.json',
41 moduleName,
42 'constructor'
43 ),
44 ],
45 [
46 OCPP20RequestCommand.HEARTBEAT,
47 OCPP20ServiceUtils.parseJsonSchemaFile<OCPP20HeartbeatRequest>(
48 '../../../assets/json-schemas/ocpp/2.0/HeartbeatRequest.json',
49 moduleName,
50 'constructor'
51 ),
52 ],
53 [
54 OCPP20RequestCommand.STATUS_NOTIFICATION,
55 OCPP20ServiceUtils.parseJsonSchemaFile<OCPP20StatusNotificationRequest>(
56 '../../../assets/json-schemas/ocpp/2.0/StatusNotificationRequest.json',
57 moduleName,
58 'constructor'
59 ),
60 ],
61 ]);
62 this.buildRequestPayload.bind(this);
63 }
64
65 public async requestHandler<RequestType extends JsonType, ResponseType extends JsonType>(
66 chargingStation: ChargingStation,
67 commandName: OCPP20RequestCommand,
68 commandParams?: JsonType,
69 params?: RequestParams
70 ): Promise<ResponseType> {
71 if (OCPP20ServiceUtils.isRequestCommandSupported(chargingStation, commandName) === true) {
72 return (await this.sendMessage(
73 chargingStation,
74 Utils.generateUUID(),
75 this.buildRequestPayload<RequestType>(chargingStation, commandName, commandParams),
76 commandName,
77 params
78 )) as ResponseType;
79 }
80 // OCPPError usage here is debatable: it's an error in the OCPP stack but not targeted to sendError().
81 throw new OCPPError(
82 ErrorType.NOT_SUPPORTED,
83 `Unsupported OCPP command '${commandName}'`,
84 commandName,
85 commandParams
86 );
87 }
88
89 private buildRequestPayload<Request extends JsonType>(
90 chargingStation: ChargingStation,
91 commandName: OCPP20RequestCommand,
92 commandParams?: JsonType
93 ): Request {
94 commandParams = commandParams as JsonObject;
95 switch (commandName) {
96 case OCPP20RequestCommand.BOOT_NOTIFICATION:
97 return commandParams as unknown as Request;
98 case OCPP20RequestCommand.HEARTBEAT:
99 return OCPPConstants.OCPP_RESPONSE_EMPTY as unknown as Request;
100 case OCPP20RequestCommand.STATUS_NOTIFICATION:
101 return {
102 timestamp: new Date(),
103 ...commandParams,
104 } as unknown as Request;
105 default:
106 // OCPPError usage here is debatable: it's an error in the OCPP stack but not targeted to sendError().
107 throw new OCPPError(
108 ErrorType.NOT_SUPPORTED,
109 // eslint-disable-next-line @typescript-eslint/restrict-template-expressions
110 `Unsupported OCPP command '${commandName}'`,
111 commandName,
112 commandParams
113 );
114 }
115 }
116 }