zigbee-herdsman 6.0.2 → 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 (252) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/package.json +12 -3
  3. package/.github/ISSUE_TEMPLATE/config.yml +0 -5
  4. package/.github/dependabot.yml +0 -22
  5. package/.github/workflows/ci.yml +0 -69
  6. package/.github/workflows/release-please.yml +0 -18
  7. package/.github/workflows/stale.yml +0 -20
  8. package/.github/workflows/typedoc.yaml +0 -47
  9. package/.release-please-manifest.json +0 -3
  10. package/.vscode/extensions.json +0 -3
  11. package/.vscode/settings.json +0 -11
  12. package/biome.json +0 -98
  13. package/examples/join-and-log.js +0 -24
  14. package/release-please-config.json +0 -9
  15. package/src/adapter/adapter.ts +0 -189
  16. package/src/adapter/adapterDiscovery.ts +0 -666
  17. package/src/adapter/const.ts +0 -12
  18. package/src/adapter/deconz/adapter/deconzAdapter.ts +0 -877
  19. package/src/adapter/deconz/driver/constants.ts +0 -246
  20. package/src/adapter/deconz/driver/driver.ts +0 -1540
  21. package/src/adapter/deconz/driver/frame.ts +0 -11
  22. package/src/adapter/deconz/driver/frameParser.ts +0 -753
  23. package/src/adapter/deconz/driver/parser.ts +0 -45
  24. package/src/adapter/deconz/driver/writer.ts +0 -22
  25. package/src/adapter/deconz/types.d.ts +0 -13
  26. package/src/adapter/ember/adapter/emberAdapter.ts +0 -2265
  27. package/src/adapter/ember/adapter/endpoints.ts +0 -86
  28. package/src/adapter/ember/adapter/oneWaitress.ts +0 -324
  29. package/src/adapter/ember/adapter/tokensManager.ts +0 -782
  30. package/src/adapter/ember/consts.ts +0 -178
  31. package/src/adapter/ember/enums.ts +0 -1746
  32. package/src/adapter/ember/ezsp/buffalo.ts +0 -1392
  33. package/src/adapter/ember/ezsp/consts.ts +0 -148
  34. package/src/adapter/ember/ezsp/enums.ts +0 -1114
  35. package/src/adapter/ember/ezsp/ezsp.ts +0 -9061
  36. package/src/adapter/ember/ezspError.ts +0 -10
  37. package/src/adapter/ember/types.ts +0 -866
  38. package/src/adapter/ember/uart/ash.ts +0 -1960
  39. package/src/adapter/ember/uart/consts.ts +0 -109
  40. package/src/adapter/ember/uart/enums.ts +0 -192
  41. package/src/adapter/ember/uart/parser.ts +0 -48
  42. package/src/adapter/ember/uart/queues.ts +0 -247
  43. package/src/adapter/ember/uart/writer.ts +0 -53
  44. package/src/adapter/ember/utils/initters.ts +0 -58
  45. package/src/adapter/ember/utils/math.ts +0 -73
  46. package/src/adapter/events.ts +0 -21
  47. package/src/adapter/ezsp/adapter/backup.ts +0 -109
  48. package/src/adapter/ezsp/adapter/ezspAdapter.ts +0 -614
  49. package/src/adapter/ezsp/driver/commands.ts +0 -2497
  50. package/src/adapter/ezsp/driver/consts.ts +0 -11
  51. package/src/adapter/ezsp/driver/driver.ts +0 -1002
  52. package/src/adapter/ezsp/driver/ezsp.ts +0 -802
  53. package/src/adapter/ezsp/driver/frame.ts +0 -101
  54. package/src/adapter/ezsp/driver/index.ts +0 -4
  55. package/src/adapter/ezsp/driver/multicast.ts +0 -78
  56. package/src/adapter/ezsp/driver/parser.ts +0 -81
  57. package/src/adapter/ezsp/driver/types/basic.ts +0 -201
  58. package/src/adapter/ezsp/driver/types/index.ts +0 -239
  59. package/src/adapter/ezsp/driver/types/named.ts +0 -2330
  60. package/src/adapter/ezsp/driver/types/struct.ts +0 -844
  61. package/src/adapter/ezsp/driver/uart.ts +0 -460
  62. package/src/adapter/ezsp/driver/utils/crc16ccitt.ts +0 -44
  63. package/src/adapter/ezsp/driver/utils/index.ts +0 -32
  64. package/src/adapter/ezsp/driver/writer.ts +0 -64
  65. package/src/adapter/index.ts +0 -3
  66. package/src/adapter/serialPort.ts +0 -58
  67. package/src/adapter/socketPortUtils.ts +0 -16
  68. package/src/adapter/tstype.ts +0 -78
  69. package/src/adapter/z-stack/adapter/adapter-backup.ts +0 -519
  70. package/src/adapter/z-stack/adapter/adapter-nv-memory.ts +0 -457
  71. package/src/adapter/z-stack/adapter/endpoints.ts +0 -57
  72. package/src/adapter/z-stack/adapter/manager.ts +0 -543
  73. package/src/adapter/z-stack/adapter/tstype.ts +0 -6
  74. package/src/adapter/z-stack/adapter/zStackAdapter.ts +0 -1190
  75. package/src/adapter/z-stack/constants/af.ts +0 -27
  76. package/src/adapter/z-stack/constants/common.ts +0 -285
  77. package/src/adapter/z-stack/constants/dbg.ts +0 -23
  78. package/src/adapter/z-stack/constants/index.ts +0 -11
  79. package/src/adapter/z-stack/constants/mac.ts +0 -128
  80. package/src/adapter/z-stack/constants/sapi.ts +0 -25
  81. package/src/adapter/z-stack/constants/sys.ts +0 -72
  82. package/src/adapter/z-stack/constants/util.ts +0 -82
  83. package/src/adapter/z-stack/constants/utils.ts +0 -14
  84. package/src/adapter/z-stack/constants/zdo.ts +0 -103
  85. package/src/adapter/z-stack/models/startup-options.ts +0 -13
  86. package/src/adapter/z-stack/structs/entries/address-manager-entry.ts +0 -44
  87. package/src/adapter/z-stack/structs/entries/address-manager-table.ts +0 -19
  88. package/src/adapter/z-stack/structs/entries/aps-link-key-data-entry.ts +0 -12
  89. package/src/adapter/z-stack/structs/entries/aps-link-key-data-table.ts +0 -21
  90. package/src/adapter/z-stack/structs/entries/aps-tc-link-key-entry.ts +0 -19
  91. package/src/adapter/z-stack/structs/entries/aps-tc-link-key-table.ts +0 -21
  92. package/src/adapter/z-stack/structs/entries/channel-list.ts +0 -8
  93. package/src/adapter/z-stack/structs/entries/has-configured.ts +0 -16
  94. package/src/adapter/z-stack/structs/entries/index.ts +0 -16
  95. package/src/adapter/z-stack/structs/entries/nib.ts +0 -66
  96. package/src/adapter/z-stack/structs/entries/nwk-key-descriptor.ts +0 -15
  97. package/src/adapter/z-stack/structs/entries/nwk-key.ts +0 -13
  98. package/src/adapter/z-stack/structs/entries/nwk-pan-id.ts +0 -8
  99. package/src/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.ts +0 -20
  100. package/src/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.ts +0 -19
  101. package/src/adapter/z-stack/structs/entries/security-manager-entry.ts +0 -33
  102. package/src/adapter/z-stack/structs/entries/security-manager-table.ts +0 -22
  103. package/src/adapter/z-stack/structs/index.ts +0 -4
  104. package/src/adapter/z-stack/structs/serializable-memory-object.ts +0 -14
  105. package/src/adapter/z-stack/structs/struct.ts +0 -367
  106. package/src/adapter/z-stack/structs/table.ts +0 -198
  107. package/src/adapter/z-stack/unpi/constants.ts +0 -33
  108. package/src/adapter/z-stack/unpi/frame.ts +0 -62
  109. package/src/adapter/z-stack/unpi/index.ts +0 -4
  110. package/src/adapter/z-stack/unpi/parser.ts +0 -56
  111. package/src/adapter/z-stack/unpi/writer.ts +0 -21
  112. package/src/adapter/z-stack/utils/channel-list.ts +0 -40
  113. package/src/adapter/z-stack/utils/index.ts +0 -2
  114. package/src/adapter/z-stack/utils/network-options.ts +0 -26
  115. package/src/adapter/z-stack/znp/buffaloZnp.ts +0 -175
  116. package/src/adapter/z-stack/znp/definition.ts +0 -2713
  117. package/src/adapter/z-stack/znp/index.ts +0 -2
  118. package/src/adapter/z-stack/znp/parameterType.ts +0 -22
  119. package/src/adapter/z-stack/znp/tstype.ts +0 -44
  120. package/src/adapter/z-stack/znp/utils.ts +0 -10
  121. package/src/adapter/z-stack/znp/znp.ts +0 -342
  122. package/src/adapter/z-stack/znp/zpiObject.ts +0 -148
  123. package/src/adapter/zboss/adapter/zbossAdapter.ts +0 -526
  124. package/src/adapter/zboss/commands.ts +0 -1184
  125. package/src/adapter/zboss/consts.ts +0 -9
  126. package/src/adapter/zboss/driver.ts +0 -422
  127. package/src/adapter/zboss/enums.ts +0 -360
  128. package/src/adapter/zboss/frame.ts +0 -227
  129. package/src/adapter/zboss/reader.ts +0 -65
  130. package/src/adapter/zboss/types.ts +0 -0
  131. package/src/adapter/zboss/uart.ts +0 -428
  132. package/src/adapter/zboss/utils.ts +0 -58
  133. package/src/adapter/zboss/writer.ts +0 -49
  134. package/src/adapter/zigate/adapter/patchZdoBuffaloBE.ts +0 -27
  135. package/src/adapter/zigate/adapter/zigateAdapter.ts +0 -618
  136. package/src/adapter/zigate/driver/LICENSE +0 -17
  137. package/src/adapter/zigate/driver/buffaloZiGate.ts +0 -212
  138. package/src/adapter/zigate/driver/commandType.ts +0 -418
  139. package/src/adapter/zigate/driver/constants.ts +0 -150
  140. package/src/adapter/zigate/driver/frame.ts +0 -197
  141. package/src/adapter/zigate/driver/messageType.ts +0 -287
  142. package/src/adapter/zigate/driver/parameterType.ts +0 -32
  143. package/src/adapter/zigate/driver/ziGateObject.ts +0 -146
  144. package/src/adapter/zigate/driver/zigate.ts +0 -423
  145. package/src/adapter/zoh/adapter/utils.ts +0 -27
  146. package/src/adapter/zoh/adapter/zohAdapter.ts +0 -838
  147. package/src/buffalo/buffalo.ts +0 -342
  148. package/src/buffalo/index.ts +0 -1
  149. package/src/controller/controller.ts +0 -1022
  150. package/src/controller/database.ts +0 -124
  151. package/src/controller/events.ts +0 -52
  152. package/src/controller/greenPower.ts +0 -603
  153. package/src/controller/helpers/index.ts +0 -1
  154. package/src/controller/helpers/installCodes.ts +0 -107
  155. package/src/controller/helpers/request.ts +0 -96
  156. package/src/controller/helpers/requestQueue.ts +0 -125
  157. package/src/controller/helpers/zclFrameConverter.ts +0 -47
  158. package/src/controller/helpers/zclTransactionSequenceNumber.ts +0 -19
  159. package/src/controller/index.ts +0 -6
  160. package/src/controller/model/device.ts +0 -1249
  161. package/src/controller/model/endpoint.ts +0 -1105
  162. package/src/controller/model/entity.ts +0 -23
  163. package/src/controller/model/group.ts +0 -424
  164. package/src/controller/model/index.ts +0 -5
  165. package/src/controller/model/zigbeeEntity.ts +0 -30
  166. package/src/controller/touchlink.ts +0 -189
  167. package/src/controller/tstype.ts +0 -274
  168. package/src/index.ts +0 -12
  169. package/src/models/backup-storage-legacy.ts +0 -48
  170. package/src/models/backup-storage-unified.ts +0 -47
  171. package/src/models/backup.ts +0 -37
  172. package/src/models/index.ts +0 -5
  173. package/src/models/network-options.ts +0 -11
  174. package/src/utils/backup.ts +0 -152
  175. package/src/utils/index.ts +0 -5
  176. package/src/utils/logger.ts +0 -20
  177. package/src/utils/patchBigIntSerialization.ts +0 -8
  178. package/src/utils/queue.ts +0 -76
  179. package/src/utils/types.d.ts +0 -3
  180. package/src/utils/utils.ts +0 -19
  181. package/src/utils/wait.ts +0 -5
  182. package/src/utils/waitress.ts +0 -96
  183. package/src/zspec/consts.ts +0 -84
  184. package/src/zspec/enums.ts +0 -22
  185. package/src/zspec/index.ts +0 -3
  186. package/src/zspec/tstypes.ts +0 -18
  187. package/src/zspec/utils.ts +0 -247
  188. package/src/zspec/zcl/buffaloZcl.ts +0 -1220
  189. package/src/zspec/zcl/definition/cluster.ts +0 -5915
  190. package/src/zspec/zcl/definition/clusters-typegen.ts +0 -588
  191. package/src/zspec/zcl/definition/clusters-types.ts +0 -7331
  192. package/src/zspec/zcl/definition/consts.ts +0 -24
  193. package/src/zspec/zcl/definition/enums.ts +0 -203
  194. package/src/zspec/zcl/definition/foundation.ts +0 -329
  195. package/src/zspec/zcl/definition/manufacturerCode.ts +0 -729
  196. package/src/zspec/zcl/definition/status.ts +0 -69
  197. package/src/zspec/zcl/definition/tstype.ts +0 -377
  198. package/src/zspec/zcl/index.ts +0 -11
  199. package/src/zspec/zcl/utils.ts +0 -321
  200. package/src/zspec/zcl/zclFrame.ts +0 -356
  201. package/src/zspec/zcl/zclHeader.ts +0 -102
  202. package/src/zspec/zcl/zclStatusError.ts +0 -10
  203. package/src/zspec/zdo/buffaloZdo.ts +0 -2336
  204. package/src/zspec/zdo/definition/clusters.ts +0 -722
  205. package/src/zspec/zdo/definition/consts.ts +0 -16
  206. package/src/zspec/zdo/definition/enums.ts +0 -99
  207. package/src/zspec/zdo/definition/status.ts +0 -105
  208. package/src/zspec/zdo/definition/tstypes.ts +0 -1062
  209. package/src/zspec/zdo/index.ts +0 -7
  210. package/src/zspec/zdo/utils.ts +0 -76
  211. package/src/zspec/zdo/zdoStatusError.ts +0 -10
  212. package/test/adapter/adapter.test.ts +0 -1062
  213. package/test/adapter/ember/ash.test.ts +0 -337
  214. package/test/adapter/ember/consts.ts +0 -131
  215. package/test/adapter/ember/emberAdapter.test.ts +0 -3449
  216. package/test/adapter/ember/ezsp.test.ts +0 -385
  217. package/test/adapter/ember/ezspBuffalo.test.ts +0 -93
  218. package/test/adapter/ember/ezspError.test.ts +0 -12
  219. package/test/adapter/ember/math.test.ts +0 -206
  220. package/test/adapter/ezsp/frame.test.ts +0 -30
  221. package/test/adapter/ezsp/uart.test.ts +0 -181
  222. package/test/adapter/z-stack/adapter.test.ts +0 -3984
  223. package/test/adapter/z-stack/constants.test.ts +0 -33
  224. package/test/adapter/z-stack/structs.test.ts +0 -115
  225. package/test/adapter/z-stack/unpi.test.ts +0 -213
  226. package/test/adapter/z-stack/znp.test.ts +0 -1284
  227. package/test/adapter/zboss/fixZdoResponse.test.ts +0 -179
  228. package/test/adapter/zigate/patchZdoBuffaloBE.test.ts +0 -81
  229. package/test/adapter/zigate/zdo.test.ts +0 -187
  230. package/test/adapter/zoh/utils.test.ts +0 -36
  231. package/test/adapter/zoh/zohAdapter.test.ts +0 -1307
  232. package/test/benchOptions.ts +0 -14
  233. package/test/buffalo.test.ts +0 -431
  234. package/test/controller.bench.ts +0 -214
  235. package/test/controller.test.ts +0 -8702
  236. package/test/greenpower.test.ts +0 -1408
  237. package/test/mockAdapters.ts +0 -65
  238. package/test/mockDevices.ts +0 -598
  239. package/test/requests.bench.ts +0 -229
  240. package/test/testUtils.ts +0 -20
  241. package/test/tsconfig.json +0 -9
  242. package/test/utils/math.ts +0 -19
  243. package/test/utils.test.ts +0 -279
  244. package/test/vitest.config.mts +0 -26
  245. package/test/zcl.test.ts +0 -2831
  246. package/test/zspec/utils.test.ts +0 -68
  247. package/test/zspec/zcl/buffalo.test.ts +0 -1374
  248. package/test/zspec/zcl/frame.test.ts +0 -960
  249. package/test/zspec/zcl/utils.test.ts +0 -273
  250. package/test/zspec/zdo/buffalo.test.ts +0 -1850
  251. package/test/zspec/zdo/utils.test.ts +0 -241
  252. 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
- }