zigbee-herdsman 6.0.1 → 6.0.3

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 (260) hide show
  1. package/CHANGELOG.md +16 -0
  2. package/dist/adapter/ezsp/driver/uart.js +1 -1
  3. package/dist/adapter/ezsp/driver/uart.js.map +1 -1
  4. package/dist/adapter/z-stack/adapter/zStackAdapter.js +4 -4
  5. package/dist/adapter/z-stack/adapter/zStackAdapter.js.map +1 -1
  6. package/dist/adapter/zigate/adapter/zigateAdapter.js +4 -4
  7. package/dist/adapter/zigate/adapter/zigateAdapter.js.map +1 -1
  8. package/dist/controller/model/device.d.ts.map +1 -1
  9. package/dist/controller/model/device.js +1 -0
  10. package/dist/controller/model/device.js.map +1 -1
  11. package/package.json +14 -6
  12. package/.github/ISSUE_TEMPLATE/config.yml +0 -5
  13. package/.github/dependabot.yml +0 -22
  14. package/.github/workflows/ci.yml +0 -64
  15. package/.github/workflows/release-please.yml +0 -18
  16. package/.github/workflows/stale.yml +0 -20
  17. package/.github/workflows/typedoc.yaml +0 -47
  18. package/.release-please-manifest.json +0 -3
  19. package/.vscode/extensions.json +0 -3
  20. package/.vscode/settings.json +0 -11
  21. package/biome.json +0 -98
  22. package/examples/join-and-log.js +0 -24
  23. package/release-please-config.json +0 -9
  24. package/src/adapter/adapter.ts +0 -189
  25. package/src/adapter/adapterDiscovery.ts +0 -666
  26. package/src/adapter/const.ts +0 -12
  27. package/src/adapter/deconz/adapter/deconzAdapter.ts +0 -877
  28. package/src/adapter/deconz/driver/constants.ts +0 -246
  29. package/src/adapter/deconz/driver/driver.ts +0 -1540
  30. package/src/adapter/deconz/driver/frame.ts +0 -11
  31. package/src/adapter/deconz/driver/frameParser.ts +0 -753
  32. package/src/adapter/deconz/driver/parser.ts +0 -45
  33. package/src/adapter/deconz/driver/writer.ts +0 -22
  34. package/src/adapter/deconz/types.d.ts +0 -13
  35. package/src/adapter/ember/adapter/emberAdapter.ts +0 -2265
  36. package/src/adapter/ember/adapter/endpoints.ts +0 -86
  37. package/src/adapter/ember/adapter/oneWaitress.ts +0 -324
  38. package/src/adapter/ember/adapter/tokensManager.ts +0 -782
  39. package/src/adapter/ember/consts.ts +0 -178
  40. package/src/adapter/ember/enums.ts +0 -1746
  41. package/src/adapter/ember/ezsp/buffalo.ts +0 -1392
  42. package/src/adapter/ember/ezsp/consts.ts +0 -148
  43. package/src/adapter/ember/ezsp/enums.ts +0 -1114
  44. package/src/adapter/ember/ezsp/ezsp.ts +0 -9061
  45. package/src/adapter/ember/ezspError.ts +0 -10
  46. package/src/adapter/ember/types.ts +0 -866
  47. package/src/adapter/ember/uart/ash.ts +0 -1960
  48. package/src/adapter/ember/uart/consts.ts +0 -109
  49. package/src/adapter/ember/uart/enums.ts +0 -192
  50. package/src/adapter/ember/uart/parser.ts +0 -48
  51. package/src/adapter/ember/uart/queues.ts +0 -247
  52. package/src/adapter/ember/uart/writer.ts +0 -53
  53. package/src/adapter/ember/utils/initters.ts +0 -58
  54. package/src/adapter/ember/utils/math.ts +0 -73
  55. package/src/adapter/events.ts +0 -21
  56. package/src/adapter/ezsp/adapter/backup.ts +0 -109
  57. package/src/adapter/ezsp/adapter/ezspAdapter.ts +0 -614
  58. package/src/adapter/ezsp/driver/commands.ts +0 -2497
  59. package/src/adapter/ezsp/driver/consts.ts +0 -11
  60. package/src/adapter/ezsp/driver/driver.ts +0 -1002
  61. package/src/adapter/ezsp/driver/ezsp.ts +0 -802
  62. package/src/adapter/ezsp/driver/frame.ts +0 -101
  63. package/src/adapter/ezsp/driver/index.ts +0 -4
  64. package/src/adapter/ezsp/driver/multicast.ts +0 -78
  65. package/src/adapter/ezsp/driver/parser.ts +0 -81
  66. package/src/adapter/ezsp/driver/types/basic.ts +0 -201
  67. package/src/adapter/ezsp/driver/types/index.ts +0 -239
  68. package/src/adapter/ezsp/driver/types/named.ts +0 -2330
  69. package/src/adapter/ezsp/driver/types/struct.ts +0 -844
  70. package/src/adapter/ezsp/driver/uart.ts +0 -460
  71. package/src/adapter/ezsp/driver/utils/crc16ccitt.ts +0 -44
  72. package/src/adapter/ezsp/driver/utils/index.ts +0 -32
  73. package/src/adapter/ezsp/driver/writer.ts +0 -64
  74. package/src/adapter/index.ts +0 -3
  75. package/src/adapter/serialPort.ts +0 -58
  76. package/src/adapter/socketPortUtils.ts +0 -16
  77. package/src/adapter/tstype.ts +0 -78
  78. package/src/adapter/z-stack/adapter/adapter-backup.ts +0 -519
  79. package/src/adapter/z-stack/adapter/adapter-nv-memory.ts +0 -457
  80. package/src/adapter/z-stack/adapter/endpoints.ts +0 -57
  81. package/src/adapter/z-stack/adapter/manager.ts +0 -543
  82. package/src/adapter/z-stack/adapter/tstype.ts +0 -6
  83. package/src/adapter/z-stack/adapter/zStackAdapter.ts +0 -1190
  84. package/src/adapter/z-stack/constants/af.ts +0 -27
  85. package/src/adapter/z-stack/constants/common.ts +0 -285
  86. package/src/adapter/z-stack/constants/dbg.ts +0 -23
  87. package/src/adapter/z-stack/constants/index.ts +0 -11
  88. package/src/adapter/z-stack/constants/mac.ts +0 -128
  89. package/src/adapter/z-stack/constants/sapi.ts +0 -25
  90. package/src/adapter/z-stack/constants/sys.ts +0 -72
  91. package/src/adapter/z-stack/constants/util.ts +0 -82
  92. package/src/adapter/z-stack/constants/utils.ts +0 -14
  93. package/src/adapter/z-stack/constants/zdo.ts +0 -103
  94. package/src/adapter/z-stack/models/startup-options.ts +0 -13
  95. package/src/adapter/z-stack/structs/entries/address-manager-entry.ts +0 -44
  96. package/src/adapter/z-stack/structs/entries/address-manager-table.ts +0 -19
  97. package/src/adapter/z-stack/structs/entries/aps-link-key-data-entry.ts +0 -12
  98. package/src/adapter/z-stack/structs/entries/aps-link-key-data-table.ts +0 -21
  99. package/src/adapter/z-stack/structs/entries/aps-tc-link-key-entry.ts +0 -19
  100. package/src/adapter/z-stack/structs/entries/aps-tc-link-key-table.ts +0 -21
  101. package/src/adapter/z-stack/structs/entries/channel-list.ts +0 -8
  102. package/src/adapter/z-stack/structs/entries/has-configured.ts +0 -16
  103. package/src/adapter/z-stack/structs/entries/index.ts +0 -16
  104. package/src/adapter/z-stack/structs/entries/nib.ts +0 -66
  105. package/src/adapter/z-stack/structs/entries/nwk-key-descriptor.ts +0 -15
  106. package/src/adapter/z-stack/structs/entries/nwk-key.ts +0 -13
  107. package/src/adapter/z-stack/structs/entries/nwk-pan-id.ts +0 -8
  108. package/src/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.ts +0 -20
  109. package/src/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.ts +0 -19
  110. package/src/adapter/z-stack/structs/entries/security-manager-entry.ts +0 -33
  111. package/src/adapter/z-stack/structs/entries/security-manager-table.ts +0 -22
  112. package/src/adapter/z-stack/structs/index.ts +0 -4
  113. package/src/adapter/z-stack/structs/serializable-memory-object.ts +0 -14
  114. package/src/adapter/z-stack/structs/struct.ts +0 -367
  115. package/src/adapter/z-stack/structs/table.ts +0 -198
  116. package/src/adapter/z-stack/unpi/constants.ts +0 -33
  117. package/src/adapter/z-stack/unpi/frame.ts +0 -62
  118. package/src/adapter/z-stack/unpi/index.ts +0 -4
  119. package/src/adapter/z-stack/unpi/parser.ts +0 -56
  120. package/src/adapter/z-stack/unpi/writer.ts +0 -21
  121. package/src/adapter/z-stack/utils/channel-list.ts +0 -40
  122. package/src/adapter/z-stack/utils/index.ts +0 -2
  123. package/src/adapter/z-stack/utils/network-options.ts +0 -26
  124. package/src/adapter/z-stack/znp/buffaloZnp.ts +0 -175
  125. package/src/adapter/z-stack/znp/definition.ts +0 -2713
  126. package/src/adapter/z-stack/znp/index.ts +0 -2
  127. package/src/adapter/z-stack/znp/parameterType.ts +0 -22
  128. package/src/adapter/z-stack/znp/tstype.ts +0 -44
  129. package/src/adapter/z-stack/znp/utils.ts +0 -10
  130. package/src/adapter/z-stack/znp/znp.ts +0 -342
  131. package/src/adapter/z-stack/znp/zpiObject.ts +0 -148
  132. package/src/adapter/zboss/adapter/zbossAdapter.ts +0 -526
  133. package/src/adapter/zboss/commands.ts +0 -1184
  134. package/src/adapter/zboss/consts.ts +0 -9
  135. package/src/adapter/zboss/driver.ts +0 -422
  136. package/src/adapter/zboss/enums.ts +0 -360
  137. package/src/adapter/zboss/frame.ts +0 -227
  138. package/src/adapter/zboss/reader.ts +0 -65
  139. package/src/adapter/zboss/types.ts +0 -0
  140. package/src/adapter/zboss/uart.ts +0 -428
  141. package/src/adapter/zboss/utils.ts +0 -58
  142. package/src/adapter/zboss/writer.ts +0 -49
  143. package/src/adapter/zigate/adapter/patchZdoBuffaloBE.ts +0 -27
  144. package/src/adapter/zigate/adapter/zigateAdapter.ts +0 -618
  145. package/src/adapter/zigate/driver/LICENSE +0 -17
  146. package/src/adapter/zigate/driver/buffaloZiGate.ts +0 -212
  147. package/src/adapter/zigate/driver/commandType.ts +0 -418
  148. package/src/adapter/zigate/driver/constants.ts +0 -150
  149. package/src/adapter/zigate/driver/frame.ts +0 -197
  150. package/src/adapter/zigate/driver/messageType.ts +0 -287
  151. package/src/adapter/zigate/driver/parameterType.ts +0 -32
  152. package/src/adapter/zigate/driver/ziGateObject.ts +0 -146
  153. package/src/adapter/zigate/driver/zigate.ts +0 -423
  154. package/src/adapter/zoh/adapter/utils.ts +0 -27
  155. package/src/adapter/zoh/adapter/zohAdapter.ts +0 -838
  156. package/src/buffalo/buffalo.ts +0 -342
  157. package/src/buffalo/index.ts +0 -1
  158. package/src/controller/controller.ts +0 -1022
  159. package/src/controller/database.ts +0 -124
  160. package/src/controller/events.ts +0 -52
  161. package/src/controller/greenPower.ts +0 -603
  162. package/src/controller/helpers/index.ts +0 -1
  163. package/src/controller/helpers/installCodes.ts +0 -107
  164. package/src/controller/helpers/request.ts +0 -96
  165. package/src/controller/helpers/requestQueue.ts +0 -125
  166. package/src/controller/helpers/zclFrameConverter.ts +0 -47
  167. package/src/controller/helpers/zclTransactionSequenceNumber.ts +0 -19
  168. package/src/controller/index.ts +0 -6
  169. package/src/controller/model/device.ts +0 -1248
  170. package/src/controller/model/endpoint.ts +0 -1105
  171. package/src/controller/model/entity.ts +0 -23
  172. package/src/controller/model/group.ts +0 -424
  173. package/src/controller/model/index.ts +0 -5
  174. package/src/controller/model/zigbeeEntity.ts +0 -30
  175. package/src/controller/touchlink.ts +0 -189
  176. package/src/controller/tstype.ts +0 -274
  177. package/src/index.ts +0 -12
  178. package/src/models/backup-storage-legacy.ts +0 -48
  179. package/src/models/backup-storage-unified.ts +0 -47
  180. package/src/models/backup.ts +0 -37
  181. package/src/models/index.ts +0 -5
  182. package/src/models/network-options.ts +0 -11
  183. package/src/utils/backup.ts +0 -152
  184. package/src/utils/index.ts +0 -5
  185. package/src/utils/logger.ts +0 -20
  186. package/src/utils/patchBigIntSerialization.ts +0 -8
  187. package/src/utils/queue.ts +0 -76
  188. package/src/utils/types.d.ts +0 -3
  189. package/src/utils/utils.ts +0 -19
  190. package/src/utils/wait.ts +0 -5
  191. package/src/utils/waitress.ts +0 -96
  192. package/src/zspec/consts.ts +0 -84
  193. package/src/zspec/enums.ts +0 -22
  194. package/src/zspec/index.ts +0 -3
  195. package/src/zspec/tstypes.ts +0 -18
  196. package/src/zspec/utils.ts +0 -247
  197. package/src/zspec/zcl/buffaloZcl.ts +0 -1220
  198. package/src/zspec/zcl/definition/cluster.ts +0 -5915
  199. package/src/zspec/zcl/definition/clusters-typegen.ts +0 -588
  200. package/src/zspec/zcl/definition/clusters-types.ts +0 -7331
  201. package/src/zspec/zcl/definition/consts.ts +0 -24
  202. package/src/zspec/zcl/definition/enums.ts +0 -203
  203. package/src/zspec/zcl/definition/foundation.ts +0 -329
  204. package/src/zspec/zcl/definition/manufacturerCode.ts +0 -729
  205. package/src/zspec/zcl/definition/status.ts +0 -69
  206. package/src/zspec/zcl/definition/tstype.ts +0 -377
  207. package/src/zspec/zcl/index.ts +0 -11
  208. package/src/zspec/zcl/utils.ts +0 -321
  209. package/src/zspec/zcl/zclFrame.ts +0 -356
  210. package/src/zspec/zcl/zclHeader.ts +0 -102
  211. package/src/zspec/zcl/zclStatusError.ts +0 -10
  212. package/src/zspec/zdo/buffaloZdo.ts +0 -2336
  213. package/src/zspec/zdo/definition/clusters.ts +0 -722
  214. package/src/zspec/zdo/definition/consts.ts +0 -16
  215. package/src/zspec/zdo/definition/enums.ts +0 -99
  216. package/src/zspec/zdo/definition/status.ts +0 -105
  217. package/src/zspec/zdo/definition/tstypes.ts +0 -1062
  218. package/src/zspec/zdo/index.ts +0 -7
  219. package/src/zspec/zdo/utils.ts +0 -76
  220. package/src/zspec/zdo/zdoStatusError.ts +0 -10
  221. package/test/adapter/adapter.test.ts +0 -1062
  222. package/test/adapter/ember/ash.test.ts +0 -337
  223. package/test/adapter/ember/consts.ts +0 -131
  224. package/test/adapter/ember/emberAdapter.test.ts +0 -3449
  225. package/test/adapter/ember/ezsp.test.ts +0 -385
  226. package/test/adapter/ember/ezspBuffalo.test.ts +0 -93
  227. package/test/adapter/ember/ezspError.test.ts +0 -12
  228. package/test/adapter/ember/math.test.ts +0 -206
  229. package/test/adapter/ezsp/frame.test.ts +0 -30
  230. package/test/adapter/ezsp/uart.test.ts +0 -181
  231. package/test/adapter/z-stack/adapter.test.ts +0 -3984
  232. package/test/adapter/z-stack/constants.test.ts +0 -33
  233. package/test/adapter/z-stack/structs.test.ts +0 -115
  234. package/test/adapter/z-stack/unpi.test.ts +0 -213
  235. package/test/adapter/z-stack/znp.test.ts +0 -1284
  236. package/test/adapter/zboss/fixZdoResponse.test.ts +0 -179
  237. package/test/adapter/zigate/patchZdoBuffaloBE.test.ts +0 -81
  238. package/test/adapter/zigate/zdo.test.ts +0 -187
  239. package/test/adapter/zoh/utils.test.ts +0 -36
  240. package/test/adapter/zoh/zohAdapter.test.ts +0 -1307
  241. package/test/buffalo.test.ts +0 -431
  242. package/test/controller.bench.ts +0 -193
  243. package/test/controller.test.ts +0 -8702
  244. package/test/greenpower.test.ts +0 -1408
  245. package/test/mockAdapters.ts +0 -65
  246. package/test/mockDevices.ts +0 -598
  247. package/test/requests.bench.ts +0 -206
  248. package/test/testUtils.ts +0 -20
  249. package/test/tsconfig.json +0 -9
  250. package/test/utils/math.ts +0 -19
  251. package/test/utils.test.ts +0 -279
  252. package/test/vitest.config.mts +0 -27
  253. package/test/zcl.test.ts +0 -2831
  254. package/test/zspec/utils.test.ts +0 -68
  255. package/test/zspec/zcl/buffalo.test.ts +0 -1374
  256. package/test/zspec/zcl/frame.test.ts +0 -960
  257. package/test/zspec/zcl/utils.test.ts +0 -273
  258. package/test/zspec/zdo/buffalo.test.ts +0 -1850
  259. package/test/zspec/zdo/utils.test.ts +0 -241
  260. package/tsconfig.json +0 -24
@@ -1,838 +0,0 @@
1
- import {Socket} from "node:net";
2
- import {dirname} from "node:path";
3
-
4
- import {OTRCPDriver} from "zigbee-on-host";
5
- import {setLogger} from "zigbee-on-host/dist/utils/logger";
6
- import type {MACCapabilities, MACHeader} from "zigbee-on-host/dist/zigbee/mac";
7
- import type {ZigbeeAPSHeader, ZigbeeAPSPayload} from "zigbee-on-host/dist/zigbee/zigbee-aps";
8
- import type {ZigbeeNWKGPHeader} from "zigbee-on-host/dist/zigbee/zigbee-nwkgp";
9
-
10
- import type {Backup} from "../../../models/backup";
11
- import {logger} from "../../../utils/logger";
12
- import {Queue} from "../../../utils/queue";
13
- import {wait} from "../../../utils/wait";
14
- import {Waitress} from "../../../utils/waitress";
15
- import * as ZSpec from "../../../zspec";
16
- import * as Zcl from "../../../zspec/zcl";
17
- import * as Zdo from "../../../zspec/zdo";
18
- import type * as ZdoTypes from "../../../zspec/zdo/definition/tstypes";
19
- import {Adapter} from "../../adapter";
20
- import type {ZclPayload} from "../../events";
21
- import {SerialPort} from "../../serialPort";
22
- import {isTcpPath} from "../../socketPortUtils";
23
- import type * as TsType from "../../tstype";
24
- import {bigUInt64ToHexBE} from "./utils";
25
-
26
- const NS = "zh:zoh";
27
-
28
- interface WaitressMatcher {
29
- sender: number | string;
30
- clusterId: number;
31
- endpoint?: number;
32
- commandId?: number;
33
- transactionSequenceNumber?: number;
34
- }
35
-
36
- type ZdoResponse = {
37
- sender: number | string;
38
- clusterId: number;
39
- response: ZdoTypes.GenericZdoResponse;
40
- seqNum: number;
41
- };
42
-
43
- const DEFAULT_REQUEST_TIMEOUT = 15000;
44
-
45
- export class ZoHAdapter extends Adapter {
46
- private serialPort?: SerialPort;
47
- private socketPort?: Socket;
48
- /** True when adapter is currently closing */
49
- // biome-ignore lint/correctness/noUnusedPrivateClassMembers: ignore
50
- private closing: boolean;
51
-
52
- private interpanLock: boolean;
53
-
54
- public readonly driver: OTRCPDriver;
55
- private readonly queue: Queue;
56
- private readonly zclWaitress: Waitress<ZclPayload, WaitressMatcher>;
57
- private readonly zdoWaitress: Waitress<ZdoResponse, WaitressMatcher>;
58
-
59
- constructor(
60
- networkOptions: TsType.NetworkOptions,
61
- serialPortOptions: TsType.SerialPortOptions,
62
- backupPath: string,
63
- adapterOptions: TsType.AdapterOptions,
64
- ) {
65
- super(networkOptions, serialPortOptions, backupPath, adapterOptions);
66
-
67
- this.hasZdoMessageOverhead = true;
68
- this.manufacturerID = Zcl.ManufacturerCode.CONNECTIVITY_STANDARDS_ALLIANCE;
69
- this.closing = false;
70
-
71
- const channel = networkOptions.channelList[0];
72
- this.driver = new OTRCPDriver(
73
- {
74
- txChannel: channel,
75
- ccaBackoffAttempts: 1,
76
- ccaRetries: 4,
77
- enableCSMACA: true,
78
- headerUpdated: true,
79
- reTx: false,
80
- securityProcessed: true,
81
- txDelay: 0,
82
- txDelayBaseTime: 0,
83
- rxChannelAfterTxDone: channel,
84
- },
85
- // NOTE: this information is overwritten on `start` if a save exists
86
- {
87
- // TODO: make this configurable
88
- eui64: Buffer.from([0x5a, 0x6f, 0x48, 0x6f, 0x6e, 0x5a, 0x32, 0x4d]).readBigUInt64LE(0),
89
- panId: this.networkOptions.panID,
90
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
91
- extendedPANId: Buffer.from(this.networkOptions.extendedPanID!).readBigUInt64LE(0),
92
- channel,
93
- nwkUpdateId: 0,
94
- txPower: this.adapterOptions.transmitPower ?? /* v8 ignore next */ 5,
95
- // ZigBeeAlliance09
96
- tcKey: Buffer.from([0x5a, 0x69, 0x67, 0x42, 0x65, 0x65, 0x41, 0x6c, 0x6c, 0x69, 0x61, 0x6e, 0x63, 0x65, 0x30, 0x39]),
97
- tcKeyFrameCounter: 0,
98
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
99
- networkKey: Buffer.from(this.networkOptions.networkKey!),
100
- networkKeyFrameCounter: 0,
101
- networkKeySequenceNumber: 0,
102
- },
103
- dirname(this.backupPath),
104
- );
105
- this.queue = new Queue(this.adapterOptions.concurrent || /* v8 ignore next */ 8); // ORed to avoid 0 (not checked in settings/queue constructor)
106
- this.zclWaitress = new Waitress(this.zclWaitressValidator, this.waitressTimeoutFormatter);
107
- this.zdoWaitress = new Waitress(this.zdoWaitressValidator, this.waitressTimeoutFormatter);
108
-
109
- this.interpanLock = false;
110
- }
111
-
112
- /**
113
- * Init the serial or socket port and hook parser/writer.
114
- */
115
- /* v8 ignore start */
116
- public async initPort(): Promise<void> {
117
- await this.closePort(); // will do nothing if nothing's open
118
-
119
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
120
- if (isTcpPath(this.serialPortOptions.path!)) {
121
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
122
- const pathUrl = new URL(this.serialPortOptions.path!);
123
- const hostname = pathUrl.hostname;
124
- const port = Number.parseInt(pathUrl.port, 10);
125
-
126
- logger.debug(`Opening TCP socket with ${hostname}:${port}`, NS);
127
-
128
- this.socketPort = new Socket();
129
-
130
- this.socketPort.setNoDelay(true);
131
- this.socketPort.setKeepAlive(true, 15000);
132
- this.driver.writer.pipe(this.socketPort);
133
- this.socketPort.pipe(this.driver.parser);
134
- this.driver.parser.on("data", this.driver.onFrame.bind(this.driver));
135
-
136
- return await new Promise((resolve, reject): void => {
137
- const openError = async (err: Error): Promise<void> => {
138
- await this.stop();
139
-
140
- reject(err);
141
- };
142
-
143
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
144
- this.socketPort!.on("connect", () => {
145
- logger.debug("Socket connected", NS);
146
- });
147
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
148
- this.socketPort!.on("ready", (): void => {
149
- logger.info("Socket ready", NS);
150
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
151
- this.socketPort!.removeListener("error", openError);
152
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
153
- this.socketPort!.once("close", this.onPortClose.bind(this));
154
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
155
- this.socketPort!.on("error", this.onPortError.bind(this));
156
-
157
- resolve();
158
- });
159
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
160
- this.socketPort!.once("error", openError);
161
-
162
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
163
- this.socketPort!.connect(port, hostname);
164
- });
165
- }
166
-
167
- const serialOpts = {
168
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
169
- path: this.serialPortOptions.path!,
170
- baudRate: typeof this.serialPortOptions.baudRate === "number" ? this.serialPortOptions.baudRate : 115200,
171
- rtscts: typeof this.serialPortOptions.rtscts === "boolean" ? this.serialPortOptions.rtscts : false,
172
- autoOpen: false,
173
- parity: "none" as const,
174
- stopBits: 1 as const,
175
- xon: false,
176
- xoff: false,
177
- };
178
-
179
- // enable software flow control if RTS/CTS not enabled in config
180
- if (!serialOpts.rtscts) {
181
- logger.info("RTS/CTS config is off, enabling software flow control.", NS);
182
- serialOpts.xon = true;
183
- serialOpts.xoff = true;
184
- }
185
-
186
- logger.debug(() => `Opening serial port with [path=${serialOpts.path} baudRate=${serialOpts.baudRate} rtscts=${serialOpts.rtscts}]`, NS);
187
- this.serialPort = new SerialPort(serialOpts);
188
-
189
- this.driver.writer.pipe(this.serialPort);
190
- this.serialPort.pipe(this.driver.parser);
191
- this.driver.parser.on("data", this.driver.onFrame.bind(this.driver));
192
-
193
- try {
194
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
195
- await this.serialPort!.asyncOpen();
196
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
197
- await this.serialPort!.asyncFlush();
198
-
199
- logger.info("Serial port opened", NS);
200
-
201
- this.serialPort.once("close", this.onPortClose.bind(this));
202
- this.serialPort.on("error", this.onPortError.bind(this));
203
- } catch (error) {
204
- await this.stop();
205
-
206
- throw error;
207
- }
208
- }
209
- /* v8 ignore stop */
210
-
211
- /**
212
- * Handle port closing
213
- * @param err A boolean for Socket, an Error for serialport
214
- */
215
- /* v8 ignore start */
216
- private onPortClose(error: boolean | Error): void {
217
- if (error) {
218
- logger.error("Port closed unexpectedly.", NS);
219
- } else {
220
- logger.info("Port closed.", NS);
221
- }
222
- }
223
- /* v8 ignore stop */
224
-
225
- /**
226
- * Handle port error
227
- * @param error
228
- */
229
- /* v8 ignore start */
230
- private onPortError(error: Error): void {
231
- logger.error(`Port ${error}`, NS);
232
-
233
- this.emit("disconnected");
234
- }
235
- /* v8 ignore stop */
236
-
237
- /* v8 ignore start */
238
- public async closePort(): Promise<void> {
239
- if (this.serialPort?.isOpen) {
240
- try {
241
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
242
- await this.serialPort!.asyncFlushAndClose();
243
- } catch (err) {
244
- logger.error(`Failed to close serial port ${err}.`, NS);
245
- }
246
-
247
- this.serialPort.removeAllListeners();
248
-
249
- this.serialPort = undefined;
250
- } else if (this.socketPort != null && !this.socketPort.closed) {
251
- this.socketPort.destroy();
252
- this.socketPort.removeAllListeners();
253
-
254
- this.socketPort = undefined;
255
- }
256
- }
257
- /* v8 ignore stop */
258
-
259
- public async start(): Promise<TsType.StartResult> {
260
- setLogger(logger); // pass the logger to ZoH
261
- await this.initPort();
262
-
263
- let result: TsType.StartResult = "resumed";
264
- const currentNetParams = await this.driver.readNetworkState();
265
-
266
- if (currentNetParams) {
267
- // Note: channel change is handled by Controller
268
- if (
269
- // TODO: add eui64 whenever added as configurable
270
- this.networkOptions.panID !== currentNetParams.panId ||
271
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
272
- Buffer.from(this.networkOptions.extendedPanID!).readBigUInt64LE(0) !== currentNetParams.extendedPANId ||
273
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
274
- !Buffer.from(this.networkOptions.networkKey!).equals(currentNetParams.networkKey)
275
- ) {
276
- await this.driver.resetNetwork();
277
-
278
- result = "reset";
279
- }
280
- } else {
281
- // no save detected, brand new network
282
- result = "reset";
283
- }
284
-
285
- await this.driver.start();
286
- await this.driver.formNetwork();
287
-
288
- this.driver.on("frame", this.onFrame.bind(this));
289
- this.driver.on("gpFrame", this.onGPFrame.bind(this));
290
- this.driver.on("deviceJoined", this.onDeviceJoined.bind(this));
291
- this.driver.on("deviceRejoined", this.onDeviceRejoined.bind(this));
292
- this.driver.on("deviceLeft", this.onDeviceLeft.bind(this));
293
- this.driver.on("deviceAuthorized", this.onDeviceAuthorized.bind(this));
294
-
295
- return result;
296
- }
297
-
298
- public async stop(): Promise<void> {
299
- this.closing = true;
300
-
301
- this.driver.removeAllListeners();
302
- this.queue.clear();
303
- this.zclWaitress.clear();
304
- this.zdoWaitress.clear();
305
- await this.driver.stop();
306
- }
307
-
308
- public async getCoordinatorIEEE(): Promise<string> {
309
- return await Promise.resolve(`0x${bigUInt64ToHexBE(this.driver.netParams.eui64)}`);
310
- }
311
-
312
- public async getCoordinatorVersion(): Promise<TsType.CoordinatorVersion> {
313
- return await Promise.resolve({
314
- type: "ZigBee on Host",
315
- meta: {
316
- major: this.driver.protocolVersionMajor,
317
- minor: this.driver.protocolVersionMinor,
318
- version: this.driver.ncpVersion,
319
- apiVersion: this.driver.rcpAPIVersion,
320
- revision: `https://github.com/Nerivec/zigbee-on-host (using: ${this.driver.ncpVersion})`,
321
- },
322
- });
323
- }
324
-
325
- /* v8 ignore start */
326
- public async reset(type: "soft" | "hard"): Promise<void> {
327
- await Promise.reject(new Error(`Reset ${type} not support`));
328
- }
329
- /* v8 ignore stop */
330
-
331
- /* v8 ignore start */
332
- public async supportsBackup(): Promise<boolean> {
333
- return await Promise.resolve(false);
334
- }
335
- /* v8 ignore stop */
336
-
337
- /* v8 ignore start */
338
- public async backup(_ieeeAddressesInDatabase: string[]): Promise<Backup> {
339
- return await Promise.reject(new Error("ZigBee on Host handles backup internally"));
340
- }
341
- /* v8 ignore stop */
342
-
343
- public async getNetworkParameters(): Promise<TsType.NetworkParameters> {
344
- return await Promise.resolve({
345
- panID: this.driver.netParams.panId,
346
- extendedPanID: `0x${bigUInt64ToHexBE(this.driver.netParams.extendedPANId)}`,
347
- channel: this.driver.netParams.channel,
348
- nwkUpdateID: this.driver.netParams.nwkUpdateId,
349
- });
350
- }
351
-
352
- /* v8 ignore start */
353
- public async addInstallCode(ieeeAddress: string, key: Buffer): Promise<void> {
354
- await Promise.reject(new Error(`not supported ${ieeeAddress}, ${key.toString("hex")}`));
355
- }
356
- /* v8 ignore stop */
357
-
358
- /* v8 ignore start */
359
- public waitFor(
360
- networkAddress: number,
361
- endpoint: number,
362
- _frameType: Zcl.FrameType,
363
- _direction: Zcl.Direction,
364
- transactionSequenceNumber: number | undefined,
365
- clusterID: number,
366
- commandIdentifier: number,
367
- timeout: number,
368
- ): {promise: Promise<ZclPayload>; cancel: () => void} {
369
- const waiter = this.zclWaitress.waitFor(
370
- {
371
- sender: networkAddress,
372
- endpoint,
373
- clusterId: clusterID,
374
- commandId: commandIdentifier,
375
- transactionSequenceNumber,
376
- },
377
- timeout,
378
- );
379
- const cancel = (): void => this.zclWaitress.remove(waiter.ID);
380
-
381
- return {cancel, promise: waiter.start().promise};
382
- }
383
- /* v8 ignore stop */
384
-
385
- // #region ZDO
386
-
387
- public async sendZdo(
388
- ieeeAddress: string,
389
- networkAddress: number,
390
- clusterId: Zdo.ClusterId,
391
- payload: Buffer,
392
- disableResponse: true,
393
- ): Promise<void>;
394
- public async sendZdo<K extends keyof ZdoTypes.RequestToResponseMap>(
395
- ieeeAddress: string,
396
- networkAddress: number,
397
- clusterId: K,
398
- payload: Buffer,
399
- disableResponse: false,
400
- ): Promise<ZdoTypes.RequestToResponseMap[K]>;
401
- public async sendZdo<K extends keyof ZdoTypes.RequestToResponseMap>(
402
- ieeeAddress: string,
403
- networkAddress: number,
404
- clusterId: K,
405
- payload: Buffer,
406
- disableResponse: boolean,
407
- ): Promise<ZdoTypes.RequestToResponseMap[K] | undefined> {
408
- if (networkAddress === ZSpec.COORDINATOR_ADDRESS) {
409
- // mock ZDO response using driver layer data for coordinator
410
- // seqNum doesn't matter since waitress bypassed, so don't bother doing any logic for it
411
- const response = this.driver.getCoordinatorZDOResponse(clusterId, payload);
412
-
413
- if (!response) {
414
- throw new Error(`Coordinator does not support ZDO cluster ${clusterId}`);
415
- }
416
-
417
- const respClusterId = clusterId | 0x8000;
418
- const result = Zdo.Buffalo.readResponse(this.hasZdoMessageOverhead, respClusterId, response) as ZdoTypes.RequestToResponseMap[K];
419
-
420
- this.emit("zdoResponse", respClusterId, result);
421
-
422
- return result;
423
- }
424
-
425
- return await this.queue.execute(async () => {
426
- this.checkInterpanLock();
427
-
428
- logger.debug(() => `~~~> [ZDO to=${ieeeAddress}:${networkAddress} clusterId=${clusterId} disableResponse=${disableResponse}]`, NS);
429
-
430
- const [, zdoSeqNum] = await this.driver.sendZDO(
431
- payload,
432
- networkAddress, // nwkDest16
433
- undefined, // nwkDest64 XXX: avoid passing EUI64 whenever not absolutely necessary
434
- clusterId, // clusterId
435
- );
436
-
437
- if (!disableResponse) {
438
- const responseClusterId = Zdo.Utils.getResponseClusterId(clusterId);
439
-
440
- if (responseClusterId) {
441
- const resp = await this.zdoWaitress
442
- .waitFor(
443
- {
444
- sender: responseClusterId === Zdo.ClusterId.NETWORK_ADDRESS_RESPONSE ? ieeeAddress : networkAddress,
445
- clusterId: responseClusterId,
446
- transactionSequenceNumber: zdoSeqNum,
447
- },
448
- DEFAULT_REQUEST_TIMEOUT,
449
- )
450
- .start().promise;
451
-
452
- return resp.response as ZdoTypes.RequestToResponseMap[K];
453
- }
454
- }
455
- }, networkAddress);
456
- }
457
-
458
- public async permitJoin(seconds: number, networkAddress?: number): Promise<void> {
459
- if (networkAddress === undefined) {
460
- // send ZDO BCAST
461
- this.driver.allowJoins(seconds, true);
462
- this.driver.gpEnterCommissioningMode(seconds);
463
-
464
- const clusterId = Zdo.ClusterId.PERMIT_JOINING_REQUEST;
465
- // `authentication`: TC significance always 1 (zb specs)
466
- const zdoPayload = Zdo.Buffalo.buildRequest(this.hasZdoMessageOverhead, clusterId, seconds, 1, []);
467
-
468
- await this.sendZdo(ZSpec.BLANK_EUI64, ZSpec.BroadcastAddress.DEFAULT, clusterId, zdoPayload, true);
469
- } else if (networkAddress === ZSpec.COORDINATOR_ADDRESS) {
470
- this.driver.allowJoins(seconds, true);
471
- this.driver.gpEnterCommissioningMode(seconds);
472
- } else {
473
- // send ZDO to networkAddress
474
- this.driver.allowJoins(seconds, false);
475
-
476
- const clusterId = Zdo.ClusterId.PERMIT_JOINING_REQUEST;
477
- // `authentication`: TC significance always 1 (zb specs)
478
- const zdoPayload = Zdo.Buffalo.buildRequest(this.hasZdoMessageOverhead, clusterId, seconds, 1, []);
479
- const result = await this.sendZdo(ZSpec.BLANK_EUI64, networkAddress, clusterId, zdoPayload, false);
480
-
481
- /* v8 ignore start */
482
- if (!Zdo.Buffalo.checkStatus<Zdo.ClusterId.PERMIT_JOINING_RESPONSE>(result)) {
483
- throw new Zdo.StatusError(result[0]);
484
- }
485
- /* v8 ignore stop */
486
- }
487
- }
488
-
489
- // #endregion
490
-
491
- // #region ZCL
492
-
493
- public async sendZclFrameToEndpoint(
494
- ieeeAddr: string,
495
- networkAddress: number,
496
- endpoint: number,
497
- zclFrame: Zcl.Frame,
498
- timeout: number,
499
- disableResponse: boolean,
500
- disableRecovery: boolean,
501
- sourceEndpoint?: number,
502
- ): Promise<ZclPayload | undefined> {
503
- /* v8 ignore start */
504
- if (networkAddress === ZSpec.COORDINATOR_ADDRESS) {
505
- // TODO: handle e.g. GP permit join
506
- logger.debug(
507
- () => `~x~> [ZCL clusterId=${zclFrame.cluster.ID} destEp=${endpoint} sourceEp=${sourceEndpoint}] Not sending to coordinator`,
508
- NS,
509
- );
510
-
511
- return;
512
- }
513
- /* v8 ignore stop */
514
-
515
- let commandResponseId: number | undefined;
516
-
517
- if (zclFrame.command.response !== undefined && disableResponse === false) {
518
- commandResponseId = zclFrame.command.response;
519
- } else if (!zclFrame.header.frameControl.disableDefaultResponse) {
520
- commandResponseId = Zcl.Foundation.defaultRsp.ID;
521
- }
522
-
523
- return await this.queue.execute<ZclPayload | undefined>(async () => {
524
- this.checkInterpanLock();
525
-
526
- logger.debug(
527
- () => `~~~> [ZCL to=${ieeeAddr}:${networkAddress} clusterId=${zclFrame.cluster.ID} destEp=${endpoint} sourceEp=${sourceEndpoint}]`,
528
- NS,
529
- );
530
-
531
- for (let i = 0; i < 2; i++) {
532
- try {
533
- await this.driver.sendUnicast(
534
- zclFrame.toBuffer(),
535
- sourceEndpoint === ZSpec.GP_ENDPOINT && endpoint === ZSpec.GP_ENDPOINT ? ZSpec.GP_PROFILE_ID : ZSpec.HA_PROFILE_ID,
536
- zclFrame.cluster.ID,
537
- networkAddress, // nwkDest16
538
- undefined, // nwkDest64 XXX: avoid passing EUI64 whenever not absolutely necessary
539
- endpoint, // destEp
540
- sourceEndpoint ?? 1, // sourceEp
541
- );
542
-
543
- if (commandResponseId !== undefined) {
544
- const resp = await this.zclWaitress
545
- .waitFor(
546
- {
547
- sender: networkAddress,
548
- clusterId: zclFrame.cluster.ID,
549
- endpoint,
550
- commandId: commandResponseId,
551
- transactionSequenceNumber: zclFrame.header.transactionSequenceNumber,
552
- },
553
- timeout,
554
- )
555
- .start().promise;
556
-
557
- return resp;
558
- }
559
-
560
- return;
561
- } catch (error) {
562
- if (disableRecovery || i === 1) {
563
- throw error;
564
- } // else retry
565
- }
566
- /* v8 ignore start */
567
- } // coverage detection failure
568
- /* v8 ignore stop */
569
- });
570
- }
571
-
572
- public async sendZclFrameToGroup(groupID: number, zclFrame: Zcl.Frame, sourceEndpoint?: number): Promise<void> {
573
- return await this.queue.execute<void>(async () => {
574
- this.checkInterpanLock();
575
-
576
- logger.debug(() => `~~~> [ZCL GROUP to=${groupID} clusterId=${zclFrame.cluster.ID} sourceEp=${sourceEndpoint}]`, NS);
577
-
578
- await this.driver.sendGroupcast(zclFrame.toBuffer(), ZSpec.HA_PROFILE_ID, zclFrame.cluster.ID, groupID, sourceEndpoint ?? 1);
579
- // settle
580
- await wait(500);
581
- });
582
- }
583
-
584
- public async sendZclFrameToAll(
585
- endpoint: number,
586
- zclFrame: Zcl.Frame,
587
- sourceEndpoint: number,
588
- destination: ZSpec.BroadcastAddress,
589
- ): Promise<void> {
590
- return await this.queue.execute<void>(async () => {
591
- this.checkInterpanLock();
592
-
593
- logger.debug(() => `~~~> [ZCL BROADCAST to=${destination} destEp=${endpoint} sourceEp=${sourceEndpoint}]`, NS);
594
-
595
- await this.driver.sendBroadcast(
596
- zclFrame.toBuffer(),
597
- sourceEndpoint === ZSpec.GP_ENDPOINT && endpoint === ZSpec.GP_ENDPOINT ? ZSpec.GP_PROFILE_ID : ZSpec.HA_PROFILE_ID,
598
- zclFrame.cluster.ID,
599
- destination,
600
- endpoint,
601
- sourceEndpoint,
602
- );
603
- // settle
604
- await wait(500);
605
- });
606
- }
607
-
608
- // #endregion
609
-
610
- // #region InterPAN
611
-
612
- /* v8 ignore start */
613
- public async setChannelInterPAN(channel: number): Promise<void> {
614
- await Promise.reject(new Error(`not supported ${channel}`));
615
- }
616
- /* v8 ignore stop */
617
-
618
- /* v8 ignore start */
619
- public async sendZclFrameInterPANToIeeeAddr(zclFrame: Zcl.Frame, ieeeAddress: string): Promise<void> {
620
- await Promise.reject(new Error(`not supported ${JSON.stringify(zclFrame)}, ${ieeeAddress}`));
621
- }
622
- /* v8 ignore stop */
623
-
624
- /* v8 ignore start */
625
- public async sendZclFrameInterPANBroadcast(zclFrame: Zcl.Frame, timeout: number): Promise<ZclPayload> {
626
- return await Promise.reject(new Error(`not supported ${JSON.stringify(zclFrame)}, ${timeout}`));
627
- }
628
- /* v8 ignore stop */
629
-
630
- /* v8 ignore start */
631
- public async restoreChannelInterPAN(): Promise<void> {
632
- await Promise.reject(new Error("not supported"));
633
- }
634
- /* v8 ignore stop */
635
-
636
- // #endregion
637
-
638
- // #region Implementation-Specific
639
-
640
- /* v8 ignore start */
641
- private checkInterpanLock(): void {
642
- if (this.interpanLock) {
643
- throw new Error("[INTERPAN MODE] Cannot execute non-InterPAN commands.");
644
- }
645
- }
646
- /* v8 ignore stop */
647
-
648
- /**
649
- * @param sender16 If undefined, sender64 is expected defined
650
- * @param sender64 If undefined, sender16 is expected defined
651
- * @param apsHeader
652
- * @param apsPayload
653
- */
654
- private onFrame(
655
- sender16: number | undefined,
656
- sender64: bigint | undefined,
657
- apsHeader: ZigbeeAPSHeader,
658
- apsPayload: ZigbeeAPSPayload,
659
- rssi: number,
660
- ): void {
661
- if (apsHeader.profileId === Zdo.ZDO_PROFILE_ID) {
662
- logger.debug(() => `<~~~ APS ZDO[sender=${sender16}:${sender64} clusterId=${apsHeader.clusterId}]`, NS);
663
-
664
- try {
665
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
666
- const result = Zdo.Buffalo.readResponse(this.hasZdoMessageOverhead, apsHeader.clusterId!, apsPayload);
667
-
668
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
669
- if (apsHeader.clusterId! === Zdo.ClusterId.NETWORK_ADDRESS_RESPONSE) {
670
- // special case to properly resolve a NETWORK_ADDRESS_RESPONSE following a NETWORK_ADDRESS_REQUEST (based on EUI64 from ZDO payload)
671
- // NOTE: if response has invalid status (no EUI64 available), response waiter will eventually time out
672
- if (Zdo.Buffalo.checkStatus<Zdo.ClusterId.NETWORK_ADDRESS_RESPONSE>(result)) {
673
- this.zdoWaitress.resolve({sender: result[1].eui64, clusterId: apsHeader.clusterId, response: result, seqNum: apsPayload[0]});
674
- }
675
- } else {
676
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
677
- this.zdoWaitress.resolve({sender: sender16!, clusterId: apsHeader.clusterId!, response: result, seqNum: apsPayload[0]});
678
- }
679
-
680
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
681
- this.emit("zdoResponse", apsHeader.clusterId!, result);
682
- /* v8 ignore start */
683
- } catch (error) {
684
- logger.error(`${(error as Error).message}`, NS);
685
- }
686
- /* v8 ignore stop */
687
- } else {
688
- logger.debug(() => `<~~~ APS[sender=${sender16}:${sender64} profileId=${apsHeader.profileId} clusterId=${apsHeader.clusterId}]`, NS);
689
-
690
- const payload: ZclPayload = {
691
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
692
- clusterID: apsHeader.clusterId!,
693
- header: Zcl.Header.fromBuffer(apsPayload),
694
- address:
695
- sender64 !== undefined
696
- ? `0x${bigUInt64ToHexBE(sender64)}`
697
- : // biome-ignore lint/style/noNonNullAssertion: ignore
698
- sender16!,
699
- data: apsPayload,
700
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
701
- endpoint: apsHeader.sourceEndpoint!,
702
- linkquality: rssi, // TODO: convert RSSI to LQA
703
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
704
- groupID: apsHeader.group!,
705
- wasBroadcast: apsHeader.frameControl.deliveryMode === 2 /* BCAST */,
706
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
707
- destinationEndpoint: apsHeader.destEndpoint!,
708
- };
709
-
710
- this.zclWaitress.resolve(payload);
711
- this.emit("zclPayload", payload);
712
- }
713
- }
714
-
715
- private onGPFrame(cmdId: number, payload: Buffer, macHeader: MACHeader, nwkHeader: ZigbeeNWKGPHeader, rssi: number): void {
716
- // transform into a ZCL frame
717
- const data = Buffer.alloc((nwkHeader.frameControlExt?.appId === 0x02 /* ZGP */ ? /* v8 ignore next */ 20 : 15) + payload.byteLength);
718
- let offset = 0;
719
- data.writeUInt8(0b00000001, offset); // frameControl: FrameType.SPECIFIC + Direction.CLIENT_TO_SERVER + disableDefaultResponse=false
720
- offset += 1;
721
- data.writeUInt8(macHeader.sequenceNumber ?? /* v8 ignore next */ 0, offset);
722
- offset += 1;
723
- data.writeUInt8(cmdId === 0xe0 ? 0x04 /* commissioning notification */ : 0x00 /* notification */, offset);
724
- offset += 1;
725
-
726
- if (nwkHeader.frameControlExt) {
727
- /* v8 ignore start */
728
- if (cmdId === 0xe0) {
729
- data.writeUInt16LE(
730
- (nwkHeader.frameControlExt.appId & 0x7) |
731
- (((nwkHeader.frameControlExt.rxAfterTx ? 1 : 0) & 0x1) << 3) |
732
- ((nwkHeader.frameControlExt.securityLevel & 0x3) << 4),
733
- offset,
734
- );
735
- /* v8 ignore stop */
736
- } else {
737
- data.writeUInt16LE(
738
- (nwkHeader.frameControlExt.appId & 0x7) |
739
- ((nwkHeader.frameControlExt.securityLevel & 0x3) << 6) |
740
- /* v8 ignore next */ (((nwkHeader.frameControlExt.rxAfterTx ? 1 : 0) & 0x3) << 11),
741
- offset,
742
- );
743
- }
744
- } else {
745
- data.writeUInt16LE(0, offset); // options, only srcID present
746
- }
747
-
748
- offset += 2;
749
-
750
- /* v8 ignore start */
751
- if (nwkHeader.frameControlExt?.appId === 0x02 /* ZGP */) {
752
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
753
- data.writeBigUInt64LE(macHeader.source64!, offset);
754
- offset += 8;
755
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
756
- data.writeUInt8(nwkHeader.endpoint!, offset);
757
- offset += 1;
758
- /* v8 ignore stop */
759
- } else {
760
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
761
- data.writeUInt32LE(nwkHeader.sourceId!, offset);
762
- offset += 4;
763
- }
764
-
765
- data.writeUInt32LE(nwkHeader.securityFrameCounter ?? 0, offset);
766
- offset += 4;
767
- data.writeUInt8(cmdId, offset);
768
- offset += 1;
769
- data.writeUInt8(payload.byteLength, offset);
770
- offset += 1;
771
- data.set(payload, offset);
772
-
773
- const zclPayload: ZclPayload = {
774
- clusterID: 0x21 /* Green Power */,
775
- header: Zcl.Header.fromBuffer(data),
776
- address:
777
- macHeader.source64 !== undefined /* v8 ignore next */
778
- ? /* v8 ignore next */ `0x${bigUInt64ToHexBE(macHeader.source64)}`
779
- : // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
780
- nwkHeader.sourceId! & 0xffff,
781
- data,
782
- endpoint: ZSpec.GP_ENDPOINT,
783
- linkquality: rssi, // TODO: convert RSSI to LQA
784
- groupID: ZSpec.GP_GROUP_ID,
785
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
786
- wasBroadcast: macHeader.destination64 === undefined && macHeader.destination16! >= 0xfff8,
787
- destinationEndpoint: ZSpec.GP_ENDPOINT,
788
- };
789
-
790
- this.zclWaitress.resolve(zclPayload);
791
- this.emit("zclPayload", zclPayload);
792
- }
793
-
794
- private onDeviceJoined(source16: number, source64: bigint, capabilities: MACCapabilities): void {
795
- // XXX: don't delay if no cap? (joined through router)
796
- if (capabilities?.rxOnWhenIdle) {
797
- this.emit("deviceJoined", {networkAddress: source16, ieeeAddr: `0x${bigUInt64ToHexBE(source64)}`});
798
- } else {
799
- // XXX: end devices can be finicky about finishing the key authorization, Z2M interview can create a bottleneck, so delay it
800
- setTimeout(() => {
801
- this.emit("deviceJoined", {networkAddress: source16, ieeeAddr: `0x${bigUInt64ToHexBE(source64)}`});
802
- }, 5000);
803
- }
804
- }
805
- private onDeviceRejoined(source16: number, source64: bigint, _capabilities: MACCapabilities): void {
806
- this.emit("deviceJoined", {networkAddress: source16, ieeeAddr: `0x${bigUInt64ToHexBE(source64)}`});
807
- }
808
-
809
- private onDeviceLeft(source16: number, source64: bigint): void {
810
- this.emit("deviceLeave", {networkAddress: source16, ieeeAddr: `0x${bigUInt64ToHexBE(source64)}`});
811
- }
812
-
813
- /* v8 ignore start */
814
- private onDeviceAuthorized(_source16: number, _source64: bigint): void {}
815
- /* v8 ignore stop */
816
-
817
- private waitressTimeoutFormatter(matcher: WaitressMatcher, timeout: number): string {
818
- return `Timeout after ${timeout}ms [sender=${matcher.sender} clusterId=${matcher.clusterId} cmdId=${matcher.commandId}]`;
819
- }
820
-
821
- private zclWaitressValidator(payload: ZclPayload, matcher: WaitressMatcher): boolean {
822
- return (
823
- // no sender in Touchlink
824
- (matcher.sender === undefined || payload.address === matcher.sender) &&
825
- payload.clusterID === matcher.clusterId &&
826
- payload.endpoint === matcher.endpoint &&
827
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
828
- payload.header!.commandIdentifier === matcher.commandId &&
829
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
830
- (matcher.transactionSequenceNumber === undefined || payload.header!.transactionSequenceNumber === matcher.transactionSequenceNumber)
831
- );
832
- }
833
-
834
- private zdoWaitressValidator(payload: ZdoResponse, matcher: WaitressMatcher): boolean {
835
- return matcher.sender === payload.sender && matcher.clusterId === payload.clusterId && matcher.transactionSequenceNumber === payload.seqNum;
836
- }
837
- // #endregion
838
- }