homebridge-econet-rheem 1.5.10 → 1.5.12-beta.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (73) hide show
  1. package/CHANGELOG.md +14 -2
  2. package/dist/homebridge/index.js.map +1 -0
  3. package/dist/{platform.d.ts → homebridge/platform.d.ts} +4 -1
  4. package/dist/homebridge/platform.js +98 -0
  5. package/dist/homebridge/platform.js.map +1 -0
  6. package/dist/homebridge/settings.js.map +1 -0
  7. package/dist/{accessories → homebridge}/thermostatAccessory.d.ts +1 -1
  8. package/dist/{accessories → homebridge}/thermostatAccessory.js +8 -6
  9. package/dist/homebridge/thermostatAccessory.js.map +1 -0
  10. package/dist/{accessories → homebridge}/waterHeaterAccessory.d.ts +1 -1
  11. package/dist/{accessories → homebridge}/waterHeaterAccessory.js +7 -5
  12. package/dist/homebridge/waterHeaterAccessory.js.map +1 -0
  13. package/dist/lang/en.d.ts +52 -0
  14. package/dist/lang/en.js +64 -0
  15. package/dist/lang/en.js.map +1 -0
  16. package/dist/model/api.d.ts +36 -0
  17. package/dist/model/api.js +367 -0
  18. package/dist/model/api.js.map +1 -0
  19. package/dist/model/auth.d.ts +11 -0
  20. package/dist/model/auth.js +52 -0
  21. package/dist/model/auth.js.map +1 -0
  22. package/dist/model/{enums.d.ts → constants.d.ts} +4 -0
  23. package/dist/model/{enums.js → constants.js} +6 -1
  24. package/dist/model/constants.js.map +1 -0
  25. package/dist/model/equipment.d.ts +9 -9
  26. package/dist/model/equipment.js +15 -29
  27. package/dist/model/equipment.js.map +1 -1
  28. package/dist/model/recoverySimulator.d.ts +2 -2
  29. package/dist/model/recoverySimulator.js +7 -8
  30. package/dist/model/recoverySimulator.js.map +1 -1
  31. package/dist/model/thermostat.d.ts +6 -6
  32. package/dist/model/thermostat.js +62 -70
  33. package/dist/model/thermostat.js.map +1 -1
  34. package/dist/model/types.d.ts +82 -0
  35. package/dist/model/types.js +12 -0
  36. package/dist/model/types.js.map +1 -0
  37. package/dist/model/waterHeater.d.ts +7 -5
  38. package/dist/model/waterHeater.js +34 -37
  39. package/dist/model/waterHeater.js.map +1 -1
  40. package/dist/tools/storage.d.ts +3 -0
  41. package/dist/tools/storage.js +22 -0
  42. package/dist/tools/storage.js.map +1 -0
  43. package/dist/tools/temperature.d.ts +3 -0
  44. package/dist/tools/temperature.js +14 -0
  45. package/dist/tools/temperature.js.map +1 -0
  46. package/dist/tools/time.d.ts +4 -0
  47. package/dist/tools/time.js +5 -0
  48. package/dist/tools/time.js.map +1 -0
  49. package/dist/tools/version.d.ts +1 -0
  50. package/dist/tools/version.js +18 -0
  51. package/dist/tools/version.js.map +1 -0
  52. package/example_responses/auth.json +45 -0
  53. package/example_responses/locations.json +1656 -0
  54. package/package.json +4 -2
  55. package/src/homebridge-ui/public/index.html +104 -0
  56. package/dist/accessories/thermostatAccessory.js.map +0 -1
  57. package/dist/accessories/waterHeaterAccessory.js.map +0 -1
  58. package/dist/index.js.map +0 -1
  59. package/dist/model/econet.d.ts +0 -48
  60. package/dist/model/econet.js +0 -319
  61. package/dist/model/econet.js.map +0 -1
  62. package/dist/model/enums.js.map +0 -1
  63. package/dist/model/utils.d.ts +0 -9
  64. package/dist/model/utils.js +0 -53
  65. package/dist/model/utils.js.map +0 -1
  66. package/dist/platform.js +0 -101
  67. package/dist/platform.js.map +0 -1
  68. package/dist/settings.js.map +0 -1
  69. package/notes +0 -7
  70. /package/dist/{index.d.ts → homebridge/index.d.ts} +0 -0
  71. /package/dist/{index.js → homebridge/index.js} +0 -0
  72. /package/dist/{settings.d.ts → homebridge/settings.d.ts} +0 -0
  73. /package/dist/{settings.js → homebridge/settings.js} +0 -0
@@ -0,0 +1,367 @@
1
+ import axios, { isAxiosError } from 'axios';
2
+ import mqtt from 'mqtt';
3
+ import { Auth } from './auth.js';
4
+ import { EquipmentType } from './constants.js';
5
+ import * as Types from './types.js';
6
+ import { Thermostat } from './thermostat.js';
7
+ import { WaterHeater } from './waterHeater.js';
8
+ import strings from '../lang/en.js';
9
+ import { safeGetItem, safeSetItem } from '../tools/storage.js';
10
+ import { MINUTE, SECOND } from '../tools/time.js';
11
+ const CLEAR_BLADE_SYSTEM_KEY = 'e2e699cb0bb0bbb88fc8858cb5a401';
12
+ const CLEAR_BLADE_SYSTEM_SECRET = 'E2E699CB0BE6C6FADDB1B0BC9A20';
13
+ const BASE_HEADERS = {
14
+ 'ClearBlade-SystemKey': CLEAR_BLADE_SYSTEM_KEY,
15
+ 'ClearBlade-SystemSecret': CLEAR_BLADE_SYSTEM_SECRET,
16
+ 'Content-Type': 'application/json; charset=UTF-8',
17
+ };
18
+ const HOST = 'rheem.clearblade.com';
19
+ const BASE_URL = `https://${HOST}/api/v/1`;
20
+ const AUTH_URL = `${BASE_URL}/user/auth`;
21
+ const LOCATIONS_URL = `${BASE_URL}/code/${CLEAR_BLADE_SYSTEM_KEY}/getUserDataForApp`;
22
+ const MQTT_URL = `mqtts://${HOST}:1884`;
23
+ const MQTT_TOPIC_REPORTED = 'user/%s/device/reported';
24
+ const MQTT_TOPIC_DESIRED = 'user/%s/device/desired';
25
+ const HTTP_TIMEOUT = 10 * SECOND;
26
+ const MQTT_KEEPALIVE = 90;
27
+ const DELAYS = [5 * SECOND, 15 * SECOND, MINUTE, 2 * MINUTE, 5 * MINUTE];
28
+ const IDLE_CONNECTION_TIMER_INTERVAL = 16 * MINUTE;
29
+ const HTTP_RETRY_CODES = [
30
+ 'ERR_NETWORK', // General network error in Axios
31
+ 'ETIMEDOUT', // Request timed out
32
+ 'ECONNREFUSED', // Connection refused by server
33
+ '429', // Too Many Requests (rate limit)
34
+ '500', // Internal Server Error
35
+ '502', // Bad Gateway
36
+ '503', // Service Unavailable
37
+ '504', // Gateway Timeout
38
+ ];
39
+ const RETRYABLE_CODES = [
40
+ 3, // MQTT: Server unavailable
41
+ 'ECONNREFUSED', 'ETIMEDOUT', 'ECONNRESET', 'ENOTFOUND',
42
+ 'EHOSTUNREACH', 'ENETUNREACH', 'EAI_AGAIN', 'EPIPE',
43
+ ];
44
+ export class EconetApi {
45
+ log;
46
+ email;
47
+ password;
48
+ storageFilePath;
49
+ verbose;
50
+ debugMQTT;
51
+ _auth;
52
+ retryIndex = 0;
53
+ equipments = new Map();
54
+ mqttClient = null;
55
+ shouldReconnect = false;
56
+ isReconnecting = false;
57
+ reconnectCount = 0;
58
+ idleMQTTTimer = null;
59
+ constructor(log, email, password, storageFilePath, verbose, debugMQTT) {
60
+ this.log = log;
61
+ this.email = email;
62
+ this.password = password;
63
+ this.storageFilePath = storageFilePath;
64
+ this.verbose = verbose;
65
+ this.debugMQTT = debugMQTT;
66
+ this.auth = Auth.load(this.storageFilePath, email);
67
+ }
68
+ static async connect(log, email, password, storageFilePath, verbose, debugMQTT) {
69
+ const api = new EconetApi(log, email, password, storageFilePath, verbose, debugMQTT);
70
+ let shouldContinue = true;
71
+ if (!api.auth) {
72
+ shouldContinue = await api.authenticate();
73
+ }
74
+ if (shouldContinue) {
75
+ await api.getLocations();
76
+ api.shouldReconnect = true;
77
+ api.mqttConnect(true);
78
+ }
79
+ return api;
80
+ }
81
+ teardown() {
82
+ this.shouldReconnect = false;
83
+ if (this.mqttClient) {
84
+ this.mqttClient.end(true);
85
+ this.mqttClient = null;
86
+ }
87
+ }
88
+ publish(payload, deviceId, serialNumber) {
89
+ if (!this.mqttClient || !this.mqttClient.connected) {
90
+ this.log.error(strings.clientNotConnected);
91
+ return;
92
+ }
93
+ if (!this.auth?.accountId) {
94
+ this.log.error(strings.authMissing);
95
+ return;
96
+ }
97
+ const dateTime = new Date().toISOString().replace(/\.\d{3}Z$/, '');
98
+ const transactionId = `ANDROID_${dateTime}`;
99
+ const data = {
100
+ transactionId,
101
+ device_name: deviceId,
102
+ serial_number: serialNumber,
103
+ ...payload,
104
+ };
105
+ const topic = MQTT_TOPIC_DESIRED.replace('%s', this.auth.accountId);
106
+ const message = JSON.stringify(data);
107
+ this.mqttClient.publish(topic, message);
108
+ if (this.verbose) {
109
+ this.log.info(strings.topicPublish, topic, message);
110
+ }
111
+ }
112
+ get auth() {
113
+ return this._auth ?? null;
114
+ }
115
+ set auth(value) {
116
+ this._auth = value;
117
+ if (this._auth) {
118
+ this._auth.save(this.storageFilePath, this.email);
119
+ }
120
+ }
121
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
122
+ async do(caller,
123
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
124
+ data, shouldRetry, url, ...parameters) {
125
+ parameters.forEach(param => {
126
+ url = url.replace('%s', param ?? '');
127
+ });
128
+ let config;
129
+ if (this.auth?.token) {
130
+ const headers = { ...BASE_HEADERS, 'ClearBlade-UserToken': this.auth?.token };
131
+ config = { headers: headers, timeout: HTTP_TIMEOUT };
132
+ }
133
+ else {
134
+ config = { headers: BASE_HEADERS, timeout: HTTP_TIMEOUT };
135
+ }
136
+ try {
137
+ let res;
138
+ if (data) {
139
+ res = await axios.post(url, data, config);
140
+ }
141
+ else {
142
+ res = await axios.get(url, config);
143
+ }
144
+ if (!res.data) {
145
+ this.logHTTP("debug" /* LogLevel.DEBUG */, caller, JSON.stringify(res.data));
146
+ throw new Error(strings.noDataReceived);
147
+ }
148
+ this.logIfVerbose(caller, res.data);
149
+ this.retryIndex = 0;
150
+ return res.data;
151
+ }
152
+ catch (err) {
153
+ if (shouldRetry) {
154
+ return this.retryIfPossible(err, caller, () => this.do(caller, data, shouldRetry, url, ...parameters));
155
+ }
156
+ else {
157
+ this.logHTTP("warn" /* LogLevel.WARN */, caller, err.message);
158
+ return null;
159
+ }
160
+ }
161
+ }
162
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
163
+ async retryIfPossible(err, caller, retry) {
164
+ if (!isAxiosError(err)) {
165
+ this.logHTTP("warn" /* LogLevel.WARN */, caller, err.message);
166
+ return null;
167
+ }
168
+ const errorCode = err.code || err.response?.status?.toString() || 'UNKNOWN';
169
+ if (!HTTP_RETRY_CODES.includes(errorCode) || this.retryIndex >= DELAYS.length) {
170
+ this.logHTTP("warn" /* LogLevel.WARN */, caller, err.message);
171
+ return null;
172
+ }
173
+ const reconnectDelay = DELAYS[Math.min(this.reconnectCount, DELAYS.length - 1)];
174
+ if (reconnectDelay <= MINUTE) {
175
+ this.log.warn(strings.httpRetrySeconds, reconnectDelay / SECOND);
176
+ }
177
+ else {
178
+ this.log.warn(strings.httpRetryMinutes, reconnectDelay / MINUTE);
179
+ }
180
+ await new Promise(resolve => setTimeout(resolve, reconnectDelay));
181
+ this.retryIndex += 1;
182
+ return await retry();
183
+ }
184
+ async authenticate() {
185
+ const data = { email: this.email, password: this.password };
186
+ const tokenData = await this.do(this.authenticate.name, data, true, AUTH_URL);
187
+ if (!tokenData) {
188
+ return false;
189
+ }
190
+ this.auth = new Auth(tokenData);
191
+ this.log.info(strings.authSuccess);
192
+ return true;
193
+ }
194
+ mqttConnect(isStartup = false) {
195
+ if (!this.equipments.size) {
196
+ return;
197
+ }
198
+ if (!this.auth?.token) {
199
+ this.log.error(strings.authMissing);
200
+ return;
201
+ }
202
+ const timeString = Date.now().toString().replace('.', '').slice(0, 13);
203
+ const clientId = `${this.email}${timeString}_android`;
204
+ const options = {
205
+ clientId,
206
+ username: this.auth.token,
207
+ password: CLEAR_BLADE_SYSTEM_KEY,
208
+ rejectUnauthorized: true,
209
+ keepalive: MQTT_KEEPALIVE,
210
+ reconnectPeriod: 0,
211
+ };
212
+ this.mqttClient = mqtt.connect(MQTT_URL, options);
213
+ this.mqttClient.on('connect', () => {
214
+ if (!this.mqttClient || !this.auth?.accountId) {
215
+ this.log.error(strings.connectionError);
216
+ return;
217
+ }
218
+ this.mqttClient.subscribe(MQTT_TOPIC_REPORTED.replace('%s', this.auth.accountId));
219
+ this.mqttClient.subscribe(MQTT_TOPIC_DESIRED.replace('%s', this.auth.accountId));
220
+ this.log.info(strings.connected);
221
+ if (isStartup) {
222
+ const randIndex = Math.floor(Math.random() * strings.welcomeMessages.length);
223
+ this.log.info(strings.setupComplete, strings.welcomeMessages[randIndex]);
224
+ }
225
+ });
226
+ this.mqttClient.on('message', (topic, message) => {
227
+ this.reconnectCount = 0;
228
+ this.resetIdleMQTTTimer();
229
+ try {
230
+ const data = JSON.parse(message.toString());
231
+ if (this.verbose) {
232
+ this.log.info(strings.topicUpdate, topic, JSON.stringify(data));
233
+ }
234
+ const serial = data.serial_number;
235
+ const equipment = this.equipments.get(serial);
236
+ if (equipment) {
237
+ equipment.updateFromMQTT(data);
238
+ if (this.debugMQTT) {
239
+ this.saveMQTT(data);
240
+ }
241
+ }
242
+ }
243
+ catch (e) {
244
+ this.log.error(strings.parseFailed, message.toString());
245
+ }
246
+ });
247
+ this.mqttClient.on('offline', () => {
248
+ this.log.debug(strings.clientOffline);
249
+ });
250
+ this.mqttClient.on('close', () => {
251
+ this.log.info(strings.connectionClosed);
252
+ this.mqttReconnect();
253
+ });
254
+ this.mqttClient.on('error', (err) => {
255
+ if (err.code !== undefined && RETRYABLE_CODES.includes(err.code)) {
256
+ if (this.verbose) {
257
+ this.log.error(strings.clientError, err);
258
+ }
259
+ this.mqttReconnect();
260
+ }
261
+ else {
262
+ this.log.error(strings.clientError, err);
263
+ }
264
+ });
265
+ }
266
+ resetIdleMQTTTimer() {
267
+ if (this.idleMQTTTimer) {
268
+ clearTimeout(this.idleMQTTTimer);
269
+ }
270
+ this.idleMQTTTimer = setTimeout(() => {
271
+ this.log.info(strings.idleConnection);
272
+ this.mqttReconnect();
273
+ }, IDLE_CONNECTION_TIMER_INTERVAL);
274
+ }
275
+ async mqttReconnect() {
276
+ if (!this.shouldReconnect || this.isReconnecting) {
277
+ return;
278
+ }
279
+ this.isReconnecting = true;
280
+ if (this.mqttClient) {
281
+ this.mqttClient.end(true);
282
+ this.mqttClient = null;
283
+ }
284
+ this.reconnectCount++;
285
+ if (this.reconnectCount % DELAYS.length === 0) {
286
+ try {
287
+ this.log.error(strings.unstableConnection);
288
+ this.log.info(strings.reauthenticate);
289
+ await this.authenticate();
290
+ }
291
+ catch (error) {
292
+ this.log.error(strings.reauthFailed, error);
293
+ }
294
+ }
295
+ const reconnectDelay = DELAYS[Math.min(this.reconnectCount, DELAYS.length - 1)];
296
+ if (reconnectDelay <= MINUTE) {
297
+ this.log.info(strings.reconnectInSeconds, reconnectDelay / SECOND);
298
+ }
299
+ else {
300
+ this.log.info(strings.reconnectInMinutes, reconnectDelay / MINUTE);
301
+ }
302
+ setTimeout(() => {
303
+ this.isReconnecting = false;
304
+ this.mqttConnect();
305
+ }, reconnectDelay);
306
+ }
307
+ async getLocations() {
308
+ const data = {
309
+ location_only: false,
310
+ type: 'com.econet.econetconsumerandroid',
311
+ version: '6.0.0-375-01b4870e',
312
+ };
313
+ const locationsData = await this.do(this.getLocations.name, data, true, LOCATIONS_URL);
314
+ if (!locationsData) {
315
+ return;
316
+ }
317
+ locationsData.results.locations.forEach(location => {
318
+ location.equiptments.forEach(equipmentData => {
319
+ let equipment = null;
320
+ switch (equipmentData.device_type) {
321
+ case EquipmentType.THERMOSTAT:
322
+ equipment = new Thermostat(this, equipmentData);
323
+ break;
324
+ case EquipmentType.WATER_HEATER:
325
+ equipment = new WaterHeater(this, equipmentData, this.storageFilePath);
326
+ break;
327
+ default:
328
+ this.log.error(strings.unsupportedEquipment, equipmentData.device_type);
329
+ }
330
+ if (equipment) {
331
+ this.equipments.set(equipment.serialNumber, equipment);
332
+ }
333
+ });
334
+ });
335
+ }
336
+ saveMQTT(data) {
337
+ const ignoreKeys = new Set(['transactionId', 'device_name', 'serial_number']);
338
+ for (const [key, value] of Object.entries(data)) {
339
+ if (ignoreKeys.has(key) || value.toString.length === 0) {
340
+ continue;
341
+ }
342
+ let valuesString = safeGetItem(this.storageFilePath, key);
343
+ let valuesArray = valuesString ? JSON.parse(valuesString) : [];
344
+ const valuesSet = new Set(valuesArray);
345
+ valuesSet.add(value);
346
+ valuesArray = Array.from(valuesSet);
347
+ valuesString = JSON.stringify(valuesArray);
348
+ safeSetItem(this.storageFilePath, key, valuesString);
349
+ }
350
+ }
351
+ logHTTP(level, caller, message) {
352
+ this.log.log(level, '[HTTP %s()] %s', caller, message);
353
+ }
354
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
355
+ logIfVerbose(caller, data) {
356
+ if (!this.verbose) {
357
+ return;
358
+ }
359
+ let message = JSON.stringify(data);
360
+ Types.SENSITIVE_KEYS.forEach(key => {
361
+ const regex = new RegExp(`"${key}"\\s*:\\s*(".*?"|\\d+|true|false|null)`, 'gi');
362
+ message = message.replace(regex, `"${key}": "${strings.redacted}"`);
363
+ });
364
+ this.logHTTP("info" /* LogLevel.INFO */, caller, message);
365
+ }
366
+ }
367
+ //# sourceMappingURL=api.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"api.js","sourceRoot":"","sources":["../../src/model/api.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import { TokenData } from './types.js';
2
+ export declare class Auth {
3
+ private readonly data;
4
+ constructor(data: TokenData);
5
+ get token(): string;
6
+ get userId(): string;
7
+ get accountId(): string;
8
+ private static digest;
9
+ save(filePath: string, encryptionKey: string): void;
10
+ static load(filePath: string, encryptionKey: string): Auth | null;
11
+ }
@@ -0,0 +1,52 @@
1
+ import crypto from 'crypto';
2
+ import { safeGetItem, safeSetItem } from '../tools/storage.js';
3
+ const STORAGE_AUTH_KEY = 'auth';
4
+ export class Auth {
5
+ data;
6
+ constructor(data) {
7
+ this.data = data;
8
+ }
9
+ get token() {
10
+ return this.data.user_token;
11
+ }
12
+ get userId() {
13
+ return this.data.user_id;
14
+ }
15
+ get accountId() {
16
+ return this.data.options.account_id;
17
+ }
18
+ static digest(encryptionKey) {
19
+ return crypto.createHash('sha256').update(encryptionKey).digest();
20
+ }
21
+ save(filePath, encryptionKey) {
22
+ const serailzed = JSON.stringify({
23
+ data: this.data,
24
+ });
25
+ const digest = Auth.digest(encryptionKey);
26
+ const iv = crypto.randomBytes(16);
27
+ const cipher = crypto.createCipheriv('aes-256-cbc', digest, iv);
28
+ const encrypted = Buffer.concat([cipher.update(serailzed, 'utf8'), cipher.final()]);
29
+ const final = iv.toString('hex') + ':' + encrypted.toString('hex');
30
+ safeSetItem(filePath, STORAGE_AUTH_KEY, final);
31
+ }
32
+ static load(filePath, encryptionKey) {
33
+ const final = safeGetItem(filePath, STORAGE_AUTH_KEY);
34
+ if (!final) {
35
+ return null;
36
+ }
37
+ try {
38
+ const digest = Auth.digest(encryptionKey);
39
+ const [ivHex, encryptedHex] = final.split(':');
40
+ const iv = Buffer.from(ivHex, 'hex');
41
+ const encrypted = Buffer.from(encryptedHex, 'hex');
42
+ const decipher = crypto.createDecipheriv('aes-256-cbc', digest, iv);
43
+ const decrypted = Buffer.concat([decipher.update(encrypted), decipher.final()]).toString('utf8');
44
+ const obj = JSON.parse(decrypted);
45
+ return new Auth(obj.data);
46
+ }
47
+ catch {
48
+ return null;
49
+ }
50
+ }
51
+ }
52
+ //# sourceMappingURL=auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.js","sourceRoot":"","sources":["../../src/model/auth.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,QAAQ,CAAC;AAI5B,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AAE/D,MAAM,gBAAgB,GAAG,MAAM,CAAC;AAEhC,MAAM,OAAO,IAAI;IAGI;IADnB,YACmB,IAAe;QAAf,SAAI,GAAJ,IAAI,CAAW;IAChC,CAAC;IAEH,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC;IAC9B,CAAC;IAED,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC;IAC3B,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC;IACtC,CAAC;IAEO,MAAM,CAAC,MAAM,CAAC,aAAqB;QACzC,OAAO,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,MAAM,EAAE,CAAC;IACpE,CAAC;IAED,IAAI,CAAC,QAAgB,EAAE,aAAqB;QAE1C,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;YAC/B,IAAI,EAAE,IAAI,CAAC,IAAI;SAChB,CAAC,CAAC;QAEH,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;QAC1C,MAAM,EAAE,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;QAClC,MAAM,MAAM,GAAG,MAAM,CAAC,cAAc,CAAC,aAAa,EAAE,MAAM,EAAE,EAAE,CAAC,CAAC;QAChE,MAAM,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,EAAE,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QACpF,MAAM,KAAK,GAAG,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,GAAG,GAAG,GAAG,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAEnE,WAAW,CAAC,QAAQ,EAAE,gBAAgB,EAAE,KAAK,CAAC,CAAC;IACjD,CAAC;IAED,MAAM,CAAC,IAAI,CAAC,QAAgB,EAAE,aAAqB;QAEjD,MAAM,KAAK,GAAG,WAAW,CAAC,QAAQ,EAAE,gBAAgB,CAAC,CAAC;QACtD,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,CAAC;YAEH,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;YAC1C,MAAM,CAAC,KAAK,EAAE,YAAY,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC/C,MAAM,EAAE,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;YACrC,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;YACnD,MAAM,QAAQ,GAAG,MAAM,CAAC,gBAAgB,CAAC,aAAa,EAAE,MAAM,EAAE,EAAE,CAAC,CAAC;YACpE,MAAM,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YAEjG,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAyC,CAAC;YAC1E,OAAO,IAAI,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAE5B,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;CACF"}
@@ -1,3 +1,7 @@
1
+ export declare enum EquipmentType {
2
+ THERMOSTAT = "HVAC",
3
+ WATER_HEATER = "WH"
4
+ }
1
5
  export declare enum ThermostatOperationMode {
2
6
  OFF = 1,
3
7
  HEATING = 2,
@@ -1,3 +1,8 @@
1
+ export var EquipmentType;
2
+ (function (EquipmentType) {
3
+ EquipmentType["THERMOSTAT"] = "HVAC";
4
+ EquipmentType["WATER_HEATER"] = "WH";
5
+ })(EquipmentType || (EquipmentType = {}));
1
6
  export var ThermostatOperationMode;
2
7
  (function (ThermostatOperationMode) {
3
8
  ThermostatOperationMode[ThermostatOperationMode["OFF"] = 1] = "OFF";
@@ -13,4 +18,4 @@ export var TemperatureUnits;
13
18
  TemperatureUnits[TemperatureUnits["FAHRENHEIT"] = 1] = "FAHRENHEIT";
14
19
  TemperatureUnits[TemperatureUnits["CELSIUS"] = 2] = "CELSIUS";
15
20
  })(TemperatureUnits || (TemperatureUnits = {}));
16
- //# sourceMappingURL=enums.js.map
21
+ //# sourceMappingURL=constants.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.js","sourceRoot":"","sources":["../../src/model/constants.ts"],"names":[],"mappings":"AAAA,MAAM,CAAN,IAAY,aAGX;AAHD,WAAY,aAAa;IACvB,oCAAmB,CAAA;IACnB,oCAAmB,CAAA;AACrB,CAAC,EAHW,aAAa,KAAb,aAAa,QAGxB;AAED,MAAM,CAAN,IAAY,uBAQX;AARD,WAAY,uBAAuB;IACjC,mEAAO,CAAA;IACP,2EAAW,CAAA;IACX,2EAAW,CAAA;IACX,qEAAQ,CAAA;IACR,6EAAY,CAAA;IACZ,yFAAkB,CAAA;IAClB,4EAAY,CAAA;AACd,CAAC,EARW,uBAAuB,KAAvB,uBAAuB,QAQlC;AAED,MAAM,CAAN,IAAY,gBAGX;AAHD,WAAY,gBAAgB;IAC1B,mEAAc,CAAA;IACd,6DAAW,CAAA;AACb,CAAC,EAHW,gBAAgB,KAAhB,gBAAgB,QAG3B"}
@@ -1,24 +1,24 @@
1
- import { EconetApi } from './econet.js';
2
- import { TemperatureUnits } from './enums.js';
1
+ import { EconetApi } from './api.js';
2
+ import { EquipmentType, TemperatureUnits } from './constants.js';
3
+ import { EquipmentData, MQTTData } from './types.js';
3
4
  export declare abstract class Equipment {
5
+ readonly api: EconetApi;
4
6
  private device_id?;
5
7
  private serial_number?;
6
8
  private device_name?;
7
- private alertCount;
9
+ private alert_count;
8
10
  private temp_units;
9
- private running;
10
- protected _api: EconetApi;
11
+ protected running: boolean;
11
12
  private _onUpdateCallback;
12
- constructor(api: EconetApi);
13
+ constructor(api: EconetApi, data: EquipmentData);
14
+ abstract get type(): EquipmentType;
13
15
  get deviceId(): string;
14
16
  get serialNumber(): string;
15
17
  get deviceName(): string;
16
18
  get hasAlert(): boolean;
17
19
  get units(): TemperatureUnits;
18
- protected abstract get runningKey(): string;
19
20
  get isRunning(): boolean;
20
21
  setOnUpdateCallback(callback: (serialNumber: string) => void): void;
21
22
  protected didUpdate(): void;
22
- protected updateFromREST(update: any): void;
23
- updateFromMQTT(update: any): void;
23
+ updateFromMQTT(update: MQTTData): void;
24
24
  }
@@ -1,15 +1,21 @@
1
- import { TemperatureUnits } from './enums.js';
1
+ import { TemperatureUnits } from './constants.js';
2
+ import strings from '../lang/en.js';
2
3
  export class Equipment {
4
+ api;
3
5
  device_id;
4
6
  serial_number;
5
7
  device_name;
6
- alertCount = 0;
8
+ alert_count = 0;
7
9
  temp_units = TemperatureUnits.CELSIUS;
8
10
  running = false;
9
- _api;
10
11
  _onUpdateCallback = null;
11
- constructor(api) {
12
- this._api = api;
12
+ constructor(api, data) {
13
+ this.api = api;
14
+ this.device_id = data.device_name;
15
+ this.serial_number = data.serial_number;
16
+ this.device_name = data['@NAME'].value;
17
+ this.alert_count = data['@ALERTCOUNT'];
18
+ this.temp_units = data['@SETPOINT'].constraints.units.includes('F') ? TemperatureUnits.FAHRENHEIT : TemperatureUnits.CELSIUS;
13
19
  }
14
20
  get deviceId() {
15
21
  return this.device_id || 'undefined';
@@ -21,7 +27,7 @@ export class Equipment {
21
27
  return this.device_name || 'undefined';
22
28
  }
23
29
  get hasAlert() {
24
- return this.alertCount > 0;
30
+ return this.alert_count > 0;
25
31
  }
26
32
  get units() {
27
33
  return this.temp_units;
@@ -37,30 +43,10 @@ export class Equipment {
37
43
  this._onUpdateCallback(this.serialNumber);
38
44
  }
39
45
  }
40
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
41
- updateFromREST(update) {
42
- this.device_id = update.device_name;
43
- this.serial_number = update.serial_number;
44
- this.device_name = update['@NAME'].value ?? '';
45
- if ('@ALERTCOUNT' in update) {
46
- this.alertCount = update['@ALERTCOUNT'] ?? 0;
47
- }
48
- if ('@SETPOINT' in update) {
49
- this.temp_units = update['@SETPOINT'].constraints.units.includes('F') ? TemperatureUnits.FAHRENHEIT : TemperatureUnits.CELSIUS;
50
- }
51
- if (this.runningKey in update) {
52
- this.running = update[this.runningKey].replace(/\s/g, '').length > 0;
53
- }
54
- }
55
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
56
46
  updateFromMQTT(update) {
57
- if ('@ALERTCOUNT' in update) {
58
- this.alertCount = update['@ALERTCOUNT'] ?? 0;
59
- this._api.log.debug(`${this.deviceName} alert count = ${this.alertCount}`);
60
- }
61
- if (this.runningKey in update) {
62
- this.running = update[this.runningKey].replace(/\s/g, '').length > 0;
63
- this._api.log.debug(`${this.deviceName} running = ${this.running}`);
47
+ if (update['@ALERTCOUNT']) {
48
+ this.alert_count = update['@ALERTCOUNT'];
49
+ this.api.log.debug(strings.alertCount, this.deviceName, this.alert_count);
64
50
  }
65
51
  }
66
52
  }
@@ -1 +1 @@
1
- {"version":3,"file":"equipment.js","sourceRoot":"","sources":["../../src/model/equipment.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"equipment.js","sourceRoot":"","sources":["../../src/model/equipment.ts"],"names":[],"mappings":"AACA,OAAO,EAAiB,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAGjE,OAAO,OAAO,MAAM,eAAe,CAAC;AAEpC,MAAM,OAAgB,SAAS;IAUR;IATb,SAAS,CAAiB;IAC1B,aAAa,CAAiB;IAC9B,WAAW,CAAiB;IAC5B,WAAW,GAAW,CAAC,CAAC;IACxB,UAAU,GAAG,gBAAgB,CAAC,OAAO,CAAC;IACpC,OAAO,GAAY,KAAK,CAAC;IAE3B,iBAAiB,GAA4C,IAAI,CAAC;IAE1E,YAAqB,GAAc,EAAE,IAAmB;QAAnC,QAAG,GAAH,GAAG,CAAW;QACjC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC;QAClC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC;QACxC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,KAAK,CAAC;QACvC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC;QACvC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,gBAAgB,CAAC,UAAU,CAAC,CAAC,CAAC,gBAAgB,CAAC,OAAO,CAAC;IAC/H,CAAC;IAID,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,SAAS,IAAI,WAAW,CAAC;IACvC,CAAC;IAED,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,aAAa,IAAI,WAAW,CAAC;IAC3C,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,WAAW,IAAI,WAAW,CAAC;IACzC,CAAC;IAED,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;IAC9B,CAAC;IAED,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,OAAO,CAAC;IACtB,CAAC;IAED,mBAAmB,CAAC,QAAwC;QAC1D,IAAI,CAAC,iBAAiB,GAAG,QAAQ,CAAC;IACpC,CAAC;IAES,SAAS;QACjB,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC5C,CAAC;IACH,CAAC;IAED,cAAc,CAAC,MAAgB;QAE7B,IAAI,MAAM,CAAC,aAAa,CAAC,EAAE,CAAC;YAC1B,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;YACzC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,EAAE,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;QAC5E,CAAC;IACH,CAAC;CACF"}
@@ -1,9 +1,9 @@
1
- import { EconetApi } from './econet.js';
1
+ import { EconetApi } from './api.js';
2
2
  import { WaterHeater } from './waterHeater.js';
3
3
  export declare class RecoverySimulator {
4
4
  private readonly api;
5
+ readonly waterHeater: WaterHeater;
5
6
  private readonly onUpdate;
6
- private readonly recoveryRatesFilePath;
7
7
  private readonly minRecoveryRate;
8
8
  private readonly defaultRecoveryRate;
9
9
  private recoveryRates;