zigbee-herdsman 6.0.1 → 6.0.3

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