1 // Partial Copyright Jerome Benoit. 2021. All Rights Reserved.
4 import path from
'path';
5 import { fileURLToPath
} from
'url';
7 import { JSONSchemaType
} from
'ajv';
9 import OCPPError from
'../../../exception/OCPPError';
10 import { JsonObject
, JsonType
} from
'../../../types/JsonType';
11 import { OCPP16MeterValuesRequest
} from
'../../../types/ocpp/1.6/MeterValues';
13 DiagnosticsStatusNotificationRequest
,
14 OCPP16BootNotificationRequest
,
15 OCPP16HeartbeatRequest
,
17 OCPP16StatusNotificationRequest
,
18 } from
'../../../types/ocpp/1.6/Requests';
20 OCPP16AuthorizeRequest
,
21 OCPP16StartTransactionRequest
,
22 OCPP16StopTransactionRequest
,
23 } from
'../../../types/ocpp/1.6/Transaction';
24 import { ErrorType
} from
'../../../types/ocpp/ErrorType';
25 import { RequestParams
} from
'../../../types/ocpp/Requests';
26 import Constants from
'../../../utils/Constants';
27 import logger from
'../../../utils/Logger';
28 import Utils from
'../../../utils/Utils';
29 import type ChargingStation from
'../../ChargingStation';
30 import { ChargingStationUtils
} from
'../../ChargingStationUtils';
31 import OCPPRequestService from
'../OCPPRequestService';
32 import type OCPPResponseService from
'../OCPPResponseService';
33 import { OCPP16ServiceUtils
} from
'./OCPP16ServiceUtils';
35 const moduleName
= 'OCPP16RequestService';
37 export default class OCPP16RequestService
extends OCPPRequestService
{
38 private jsonSchemas
: Map
<OCPP16RequestCommand
, JSONSchemaType
<JsonObject
>>;
40 public constructor(ocppResponseService
: OCPPResponseService
) {
41 if (new.target
?.name
=== moduleName
) {
42 throw new TypeError(`Cannot construct ${new.target?.name} instances directly`);
44 super(ocppResponseService
);
45 this.jsonSchemas
= new Map
<OCPP16RequestCommand
, JSONSchemaType
<JsonObject
>>([
47 OCPP16RequestCommand
.AUTHORIZE
,
51 path
.dirname(fileURLToPath(import.meta
.url
)),
52 '../../../assets/json-schemas/ocpp/1.6/Authorize.json'
56 ) as JSONSchemaType
<OCPP16AuthorizeRequest
>,
59 OCPP16RequestCommand
.BOOT_NOTIFICATION
,
63 path
.dirname(fileURLToPath(import.meta
.url
)),
64 '../../../assets/json-schemas/ocpp/1.6/BootNotification.json'
68 ) as JSONSchemaType
<OCPP16BootNotificationRequest
>,
71 OCPP16RequestCommand
.DIAGNOSTICS_STATUS_NOTIFICATION
,
75 path
.dirname(fileURLToPath(import.meta
.url
)),
76 '../../../assets/json-schemas/ocpp/1.6/DiagnosticsStatusNotification.json'
80 ) as JSONSchemaType
<DiagnosticsStatusNotificationRequest
>,
83 OCPP16RequestCommand
.HEARTBEAT
,
87 path
.dirname(fileURLToPath(import.meta
.url
)),
88 '../../../assets/json-schemas/ocpp/1.6/Heartbeat.json'
92 ) as JSONSchemaType
<OCPP16HeartbeatRequest
>,
95 OCPP16RequestCommand
.METER_VALUES
,
99 path
.dirname(fileURLToPath(import.meta
.url
)),
100 '../../../assets/json-schemas/ocpp/1.6/MeterValues.json'
104 ) as JSONSchemaType
<OCPP16MeterValuesRequest
>,
107 OCPP16RequestCommand
.STATUS_NOTIFICATION
,
111 path
.dirname(fileURLToPath(import.meta
.url
)),
112 '../../../assets/json-schemas/ocpp/1.6/StatusNotification.json'
116 ) as JSONSchemaType
<OCPP16StatusNotificationRequest
>,
119 OCPP16RequestCommand
.START_TRANSACTION
,
123 path
.dirname(fileURLToPath(import.meta
.url
)),
124 '../../../assets/json-schemas/ocpp/1.6/StartTransaction.json'
128 ) as JSONSchemaType
<OCPP16StartTransactionRequest
>,
131 OCPP16RequestCommand
.STOP_TRANSACTION
,
135 path
.dirname(fileURLToPath(import.meta
.url
)),
136 '../../../assets/json-schemas/ocpp/1.6/StopTransaction.json'
140 ) as JSONSchemaType
<OCPP16StopTransactionRequest
>,
145 public async requestHandler
<Request
extends JsonType
, Response
extends JsonType
>(
146 chargingStation
: ChargingStation
,
147 commandName
: OCPP16RequestCommand
,
148 commandParams
?: JsonType
,
149 params
?: RequestParams
150 ): Promise
<Response
> {
151 if (ChargingStationUtils
.isRequestCommandSupported(commandName
, chargingStation
)) {
152 const requestPayload
= this.buildRequestPayload
<Request
>(
157 if (this.jsonSchemas
.has(commandName
)) {
158 this.validateRequestPayload(
161 this.jsonSchemas
.get(commandName
),
166 `${chargingStation.logPrefix()} ${moduleName}.requestHandler: No JSON schema found for command ${commandName} PDU validation`
169 return (await this.sendMessage(
171 Utils
.generateUUID(),
175 )) as unknown
as Response
;
178 ErrorType
.NOT_SUPPORTED
,
179 `${moduleName}.requestHandler: Unsupported OCPP command '${commandName}'`,
185 private buildRequestPayload
<Request
extends JsonType
>(
186 chargingStation
: ChargingStation
,
187 commandName
: OCPP16RequestCommand
,
188 commandParams
?: JsonType
190 let connectorId
: number;
191 commandParams
= commandParams
as JsonObject
;
192 switch (commandName
) {
193 case OCPP16RequestCommand
.AUTHORIZE
:
195 ...(!Utils
.isUndefined(commandParams
?.idTag
)
196 ? { idTag
: commandParams
.idTag
}
197 : { idTag
: Constants
.DEFAULT_IDTAG
}),
198 } as unknown
as Request
;
199 case OCPP16RequestCommand
.BOOT_NOTIFICATION
:
201 chargePointModel
: commandParams
?.chargePointModel
,
202 chargePointVendor
: commandParams
?.chargePointVendor
,
203 ...(!Utils
.isUndefined(commandParams
?.chargeBoxSerialNumber
) && {
204 chargeBoxSerialNumber
: commandParams
.chargeBoxSerialNumber
,
206 ...(!Utils
.isUndefined(commandParams
?.chargePointSerialNumber
) && {
207 chargePointSerialNumber
: commandParams
.chargePointSerialNumber
,
209 ...(!Utils
.isUndefined(commandParams
?.firmwareVersion
) && {
210 firmwareVersion
: commandParams
.firmwareVersion
,
212 ...(!Utils
.isUndefined(commandParams
?.iccid
) && { iccid
: commandParams
.iccid
}),
213 ...(!Utils
.isUndefined(commandParams
?.imsi
) && { imsi
: commandParams
.imsi
}),
214 ...(!Utils
.isUndefined(commandParams
?.meterSerialNumber
) && {
215 meterSerialNumber
: commandParams
.meterSerialNumber
,
217 ...(!Utils
.isUndefined(commandParams
?.meterType
) && {
218 meterType
: commandParams
.meterType
,
220 } as unknown
as Request
;
221 case OCPP16RequestCommand
.DIAGNOSTICS_STATUS_NOTIFICATION
:
223 status: commandParams
?.diagnosticsStatus
,
224 } as unknown
as Request
;
225 case OCPP16RequestCommand
.HEARTBEAT
:
226 return {} as unknown
as Request
;
227 case OCPP16RequestCommand
.METER_VALUES
:
229 connectorId
: commandParams
?.connectorId
,
230 transactionId
: commandParams
?.transactionId
,
231 meterValue
: commandParams
?.meterValue
,
232 } as unknown
as Request
;
233 case OCPP16RequestCommand
.STATUS_NOTIFICATION
:
235 connectorId
: commandParams
?.connectorId
,
236 status: commandParams
?.status,
237 errorCode
: commandParams
?.errorCode
,
238 } as unknown
as Request
;
239 case OCPP16RequestCommand
.START_TRANSACTION
:
241 connectorId
: commandParams
?.connectorId
,
242 ...(!Utils
.isUndefined(commandParams
?.idTag
)
243 ? { idTag
: commandParams
?.idTag
}
244 : { idTag
: Constants
.DEFAULT_IDTAG
}),
245 meterStart
: chargingStation
.getEnergyActiveImportRegisterByConnectorId(
246 commandParams
?.connectorId
as number
248 timestamp
: new Date().toISOString(),
249 } as unknown
as Request
;
250 case OCPP16RequestCommand
.STOP_TRANSACTION
:
251 connectorId
= chargingStation
.getConnectorIdByTransactionId(
252 commandParams
?.transactionId
as number
255 transactionId
: commandParams
?.transactionId
,
256 ...(!Utils
.isUndefined(commandParams
?.idTag
) && { idTag
: commandParams
.idTag
}),
257 meterStop
: commandParams
?.meterStop
,
258 timestamp
: new Date().toISOString(),
259 ...(commandParams
?.reason
&& { reason
: commandParams
.reason
}),
260 ...(chargingStation
.getTransactionDataMeterValues() && {
261 transactionData
: OCPP16ServiceUtils
.buildTransactionDataMeterValues(
262 chargingStation
.getConnectorStatus(connectorId
).transactionBeginMeterValue
,
263 OCPP16ServiceUtils
.buildTransactionEndMeterValue(
266 commandParams
?.meterStop
as number
270 } as unknown
as Request
;
273 ErrorType
.NOT_SUPPORTED
,
274 // eslint-disable-next-line @typescript-eslint/restrict-template-expressions
275 `${moduleName}.buildRequestPayload: Unsupported OCPP command '${commandName}'`,