Refine TS and linter configuration
[e-mobility-charging-stations-simulator.git] / src / ui / web / src / composable / UIClient.ts
CommitLineData
6c1761d4
JB
1import type { JsonType } from '@/type/JsonType';
2import { ProcedureName, ResponseStatus } from '@/type/UIProtocol';
3import type { ProtocolResponse, ResponsePayload } from '@/type/UIProtocol';
4
32de5a57
LM
5import Utils from './Utils';
6import config from '@/assets/config';
7import { v4 as uuidv4 } from 'uuid';
8
9type ResponseHandler = {
1f7fa4de 10 procedureName: ProcedureName;
32de5a57
LM
11 resolve: (value: ResponsePayload | PromiseLike<ResponsePayload>) => void;
12 reject: (reason?: any) => void;
32de5a57
LM
13};
14
15export default class UIClient {
16 private static _instance: UIClient | null = null;
17
5a010bf0 18 private _ws!: WebSocket;
32de5a57
LM
19 private _responseHandlers: Map<string, ResponseHandler>;
20
21 private constructor() {
5a010bf0 22 this.openWS();
32de5a57 23 this._responseHandlers = new Map<string, ResponseHandler>();
32de5a57
LM
24 }
25
26 public static get instance() {
27 if (UIClient._instance === null) {
28 UIClient._instance = new UIClient();
29 }
30 return UIClient._instance;
31 }
32
16fe3949 33 public registerWSonOpenListener(listener: (this: WebSocket, ev: Event) => void) {
32de5a57
LM
34 this._ws.addEventListener('open', listener);
35 }
36
5a010bf0
JB
37 public async startSimulator(): Promise<ResponsePayload> {
38 return this.sendRequest(ProcedureName.START_SIMULATOR, {});
39 }
40
41 public async stopSimulator(): Promise<ResponsePayload> {
42 return this.sendRequest(ProcedureName.STOP_SIMULATOR, {});
43 }
32de5a57 44
5a010bf0 45 public async listChargingStations(): Promise<ResponsePayload> {
32de5a57
LM
46 return this.sendRequest(ProcedureName.LIST_CHARGING_STATIONS, {});
47 }
48
8fc2e5cc
JB
49 public async startChargingStation(hashId: string): Promise<ResponsePayload> {
50 return this.sendRequest(ProcedureName.START_CHARGING_STATION, { hashId });
51 }
52
53 public async stopChargingStation(hashId: string): Promise<ResponsePayload> {
54 return this.sendRequest(ProcedureName.STOP_CHARGING_STATION, { hashId });
55 }
56
57 public async openConnection(hashId: string): Promise<ResponsePayload> {
58 return this.sendRequest(ProcedureName.OPEN_CONNECTION, {
59 hashId,
60 });
61 }
62
63 public async closeConnection(hashId: string): Promise<ResponsePayload> {
64 return this.sendRequest(ProcedureName.CLOSE_CONNECTION, {
65 hashId,
66 });
67 }
68
32de5a57
LM
69 public async startTransaction(
70 hashId: string,
71 connectorId: number,
5a010bf0 72 idTag: string | undefined
32de5a57 73 ): Promise<ResponsePayload> {
32de5a57
LM
74 return this.sendRequest(ProcedureName.START_TRANSACTION, {
75 hashId,
76 connectorId,
77 idTag,
78 });
79 }
80
5a010bf0
JB
81 public async stopTransaction(
82 hashId: string,
83 transactionId: number | undefined
84 ): Promise<ResponsePayload> {
32de5a57
LM
85 return this.sendRequest(ProcedureName.STOP_TRANSACTION, {
86 hashId,
87 transactionId,
88 });
89 }
90
5a010bf0
JB
91 private openWS(): void {
92 this._ws = new WebSocket(
93 `ws://${config.emobility.host}:${config.emobility.port}`,
94 config.emobility.protocol
95 );
96 this._ws.onmessage = this.responseHandler.bind(this);
a745e412
JB
97 this._ws.onerror = (error) => {
98 console.error('WebSocket error: ', error);
99 };
5a010bf0
JB
100 }
101
32de5a57
LM
102 private setResponseHandler(
103 id: string,
1f7fa4de 104 procedureName: ProcedureName,
32de5a57 105 resolve: (value: ResponsePayload | PromiseLike<ResponsePayload>) => void,
1f7fa4de 106 reject: (reason?: any) => void
32de5a57
LM
107 ): void {
108 this._responseHandlers.set(id, { resolve, reject, procedureName });
109 }
110
111 private getResponseHandler(id: string): ResponseHandler | undefined {
112 return this._responseHandlers.get(id);
113 }
114
1f7fa4de
JB
115 private deleteResponseHandler(id: string): boolean {
116 return this._responseHandlers.delete(id);
117 }
118
32de5a57
LM
119 private async sendRequest(command: ProcedureName, data: JsonType): Promise<ResponsePayload> {
120 let uuid: string;
121 return Utils.promiseWithTimeout(
122 new Promise((resolve, reject) => {
123 uuid = uuidv4();
124 const msg = JSON.stringify([uuid, command, data]);
125
5a010bf0
JB
126 if (this._ws.readyState !== WebSocket.OPEN) {
127 this.openWS();
128 }
129 if (this._ws.readyState === WebSocket.OPEN) {
32de5a57
LM
130 this._ws.send(msg);
131 } else {
132 throw new Error(`Send request ${command} message: connection not opened`);
133 }
134
1f7fa4de 135 this.setResponseHandler(uuid, command, resolve, reject);
32de5a57
LM
136 }),
137 60 * 1000,
138 Error(`Send request ${command} message timeout`),
139 () => {
140 this._responseHandlers.delete(uuid);
141 }
142 );
143 }
144
97f0a1a5 145 private responseHandler(messageEvent: MessageEvent<string>): void {
32de5a57
LM
146 const data = JSON.parse(messageEvent.data) as ProtocolResponse;
147
53e5fd67
JB
148 if (Array.isArray(data) === false) {
149 throw new Error('Response not an array: ' + JSON.stringify(data, null, 2));
32de5a57
LM
150 }
151
152 const [uuid, response] = data;
153
154 if (this._responseHandlers.has(uuid) === true) {
155 switch (response.status) {
156 case ResponseStatus.SUCCESS:
157 this.getResponseHandler(uuid)?.resolve(response);
158 break;
159 case ResponseStatus.FAILURE:
160 this.getResponseHandler(uuid)?.reject(response);
161 break;
162 default:
a745e412 163 console.error(`Response status not supported: ${response.status}`);
32de5a57 164 }
1f7fa4de 165 this.deleteResponseHandler(uuid);
32de5a57
LM
166 } else {
167 throw new Error('Not a response to a request: ' + JSON.stringify(data, null, 2));
168 }
169 }
170}