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,1190 +0,0 @@
1
- import assert from "node:assert";
2
-
3
- import debounce from "debounce";
4
-
5
- import type * as Models from "../../../models";
6
- import {Queue, Waitress, wait} from "../../../utils";
7
- import {logger} from "../../../utils/logger";
8
- import * as ZSpec from "../../../zspec";
9
- import type {BroadcastAddress} from "../../../zspec/enums";
10
- import type {Eui64} from "../../../zspec/tstypes";
11
- import * as Zcl from "../../../zspec/zcl";
12
- import * as Zdo from "../../../zspec/zdo";
13
- import type * as ZdoTypes from "../../../zspec/zdo/definition/tstypes";
14
- import Adapter from "../../adapter";
15
- import type * as Events from "../../events";
16
- import type {AdapterOptions, CoordinatorVersion, NetworkOptions, NetworkParameters, SerialPortOptions, StartResult} from "../../tstype";
17
- import * as Constants from "../constants";
18
- import {Constants as UnpiConstants} from "../unpi";
19
- import {Znp, type ZpiObject} from "../znp";
20
- import Definition from "../znp/definition";
21
- import {isMtCmdAreqZdo} from "../znp/utils";
22
- import {ZnpAdapterManager} from "./manager";
23
- import {ZnpVersion} from "./tstype";
24
-
25
- const NS = "zh:zstack";
26
- const Subsystem = UnpiConstants.Subsystem;
27
- const Type = UnpiConstants.Type;
28
- const {ZnpCommandStatus, AddressMode} = Constants.COMMON;
29
-
30
- const DataConfirmTimeout = 9999; // Not an actual code
31
- const DataConfirmErrorCodeLookup: {[k: number]: string} = {
32
- [DataConfirmTimeout]: "Timeout",
33
- 26: "MAC no resources",
34
- 183: "APS no ack",
35
- 205: "No network route",
36
- 225: "MAC channel access failure",
37
- 233: "MAC no ack",
38
- 240: "MAC transaction expired",
39
- };
40
-
41
- interface WaitressMatcher {
42
- address?: number | string;
43
- endpoint: number;
44
- transactionSequenceNumber?: number;
45
- frameType: Zcl.FrameType;
46
- clusterID: number;
47
- commandIdentifier: number;
48
- direction: number;
49
- }
50
-
51
- class DataConfirmError extends Error {
52
- public code: number;
53
- constructor(code: number) {
54
- const message = `Data request failed with error: '${DataConfirmErrorCodeLookup[code]}' (${code})`;
55
- super(message);
56
- this.code = code;
57
- }
58
- }
59
-
60
- export class ZStackAdapter extends Adapter {
61
- private deviceAnnounceRouteDiscoveryDebouncers: Map<number, () => void>;
62
- private znp: Znp;
63
- // @ts-expect-error initialized in `start`
64
- private adapterManager: ZnpAdapterManager;
65
- private transactionID: number;
66
- // @ts-expect-error initialized in `start`
67
- private version: {
68
- product: number;
69
- transportrev: number;
70
- majorrel: number;
71
- minorrel: number;
72
- maintrel: number;
73
- revision: string;
74
- };
75
- private closing: boolean;
76
- // @ts-expect-error initialized in `start`
77
- private queue: Queue;
78
- private supportsLED?: boolean;
79
- private interpanLock: boolean;
80
- private interpanEndpointRegistered: boolean;
81
- private waitress: Waitress<Events.ZclPayload, WaitressMatcher>;
82
-
83
- public constructor(networkOptions: NetworkOptions, serialPortOptions: SerialPortOptions, backupPath: string, adapterOptions: AdapterOptions) {
84
- super(networkOptions, serialPortOptions, backupPath, adapterOptions);
85
- this.hasZdoMessageOverhead = false;
86
- this.manufacturerID = Zcl.ManufacturerCode.TEXAS_INSTRUMENTS;
87
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
88
- this.znp = new Znp(this.serialPortOptions.path!, this.serialPortOptions.baudRate!, this.serialPortOptions.rtscts!);
89
-
90
- this.transactionID = 0;
91
- this.deviceAnnounceRouteDiscoveryDebouncers = new Map();
92
- this.interpanLock = false;
93
- this.interpanEndpointRegistered = false;
94
- this.closing = false;
95
- this.waitress = new Waitress<Events.ZclPayload, WaitressMatcher>(this.waitressValidator, this.waitressTimeoutFormatter);
96
-
97
- this.znp.on("received", this.onZnpRecieved.bind(this));
98
- this.znp.on("close", this.onZnpClose.bind(this));
99
- }
100
-
101
- /**
102
- * Adapter methods
103
- */
104
- public async start(): Promise<StartResult> {
105
- await this.znp.open();
106
-
107
- const attempts = 3;
108
- for (let i = 0; i < attempts; i++) {
109
- try {
110
- await this.znp.request(Subsystem.SYS, "ping", {capabilities: 1});
111
- break;
112
- } catch (e) {
113
- if (attempts - 1 === i) {
114
- throw new Error(`Failed to connect to the adapter (${e})`);
115
- }
116
- }
117
- }
118
-
119
- // Old firmware did not support version, assume it's Z-Stack 1.2 for now.
120
- try {
121
- this.version = (await this.znp.requestWithReply(Subsystem.SYS, "version", {})).payload as typeof this.version;
122
- } catch {
123
- logger.debug("Failed to get zStack version, assuming 1.2", NS);
124
- this.version = {transportrev: 2, product: 0, majorrel: 2, minorrel: 0, maintrel: 0, revision: ""};
125
- }
126
-
127
- const concurrent = this.adapterOptions?.concurrent ? this.adapterOptions.concurrent : this.version.product === ZnpVersion.ZStack3x0 ? 16 : 2;
128
-
129
- logger.debug(`Adapter concurrent: ${concurrent}`, NS);
130
-
131
- this.queue = new Queue(concurrent);
132
-
133
- logger.debug(`Detected znp version '${ZnpVersion[this.version.product]}' (${JSON.stringify(this.version)})`, NS);
134
- this.adapterManager = new ZnpAdapterManager(this, this.znp, {
135
- backupPath: this.backupPath,
136
- version: this.version.product,
137
- greenPowerGroup: ZSpec.GP_GROUP_ID,
138
- networkOptions: this.networkOptions,
139
- adapterOptions: this.adapterOptions,
140
- });
141
-
142
- const startResult = this.adapterManager.start();
143
-
144
- if (this.adapterOptions.disableLED) {
145
- // Wait a bit for adapter to startup, otherwise led doesn't disable (tested with CC2531)
146
- await wait(200);
147
- await this.setLED("disable");
148
- }
149
-
150
- if (this.adapterOptions.transmitPower != null) {
151
- await this.znp.request(Subsystem.SYS, "stackTune", {operation: 0, value: this.adapterOptions.transmitPower});
152
- }
153
-
154
- return await startResult;
155
- }
156
-
157
- public async stop(): Promise<void> {
158
- this.closing = true;
159
- await this.znp.close();
160
- }
161
-
162
- public async getCoordinatorIEEE(): Promise<string> {
163
- return await this.queue.execute(async () => {
164
- this.checkInterpanLock();
165
- const deviceInfo = await this.znp.requestWithReply(Subsystem.UTIL, "getDeviceInfo", {});
166
-
167
- return deviceInfo.payload.ieeeaddr;
168
- });
169
- }
170
-
171
- public async getCoordinatorVersion(): Promise<CoordinatorVersion> {
172
- return await Promise.resolve({type: ZnpVersion[this.version.product], meta: this.version});
173
- }
174
-
175
- public async permitJoin(seconds: number, networkAddress?: number): Promise<void> {
176
- const clusterId = Zdo.ClusterId.PERMIT_JOINING_REQUEST;
177
- // `authentication`: TC significance always 1 (zb specs)
178
- const zdoPayload = Zdo.Buffalo.buildRequest(this.hasZdoMessageOverhead, clusterId, seconds, 1, []);
179
-
180
- if (networkAddress === undefined) {
181
- await this.sendZdo(ZSpec.BLANK_EUI64, ZSpec.BroadcastAddress.DEFAULT, clusterId, zdoPayload, true);
182
- } else {
183
- // NOTE: `sendZdo` takes care of adjusting the payload as appropriate based on `networkAddress === 0` or not
184
- const result = await this.sendZdo(ZSpec.BLANK_EUI64, networkAddress, clusterId, zdoPayload, false);
185
-
186
- /* v8 ignore start */
187
- if (!Zdo.Buffalo.checkStatus<Zdo.ClusterId.PERMIT_JOINING_RESPONSE>(result)) {
188
- // TODO: will disappear once moved upstream
189
- throw new Zdo.StatusError(result[0]);
190
- }
191
- /* v8 ignore stop */
192
- }
193
-
194
- await this.queue.execute<void>(async () => {
195
- this.checkInterpanLock();
196
- await this.setLED(seconds === 0 ? "off" : "on");
197
- });
198
- }
199
-
200
- public async reset(type: "soft" | "hard"): Promise<void> {
201
- if (type === "soft") {
202
- await this.znp.request(Subsystem.SYS, "resetReq", {type: Constants.SYS.resetType.SOFT});
203
- } else {
204
- await this.znp.request(Subsystem.SYS, "resetReq", {type: Constants.SYS.resetType.HARD});
205
- }
206
- }
207
-
208
- private async setLED(action: "disable" | "on" | "off"): Promise<void> {
209
- if (this.supportsLED == null) {
210
- // Only zStack3x0 with 20210430 and greater support LED
211
- const zStack3x0 = this.version.product === ZnpVersion.ZStack3x0;
212
- this.supportsLED = !zStack3x0 || (zStack3x0 && Number.parseInt(this.version.revision, 10) >= 20210430);
213
- }
214
-
215
- if (!this.supportsLED || (this.adapterOptions.disableLED && action !== "disable")) {
216
- return;
217
- }
218
-
219
- // Firmwares build on and after 20211029 should handle LED themselves
220
- const firmwareControlsLed = Number.parseInt(this.version.revision, 10) >= 20211029;
221
- const lookup = {
222
- disable: firmwareControlsLed ? {ledid: 0xff, mode: 5} : {ledid: 3, mode: 0},
223
- on: firmwareControlsLed ? null : {ledid: 3, mode: 1},
224
- off: firmwareControlsLed ? null : {ledid: 3, mode: 0},
225
- };
226
-
227
- const payload = lookup[action];
228
- if (payload) {
229
- await this.znp.request(Subsystem.UTIL, "ledControl", payload, undefined, 500).catch(() => {
230
- // We cannot 100% correctly determine if an adapter supports LED. E.g. the zStack 1.2 20190608
231
- // fw supports led on the CC2531 but not on the CC2530. Therefore if a led request fails never thrown
232
- // an error but instead mark the led as unsupported.
233
- // https://github.com/Koenkk/zigbee-herdsman/issues/377
234
- // https://github.com/Koenkk/zigbee2mqtt/issues/7693
235
- this.supportsLED = false;
236
- });
237
- }
238
- }
239
-
240
- private async requestNetworkAddress(ieeeAddr: string): Promise<number> {
241
- /**
242
- * NOTE: There are cases where multiple nwkAddrRsp are recevied with different network addresses,
243
- * this is currently not handled, the first nwkAddrRsp is taken.
244
- */
245
- logger.debug(`Request network address of '${ieeeAddr}'`, NS);
246
-
247
- const clusterId = Zdo.ClusterId.NETWORK_ADDRESS_REQUEST;
248
- const zdoPayload = Zdo.Buffalo.buildRequest(this.hasZdoMessageOverhead, clusterId, ieeeAddr as Eui64, false, 0);
249
-
250
- const result = await this.sendZdoInternal(ieeeAddr, ZSpec.NULL_NODE_ID, clusterId, zdoPayload, false, true);
251
-
252
- if (Zdo.Buffalo.checkStatus<Zdo.ClusterId.NETWORK_ADDRESS_RESPONSE>(result)) {
253
- return result[1].nwkAddress;
254
- /* v8 ignore start */
255
- }
256
-
257
- // TODO: will disappear once moved upstream
258
- throw new Zdo.StatusError(result[0]);
259
- /* v8 ignore stop */
260
- }
261
-
262
- private supportsAssocRemove(): boolean {
263
- return this.version.product === ZnpVersion.ZStack3x0 && Number.parseInt(this.version.revision, 10) >= 20200805;
264
- }
265
-
266
- private supportsAssocAdd(): boolean {
267
- return this.version.product === ZnpVersion.ZStack3x0 && Number.parseInt(this.version.revision, 10) >= 20201026;
268
- }
269
-
270
- private async discoverRoute(networkAddress: number, waitSettled = true): Promise<void> {
271
- logger.debug(`Discovering route to ${networkAddress}`, NS);
272
- const payload = {dstAddr: networkAddress, options: 0, radius: Constants.AF.DEFAULT_RADIUS};
273
- await this.znp.request(Subsystem.ZDO, "extRouteDisc", payload);
274
-
275
- if (waitSettled) {
276
- await wait(3000);
277
- }
278
- }
279
-
280
- public async sendZdo(
281
- ieeeAddress: string,
282
- networkAddress: number,
283
- clusterId: Zdo.ClusterId,
284
- payload: Buffer,
285
- disableResponse: true,
286
- ): Promise<void>;
287
- public async sendZdo<K extends keyof ZdoTypes.RequestToResponseMap>(
288
- ieeeAddress: string,
289
- networkAddress: number,
290
- clusterId: K,
291
- payload: Buffer,
292
- disableResponse: false,
293
- ): Promise<ZdoTypes.RequestToResponseMap[K]>;
294
- public async sendZdo<K extends keyof ZdoTypes.RequestToResponseMap>(
295
- ieeeAddress: string,
296
- networkAddress: number,
297
- clusterId: K,
298
- payload: Buffer,
299
- disableResponse: boolean,
300
- ): Promise<ZdoTypes.RequestToResponseMap[K] | undefined> {
301
- return await this.sendZdoInternal(ieeeAddress, networkAddress, clusterId, payload, disableResponse, false);
302
- }
303
-
304
- private async sendZdoInternal(
305
- ieeeAddress: string,
306
- networkAddress: number,
307
- clusterId: Zdo.ClusterId,
308
- payload: Buffer,
309
- disableResponse: boolean,
310
- skipQueue: boolean,
311
- ): Promise<undefined>;
312
- private async sendZdoInternal<K extends keyof ZdoTypes.RequestToResponseMap>(
313
- ieeeAddress: string,
314
- networkAddress: number,
315
- clusterId: K,
316
- payload: Buffer,
317
- disableResponse: false,
318
- skipQueue: boolean,
319
- ): Promise<ZdoTypes.RequestToResponseMap[K]>;
320
- private async sendZdoInternal<K extends keyof ZdoTypes.RequestToResponseMap>(
321
- ieeeAddress: string,
322
- networkAddress: number,
323
- clusterId: K,
324
- payload: Buffer,
325
- disableResponse: boolean,
326
- skipQueue: boolean,
327
- ): Promise<ZdoTypes.RequestToResponseMap[K] | undefined> {
328
- const func = async (): Promise<ZdoTypes.RequestToResponseMap[K] | undefined> => {
329
- this.checkInterpanLock();
330
-
331
- // stack-specific requirements
332
- switch (clusterId) {
333
- case Zdo.ClusterId.PERMIT_JOINING_REQUEST: {
334
- const finalPayload = Buffer.alloc(payload.length + 3);
335
- finalPayload.writeUInt8(ZSpec.BroadcastAddress[networkAddress] ? AddressMode.ADDR_BROADCAST : AddressMode.ADDR_16BIT, 0);
336
- // zstack uses AddressMode.ADDR_16BIT + ZSpec.BroadcastAddress.DEFAULT to signal "coordinator-only"
337
- finalPayload.writeUInt16LE(networkAddress === 0 ? ZSpec.BroadcastAddress.DEFAULT : networkAddress, 1);
338
- finalPayload.set(payload, 3);
339
-
340
- payload = finalPayload;
341
- break;
342
- }
343
-
344
- case Zdo.ClusterId.NWK_UPDATE_REQUEST: {
345
- // extra zeroes for empty nwkManagerAddr if necessary
346
- const zeroes = 9 - payload.length - 1; /* zstack doesn't have nwkUpdateId */
347
- const finalPayload = Buffer.alloc(payload.length + 3 + zeroes);
348
- finalPayload.writeUInt16LE(networkAddress, 0);
349
- finalPayload.writeUInt8(ZSpec.BroadcastAddress[networkAddress] ? AddressMode.ADDR_BROADCAST : AddressMode.ADDR_16BIT, 2);
350
- finalPayload.set(payload, 3);
351
-
352
- payload = finalPayload;
353
- break;
354
- }
355
-
356
- case Zdo.ClusterId.BIND_REQUEST:
357
- case Zdo.ClusterId.UNBIND_REQUEST: {
358
- // extra zeroes for uint16 (in place of ieee when MULTICAST) and endpoint (not used when MULTICAST)
359
- const zeroes = 21 - payload.length;
360
- const finalPayload = Buffer.alloc(payload.length + 2 + zeroes);
361
- finalPayload.writeUInt16LE(networkAddress, 0);
362
- finalPayload.set(payload, 2);
363
-
364
- payload = finalPayload;
365
- break;
366
- }
367
-
368
- case Zdo.ClusterId.NETWORK_ADDRESS_REQUEST:
369
- case Zdo.ClusterId.IEEE_ADDRESS_REQUEST: {
370
- // no modification necessary
371
- break;
372
- }
373
-
374
- default: {
375
- const finalPayload = Buffer.alloc(payload.length + 2);
376
- finalPayload.writeUInt16LE(networkAddress, 0);
377
- finalPayload.set(payload, 2);
378
-
379
- payload = finalPayload;
380
- break;
381
- }
382
- }
383
-
384
- let waiter: ReturnType<typeof this.znp.waitFor> | undefined;
385
-
386
- if (!disableResponse) {
387
- const responseClusterId = Zdo.Utils.getResponseClusterId(clusterId);
388
-
389
- if (responseClusterId) {
390
- const cmd = Definition[Subsystem.ZDO].find((c) => isMtCmdAreqZdo(c) && c.zdoClusterId === responseClusterId);
391
- assert(cmd, `Response for ZDO cluster ID '${responseClusterId}' not supported.`);
392
-
393
- waiter = this.znp.waitFor(
394
- UnpiConstants.Type.AREQ,
395
- Subsystem.ZDO,
396
- cmd.name,
397
- responseClusterId === Zdo.ClusterId.NETWORK_ADDRESS_RESPONSE ? ieeeAddress : networkAddress,
398
- undefined,
399
- undefined,
400
- );
401
- }
402
- }
403
-
404
- try {
405
- await this.znp.requestZdo(clusterId, payload, waiter?.ID);
406
- } catch (error) {
407
- if (clusterId === Zdo.ClusterId.NODE_DESCRIPTOR_REQUEST) {
408
- // Discover route when node descriptor request fails
409
- // https://github.com/Koenkk/zigbee2mqtt/issues/3276
410
- logger.debug(`Discover route to '${networkAddress}' because node descriptor request failed`, NS);
411
- await this.discoverRoute(networkAddress);
412
- await this.znp.requestZdo(clusterId, payload, /* v8 ignore next */ waiter?.ID);
413
- } else {
414
- throw error;
415
- }
416
- }
417
-
418
- if (waiter) {
419
- const response = await waiter.start().promise;
420
-
421
- return response.payload.zdo;
422
- }
423
- };
424
- return skipQueue ? await func() : await this.queue.execute(func, networkAddress);
425
- }
426
-
427
- public async sendZclFrameToEndpoint(
428
- ieeeAddr: string,
429
- networkAddress: number,
430
- endpoint: number,
431
- zclFrame: Zcl.Frame,
432
- timeout: number,
433
- disableResponse: boolean,
434
- disableRecovery: boolean,
435
- sourceEndpoint?: number,
436
- ): Promise<Events.ZclPayload | undefined> {
437
- return await this.queue.execute<Events.ZclPayload | undefined>(async () => {
438
- this.checkInterpanLock();
439
- return await this.sendZclFrameToEndpointInternal(
440
- ieeeAddr,
441
- networkAddress,
442
- endpoint,
443
- sourceEndpoint || 1,
444
- zclFrame,
445
- timeout,
446
- disableResponse,
447
- disableRecovery,
448
- 0,
449
- 0,
450
- false,
451
- false,
452
- false,
453
- undefined,
454
- );
455
- }, networkAddress);
456
- }
457
-
458
- private async sendZclFrameToEndpointInternal(
459
- ieeeAddr: string,
460
- networkAddress: number,
461
- endpoint: number,
462
- sourceEndpoint: number,
463
- zclFrame: Zcl.Frame,
464
- timeout: number,
465
- disableResponse: boolean,
466
- disableRecovery: boolean,
467
- responseAttempt: number,
468
- dataRequestAttempt: number,
469
- checkedNetworkAddress: boolean,
470
- discoveredRoute: boolean,
471
- assocRemove: boolean,
472
- assocRestore?: {ieeeadr: string; nwkaddr: number; noderelation: number},
473
- ): Promise<Events.ZclPayload | undefined> {
474
- logger.debug(
475
- `sendZclFrameToEndpointInternal ${ieeeAddr}:${networkAddress}/${endpoint} ` +
476
- `(${responseAttempt},${dataRequestAttempt},${this.queue.count()})`,
477
- NS,
478
- );
479
- let response = null;
480
- const command = zclFrame.command;
481
- if (command.response !== undefined && disableResponse === false) {
482
- response = this.waitForInternal(
483
- networkAddress,
484
- endpoint,
485
- zclFrame.header.frameControl.frameType,
486
- Zcl.Direction.SERVER_TO_CLIENT,
487
- zclFrame.header.transactionSequenceNumber,
488
- zclFrame.cluster.ID,
489
- command.response,
490
- timeout,
491
- );
492
- } else if (!zclFrame.header.frameControl.disableDefaultResponse) {
493
- response = this.waitForInternal(
494
- networkAddress,
495
- endpoint,
496
- Zcl.FrameType.GLOBAL,
497
- Zcl.Direction.SERVER_TO_CLIENT,
498
- zclFrame.header.transactionSequenceNumber,
499
- zclFrame.cluster.ID,
500
- Zcl.Foundation.defaultRsp.ID,
501
- timeout,
502
- );
503
- }
504
-
505
- const dataConfirmResult = await this.dataRequest(
506
- networkAddress,
507
- endpoint,
508
- sourceEndpoint,
509
- zclFrame.cluster.ID,
510
- Constants.AF.DEFAULT_RADIUS,
511
- zclFrame.toBuffer(),
512
- timeout,
513
- );
514
-
515
- if (dataConfirmResult !== ZnpCommandStatus.SUCCESS) {
516
- // In case dataConfirm timesout (= null) or gives an error, try to recover
517
- logger.debug(`Data confirm error (${ieeeAddr}:${networkAddress},${dataConfirmResult},${dataRequestAttempt})`, NS);
518
- if (response !== null) response.cancel();
519
-
520
- /**
521
- * In case we did an assocRemove in the previous attempt and it still fails after this, assume that the
522
- * coordinator is still the parent of the device (but for some reason the device is not available now).
523
- * Re-add the device to the assoc table, otherwise we will never be able to reach it anymore.
524
- */
525
- if (assocRemove && assocRestore && this.supportsAssocAdd()) {
526
- logger.debug(`assocAdd(${assocRestore.ieeeadr})`, NS);
527
- await this.znp.request(Subsystem.UTIL, "assocAdd", assocRestore);
528
- assocRestore = undefined;
529
- }
530
-
531
- const recoverableErrors = [
532
- ZnpCommandStatus.NWK_NO_ROUTE,
533
- ZnpCommandStatus.MAC_NO_ACK,
534
- ZnpCommandStatus.MAC_CHANNEL_ACCESS_FAILURE,
535
- ZnpCommandStatus.MAC_TRANSACTION_EXPIRED,
536
- ZnpCommandStatus.BUFFER_FULL,
537
- ZnpCommandStatus.MAC_NO_RESOURCES,
538
- ];
539
-
540
- if (dataRequestAttempt >= 4 || !recoverableErrors.includes(dataConfirmResult) || disableRecovery) {
541
- throw new DataConfirmError(dataConfirmResult);
542
- }
543
-
544
- if (
545
- dataConfirmResult === ZnpCommandStatus.MAC_CHANNEL_ACCESS_FAILURE ||
546
- dataConfirmResult === ZnpCommandStatus.BUFFER_FULL ||
547
- dataConfirmResult === ZnpCommandStatus.MAC_NO_RESOURCES
548
- ) {
549
- /**
550
- * MAC_CHANNEL_ACCESS_FAILURE: When many commands at once are executed we can end up in a MAC
551
- * channel access failure error. This is because there is too much traffic on the network.
552
- * Retry this command once after a cooling down period.
553
- * BUFFER_FULL: When many commands are executed at once the buffer can get full, wait
554
- * some time and retry.
555
- * MAC_NO_RESOURCES: Operation could not be completed because no memory resources are available,
556
- * wait some time and retry.
557
- */
558
- await wait(2000);
559
- return await this.sendZclFrameToEndpointInternal(
560
- ieeeAddr,
561
- networkAddress,
562
- endpoint,
563
- sourceEndpoint,
564
- zclFrame,
565
- timeout,
566
- disableResponse,
567
- disableRecovery,
568
- responseAttempt,
569
- dataRequestAttempt + 1,
570
- checkedNetworkAddress,
571
- discoveredRoute,
572
- assocRemove,
573
- assocRestore,
574
- );
575
- }
576
-
577
- let doAssocRemove = false;
578
- if (
579
- !assocRemove &&
580
- dataConfirmResult === ZnpCommandStatus.MAC_TRANSACTION_EXPIRED &&
581
- dataRequestAttempt >= 1 &&
582
- this.supportsAssocRemove()
583
- ) {
584
- const match = await this.znp.requestWithReply(Subsystem.UTIL, "assocGetWithAddress", {
585
- extaddr: ieeeAddr,
586
- nwkaddr: networkAddress,
587
- });
588
-
589
- if (match.payload.nwkaddr !== 0xfffe && match.payload.noderelation !== 255) {
590
- doAssocRemove = true;
591
- assocRestore = {ieeeadr: ieeeAddr, nwkaddr: networkAddress, noderelation: match.payload.noderelation};
592
- }
593
-
594
- assocRemove = true;
595
- }
596
-
597
- // NWK_NO_ROUTE: no network route => rediscover route
598
- // MAC_NO_ACK: route may be corrupted
599
- // MAC_TRANSACTION_EXPIRED: Mac layer is sleeping
600
- if (doAssocRemove) {
601
- /**
602
- * Since child aging is disabled on the firmware, when a end device is directly connected
603
- * to the coordinator and changes parent and the coordinator does not recevie this update,
604
- * it still thinks it's directly connected.
605
- * A discoverRoute() is not send out in this case, therefore remove it from the associated device
606
- * list and try again.
607
- * Note: assocRemove is a custom command, not available by default, only available on recent
608
- * z-stack-firmware firmware version. In case it's not supported by the coordinator we will
609
- * automatically timeout after 60000ms.
610
- */
611
- logger.debug(`assocRemove(${ieeeAddr})`, NS);
612
- await this.znp.request(Subsystem.UTIL, "assocRemove", {ieeeadr: ieeeAddr});
613
- } else if (!discoveredRoute && dataRequestAttempt >= 1) {
614
- discoveredRoute = true;
615
- await this.discoverRoute(networkAddress);
616
- } else if (!checkedNetworkAddress && dataRequestAttempt >= 1) {
617
- // Figure out once if the network address has been changed.
618
- try {
619
- checkedNetworkAddress = true;
620
- const actualNetworkAddress = await this.requestNetworkAddress(ieeeAddr);
621
- if (networkAddress !== actualNetworkAddress) {
622
- logger.debug("Failed because request was done with wrong network address", NS);
623
- discoveredRoute = true;
624
- networkAddress = actualNetworkAddress;
625
- await this.discoverRoute(actualNetworkAddress);
626
- } else {
627
- logger.debug("Network address did not change", NS);
628
- }
629
- /* v8 ignore start */
630
- } catch {
631
- /* empty */
632
- }
633
- /* v8 ignore stop */
634
- } else {
635
- logger.debug("Wait 2000ms", NS);
636
- await wait(2000);
637
- }
638
-
639
- return await this.sendZclFrameToEndpointInternal(
640
- ieeeAddr,
641
- networkAddress,
642
- endpoint,
643
- sourceEndpoint,
644
- zclFrame,
645
- timeout,
646
- disableResponse,
647
- disableRecovery,
648
- responseAttempt,
649
- dataRequestAttempt + 1,
650
- checkedNetworkAddress,
651
- discoveredRoute,
652
- assocRemove,
653
- assocRestore,
654
- );
655
- }
656
-
657
- if (response !== null) {
658
- try {
659
- const result = await response.start().promise;
660
- return result;
661
- } catch (error) {
662
- logger.debug(`Response timeout (${ieeeAddr}:${networkAddress},${responseAttempt})`, NS);
663
- if (responseAttempt < 1 && !disableRecovery) {
664
- // No response could be because the radio of the end device is turned off:
665
- // Sometimes the coordinator does not properly set the PENDING flag.
666
- // Try to rewrite the device entry in the association table, this fixes it sometimes.
667
- const match = await this.znp.requestWithReply(Subsystem.UTIL, "assocGetWithAddress", {
668
- extaddr: ieeeAddr,
669
- nwkaddr: networkAddress,
670
- });
671
- logger.debug(
672
- `Response timeout recovery: Node relation ${match.payload.noderelation} (${ieeeAddr} / ${match.payload.nwkaddr})`,
673
- NS,
674
- );
675
- if (
676
- this.supportsAssocAdd() &&
677
- this.supportsAssocRemove() &&
678
- match.payload.nwkaddr !== 0xfffe &&
679
- match.payload.noderelation === 1
680
- ) {
681
- logger.debug(`Response timeout recovery: Rewrite association table entry (${ieeeAddr})`, NS);
682
- await this.znp.request(Subsystem.UTIL, "assocRemove", {ieeeadr: ieeeAddr});
683
- await this.znp.request(Subsystem.UTIL, "assocAdd", {
684
- ieeeadr: ieeeAddr,
685
- nwkaddr: networkAddress,
686
- noderelation: match.payload.noderelation,
687
- });
688
- }
689
- // No response could be of invalid route, e.g. when message is send to wrong parent of end device.
690
- await this.discoverRoute(networkAddress);
691
- return await this.sendZclFrameToEndpointInternal(
692
- ieeeAddr,
693
- networkAddress,
694
- endpoint,
695
- sourceEndpoint,
696
- zclFrame,
697
- timeout,
698
- disableResponse,
699
- disableRecovery,
700
- responseAttempt + 1,
701
- dataRequestAttempt,
702
- checkedNetworkAddress,
703
- discoveredRoute,
704
- assocRemove,
705
- assocRestore,
706
- );
707
- }
708
-
709
- throw error;
710
- }
711
- }
712
- }
713
-
714
- public async sendZclFrameToGroup(groupID: number, zclFrame: Zcl.Frame, sourceEndpoint?: number): Promise<void> {
715
- return await this.queue.execute<void>(async () => {
716
- this.checkInterpanLock();
717
- await this.dataRequestExtended(
718
- AddressMode.ADDR_GROUP,
719
- groupID,
720
- 0xff,
721
- 0,
722
- sourceEndpoint || 1,
723
- zclFrame.cluster.ID,
724
- Constants.AF.DEFAULT_RADIUS,
725
- zclFrame.toBuffer(),
726
- 3000,
727
- true,
728
- );
729
-
730
- /**
731
- * As a group command is not confirmed and thus immidiately returns
732
- * (contrary to network address requests) we will give the
733
- * command some time to 'settle' in the network.
734
- */
735
- await wait(200);
736
- });
737
- }
738
-
739
- public async sendZclFrameToAll(endpoint: number, zclFrame: Zcl.Frame, sourceEndpoint: number, destination: BroadcastAddress): Promise<void> {
740
- return await this.queue.execute<void>(async () => {
741
- this.checkInterpanLock();
742
- await this.dataRequestExtended(
743
- AddressMode.ADDR_16BIT,
744
- destination,
745
- endpoint,
746
- 0,
747
- sourceEndpoint,
748
- zclFrame.cluster.ID,
749
- Constants.AF.DEFAULT_RADIUS,
750
- zclFrame.toBuffer(),
751
- 3000,
752
- false,
753
- 0,
754
- );
755
-
756
- /**
757
- * As a broadcast command is not confirmed and thus immidiately returns
758
- * (contrary to network address requests) we will give the
759
- * command some time to 'settle' in the network.
760
- */
761
- await wait(200);
762
- });
763
- }
764
-
765
- public async addInstallCode(ieeeAddress: string, key: Buffer, hashed: boolean): Promise<void> {
766
- assert(this.version.product !== ZnpVersion.ZStack12, "Install code is not supported for ZStack 1.2 adapter");
767
- // TODO: always use 0x2? => const hashedKey = hashed ? key : ZSpec.Utils.aes128MmoHash(key);
768
- const payload = {installCodeFormat: hashed ? 0x2 : 0x1, ieeeaddr: ieeeAddress, installCode: key};
769
- await this.znp.request(Subsystem.APP_CNF, "bdbAddInstallCode", payload);
770
- }
771
-
772
- /**
773
- * Event handlers
774
- */
775
- public onZnpClose(): void {
776
- if (!this.closing) {
777
- this.emit("disconnected");
778
- }
779
- }
780
-
781
- private onZnpRecieved(object: ZpiObject): void {
782
- if (object.type !== UnpiConstants.Type.AREQ) {
783
- return;
784
- }
785
-
786
- if (object.subsystem === Subsystem.ZDO) {
787
- if (isMtCmdAreqZdo(object.command)) {
788
- this.emit("zdoResponse", object.command.zdoClusterId, object.payload.zdo);
789
- }
790
-
791
- if (object.command.name === "tcDeviceInd") {
792
- const payload: Events.DeviceJoinedPayload = {
793
- networkAddress: object.payload.nwkaddr,
794
- ieeeAddr: object.payload.extaddr,
795
- };
796
-
797
- this.emit("deviceJoined", payload);
798
- } else if (object.command.name === "endDeviceAnnceInd") {
799
- // TODO: better way???
800
- if (Zdo.Buffalo.checkStatus<Zdo.ClusterId.END_DEVICE_ANNOUNCE>(object.payload.zdo)) {
801
- const zdoPayload = object.payload.zdo[1];
802
- // Only discover routes to end devices, if bit 1 of capabilities === 0 it's an end device.
803
- const isEndDevice = zdoPayload.capabilities.deviceType === 0;
804
-
805
- if (isEndDevice) {
806
- if (!this.deviceAnnounceRouteDiscoveryDebouncers.has(zdoPayload.nwkAddress)) {
807
- // If a device announces multiple times in a very short time, it makes no sense
808
- // to rediscover the route every time.
809
- const debouncer = debounce(
810
- () => {
811
- this.queue.execute<void>(async () => {
812
- await this.discoverRoute(zdoPayload.nwkAddress, false).catch(() => {});
813
- }, zdoPayload.nwkAddress);
814
- },
815
- 60 * 1000,
816
- {immediate: true},
817
- );
818
- this.deviceAnnounceRouteDiscoveryDebouncers.set(zdoPayload.nwkAddress, debouncer);
819
- }
820
-
821
- const debouncer = this.deviceAnnounceRouteDiscoveryDebouncers.get(zdoPayload.nwkAddress);
822
- assert(debouncer);
823
- debouncer();
824
- }
825
- }
826
- } else if (object.command.name === "concentratorIndCb") {
827
- // Some routers may change short addresses and the announcement
828
- // is missed by the coordinator. This can happen when there are
829
- // power outages or other interruptions in service. They may
830
- // not send additional announcements, causing the device to go
831
- // offline. However, those devices may instead send
832
- // Concentrator Indicator Callback commands, which contain both
833
- // the short and the long address allowing us to update our own
834
- // mappings.
835
- // https://e2e.ti.com/cfs-file/__key/communityserver-discussions-components-files/158/4403.zstacktask.c
836
- // https://github.com/Koenkk/zigbee-herdsman/issues/74
837
- this.emit("zdoResponse", Zdo.ClusterId.NETWORK_ADDRESS_RESPONSE, [
838
- Zdo.Status.SUCCESS,
839
- {
840
- eui64: object.payload.extaddr,
841
- nwkAddress: object.payload.srcaddr,
842
- startIndex: 0,
843
- assocDevList: [],
844
- } as ZdoTypes.NetworkAddressResponse,
845
- ]);
846
- } else {
847
- if (object.command.name === "leaveInd") {
848
- if (object.payload.rejoin) {
849
- logger.debug("Device leave: Got leave indication with rejoin=true, nothing to do", NS);
850
- } else {
851
- const payload: Events.DeviceLeavePayload = {
852
- networkAddress: object.payload.srcaddr,
853
- ieeeAddr: object.payload.extaddr,
854
- };
855
-
856
- this.emit("deviceLeave", payload);
857
- }
858
- }
859
- }
860
- } else {
861
- if (object.subsystem === Subsystem.AF) {
862
- if (object.command.name === "incomingMsg" || object.command.name === "incomingMsgExt") {
863
- const payload: Events.ZclPayload = {
864
- clusterID: object.payload.clusterid,
865
- data: object.payload.data,
866
- header: Zcl.Header.fromBuffer(object.payload.data),
867
- address: object.payload.srcaddr,
868
- endpoint: object.payload.srcendpoint,
869
- linkquality: object.payload.linkquality,
870
- groupID: object.payload.groupid,
871
- wasBroadcast: object.payload.wasbroadcast === 1,
872
- destinationEndpoint: object.payload.dstendpoint,
873
- };
874
-
875
- this.waitress.resolve(payload);
876
- this.emit("zclPayload", payload);
877
- }
878
- }
879
- }
880
- }
881
-
882
- public async getNetworkParameters(): Promise<NetworkParameters> {
883
- const result = await this.znp.requestWithReply(Subsystem.ZDO, "extNwkInfo", {});
884
- return {
885
- panID: result.payload.panid as number,
886
- extendedPanID: result.payload.extendedpanid as string, // read as IEEEADDR, so `0x${string}`
887
- channel: result.payload.channel as number,
888
- /**
889
- * Return a dummy nwkUpdateId of 0, the nwkUpdateId is used when changing channels however the
890
- * zstack API does not allow to set this value. Instead it automatically increments the nwkUpdateId
891
- * based on the value in the NIB.
892
- * https://github.com/Koenkk/zigbee-herdsman/pull/1280#discussion_r1947815987
893
- */
894
- nwkUpdateID: 0,
895
- };
896
- }
897
-
898
- public async supportsBackup(): Promise<boolean> {
899
- return await Promise.resolve(true);
900
- }
901
-
902
- public async backup(ieeeAddressesInDatabase: string[]): Promise<Models.Backup> {
903
- return await this.adapterManager.backup.createBackup(ieeeAddressesInDatabase);
904
- }
905
-
906
- public async setChannelInterPAN(channel: number): Promise<void> {
907
- return await this.queue.execute<void>(async () => {
908
- this.interpanLock = true;
909
- await this.znp.request(Subsystem.AF, "interPanCtl", {cmd: 1, data: [channel]});
910
-
911
- if (!this.interpanEndpointRegistered) {
912
- // Make sure that endpoint 12 is registered to proxy the InterPAN messages.
913
- await this.znp.request(Subsystem.AF, "interPanCtl", {cmd: 2, data: [12]});
914
- this.interpanEndpointRegistered = true;
915
- }
916
- });
917
- }
918
-
919
- public async sendZclFrameInterPANToIeeeAddr(zclFrame: Zcl.Frame, ieeeAddr: string): Promise<void> {
920
- return await this.queue.execute<void>(async () => {
921
- await this.dataRequestExtended(
922
- AddressMode.ADDR_64BIT,
923
- ieeeAddr,
924
- 0xfe,
925
- 0xffff,
926
- 12,
927
- zclFrame.cluster.ID,
928
- 30,
929
- zclFrame.toBuffer(),
930
- 10000,
931
- false,
932
- );
933
- });
934
- }
935
-
936
- public async sendZclFrameInterPANBroadcast(zclFrame: Zcl.Frame, timeout: number): Promise<Events.ZclPayload> {
937
- return await this.queue.execute<Events.ZclPayload>(async () => {
938
- const command = zclFrame.command;
939
- if (command.response === undefined) {
940
- throw new Error(`Command '${command.name}' has no response, cannot wait for response`);
941
- }
942
-
943
- const response = this.waitForInternal(
944
- undefined,
945
- 0xfe,
946
- zclFrame.header.frameControl.frameType,
947
- Zcl.Direction.SERVER_TO_CLIENT,
948
- undefined,
949
- zclFrame.cluster.ID,
950
- command.response,
951
- timeout,
952
- );
953
-
954
- try {
955
- await this.dataRequestExtended(
956
- AddressMode.ADDR_16BIT,
957
- 0xffff,
958
- 0xfe,
959
- 0xffff,
960
- 12,
961
- zclFrame.cluster.ID,
962
- 30,
963
- zclFrame.toBuffer(),
964
- 10000,
965
- false,
966
- );
967
- } catch (error) {
968
- response.cancel();
969
- throw error;
970
- }
971
-
972
- return await response.start().promise;
973
- });
974
- }
975
-
976
- public async restoreChannelInterPAN(): Promise<void> {
977
- return await this.queue.execute<void>(async () => {
978
- await this.znp.request(Subsystem.AF, "interPanCtl", {cmd: 0, data: []});
979
- // Give adapter some time to restore, otherwise stuff crashes
980
- await wait(3000);
981
- this.interpanLock = false;
982
- });
983
- }
984
-
985
- private waitForInternal(
986
- networkAddress: number | undefined,
987
- endpoint: number,
988
- frameType: Zcl.FrameType,
989
- direction: Zcl.Direction,
990
- transactionSequenceNumber: number | undefined,
991
- clusterID: number,
992
- commandIdentifier: number,
993
- timeout: number,
994
- ): {start: () => {promise: Promise<Events.ZclPayload>}; cancel: () => void} {
995
- const payload = {
996
- address: networkAddress,
997
- endpoint,
998
- clusterID,
999
- commandIdentifier,
1000
- frameType,
1001
- direction,
1002
- transactionSequenceNumber,
1003
- };
1004
-
1005
- const waiter = this.waitress.waitFor(payload, timeout);
1006
- const cancel = (): void => this.waitress.remove(waiter.ID);
1007
- return {start: waiter.start, cancel};
1008
- }
1009
-
1010
- public waitFor(
1011
- networkAddress: number | undefined,
1012
- endpoint: number,
1013
- frameType: Zcl.FrameType,
1014
- direction: Zcl.Direction,
1015
- transactionSequenceNumber: number | undefined,
1016
- clusterID: number,
1017
- commandIdentifier: number,
1018
- timeout: number,
1019
- ): {promise: Promise<Events.ZclPayload>; cancel: () => void} {
1020
- const waiter = this.waitForInternal(
1021
- networkAddress,
1022
- endpoint,
1023
- frameType,
1024
- direction,
1025
- transactionSequenceNumber,
1026
- clusterID,
1027
- commandIdentifier,
1028
- timeout,
1029
- );
1030
-
1031
- return {cancel: waiter.cancel, promise: waiter.start().promise};
1032
- }
1033
-
1034
- /**
1035
- * Private methods
1036
- */
1037
- private async dataRequest(
1038
- destinationAddress: number,
1039
- destinationEndpoint: number,
1040
- sourceEndpoint: number,
1041
- clusterID: number,
1042
- radius: number,
1043
- data: Buffer,
1044
- timeout: number,
1045
- ): Promise<number> {
1046
- const transactionID = this.nextTransactionID();
1047
- const response = this.znp.waitFor(Type.AREQ, Subsystem.AF, "dataConfirm", undefined, transactionID, undefined, timeout);
1048
-
1049
- await this.znp.request(
1050
- Subsystem.AF,
1051
- "dataRequest",
1052
- {
1053
- dstaddr: destinationAddress,
1054
- destendpoint: destinationEndpoint,
1055
- srcendpoint: sourceEndpoint,
1056
- clusterid: clusterID,
1057
- transid: transactionID,
1058
- options: 0,
1059
- radius: radius,
1060
- len: data.length,
1061
- data: data,
1062
- },
1063
- response.ID,
1064
- );
1065
-
1066
- let result = null;
1067
- try {
1068
- const dataConfirm = await response.start().promise;
1069
- result = dataConfirm.payload.status;
1070
- } catch {
1071
- result = DataConfirmTimeout;
1072
- }
1073
-
1074
- return result;
1075
- }
1076
-
1077
- private async dataRequestExtended(
1078
- addressMode: number,
1079
- destinationAddressOrGroupID: number | string,
1080
- destinationEndpoint: number,
1081
- panID: number,
1082
- sourceEndpoint: number,
1083
- clusterID: number,
1084
- radius: number,
1085
- data: Buffer,
1086
- timeout: number,
1087
- confirmation: boolean,
1088
- attemptsLeft = 5,
1089
- ): Promise<ZpiObject | undefined> {
1090
- const transactionID = this.nextTransactionID();
1091
- const response = confirmation
1092
- ? this.znp.waitFor(Type.AREQ, Subsystem.AF, "dataConfirm", undefined, transactionID, undefined, timeout)
1093
- : undefined;
1094
-
1095
- await this.znp.request(
1096
- Subsystem.AF,
1097
- "dataRequestExt",
1098
- {
1099
- dstaddrmode: addressMode,
1100
- dstaddr: this.toAddressString(destinationAddressOrGroupID),
1101
- destendpoint: destinationEndpoint,
1102
- dstpanid: panID,
1103
- srcendpoint: sourceEndpoint,
1104
- clusterid: clusterID,
1105
- transid: transactionID,
1106
- options: 0, // TODO: why was this here? Constants.AF.options.DISCV_ROUTE,
1107
- radius,
1108
- len: data.length,
1109
- data: data,
1110
- },
1111
- response?.ID,
1112
- );
1113
-
1114
- if (confirmation && response) {
1115
- const dataConfirm = await response.start().promise;
1116
- if (dataConfirm.payload.status !== ZnpCommandStatus.SUCCESS) {
1117
- if (
1118
- attemptsLeft > 0 &&
1119
- (dataConfirm.payload.status === ZnpCommandStatus.MAC_CHANNEL_ACCESS_FAILURE ||
1120
- dataConfirm.payload.status === ZnpCommandStatus.BUFFER_FULL)
1121
- ) {
1122
- /**
1123
- * 225: When many commands at once are executed we can end up in a MAC channel access failure
1124
- * error. This is because there is too much traffic on the network.
1125
- * Retry this command once after a cooling down period.
1126
- */
1127
- await wait(2000);
1128
- return await this.dataRequestExtended(
1129
- addressMode,
1130
- destinationAddressOrGroupID,
1131
- destinationEndpoint,
1132
- panID,
1133
- sourceEndpoint,
1134
- clusterID,
1135
- radius,
1136
- data,
1137
- timeout,
1138
- confirmation,
1139
- attemptsLeft - 1,
1140
- );
1141
- }
1142
-
1143
- throw new DataConfirmError(dataConfirm.payload.status);
1144
- }
1145
-
1146
- return dataConfirm;
1147
- }
1148
- }
1149
-
1150
- private nextTransactionID(): number {
1151
- this.transactionID++;
1152
-
1153
- if (this.transactionID > 255) {
1154
- this.transactionID = 1;
1155
- }
1156
-
1157
- return this.transactionID;
1158
- }
1159
-
1160
- private toAddressString(address: number | string): string {
1161
- return typeof address === "number" ? `0x${address.toString(16).padStart(16, "0")}` : address.toString();
1162
- }
1163
-
1164
- private waitressTimeoutFormatter(matcher: WaitressMatcher, timeout: number): string {
1165
- return (
1166
- `Timeout - ${matcher.address} - ${matcher.endpoint}` +
1167
- ` - ${matcher.transactionSequenceNumber} - ${matcher.clusterID}` +
1168
- ` - ${matcher.commandIdentifier} after ${timeout}ms`
1169
- );
1170
- }
1171
-
1172
- private waitressValidator(payload: Events.ZclPayload, matcher: WaitressMatcher): boolean {
1173
- return Boolean(
1174
- payload.header &&
1175
- (!matcher.address || payload.address === matcher.address) &&
1176
- payload.endpoint === matcher.endpoint &&
1177
- (!matcher.transactionSequenceNumber || payload.header.transactionSequenceNumber === matcher.transactionSequenceNumber) &&
1178
- payload.clusterID === matcher.clusterID &&
1179
- matcher.frameType === payload.header.frameControl.frameType &&
1180
- matcher.commandIdentifier === payload.header.commandIdentifier &&
1181
- matcher.direction === payload.header.frameControl.direction,
1182
- );
1183
- }
1184
-
1185
- private checkInterpanLock(): void {
1186
- if (this.interpanLock) {
1187
- throw new Error("Cannot execute command, in Inter-PAN mode");
1188
- }
1189
- }
1190
- }