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,2 +0,0 @@
1
- export {Znp} from "./znp";
2
- export {ZpiObject} from "./zpiObject";
@@ -1,22 +0,0 @@
1
- enum ParameterType {
2
- UINT8 = 0,
3
- UINT16 = 1,
4
- UINT32 = 2,
5
- IEEEADDR = 3,
6
-
7
- BUFFER = 4,
8
- BUFFER8 = 5,
9
- BUFFER16 = 6,
10
- BUFFER18 = 7,
11
- BUFFER32 = 8,
12
- BUFFER42 = 9,
13
- BUFFER100 = 10,
14
-
15
- LIST_UINT8 = 11,
16
- LIST_UINT16 = 12,
17
- LIST_NETWORK = 16,
18
-
19
- INT8 = 18,
20
- }
21
-
22
- export default ParameterType;
@@ -1,44 +0,0 @@
1
- import type {ClusterId as ZdoClusterId} from "../../../zspec/zdo";
2
- import type {Type as CommandType} from "../unpi/constants";
3
- import type ParameterType from "./parameterType";
4
-
5
- export type MtType = number | number[] | string | Buffer | {[s: string]: number | string}[];
6
-
7
- export interface MtParameter {
8
- name: string;
9
- parameterType: ParameterType;
10
- }
11
-
12
- interface MtCmdBase {
13
- name: string;
14
- ID: number;
15
- type: number;
16
- request: MtParameter[];
17
- response: MtParameter[];
18
- zdoClusterId: ZdoClusterId;
19
- }
20
-
21
- interface MtCmdAreq extends Omit<MtCmdBase, "response" | "zdoClusterId"> {
22
- type: CommandType.AREQ;
23
- }
24
-
25
- interface MtCmdSreq extends Omit<MtCmdBase, "zdoClusterId"> {
26
- type: CommandType.SREQ;
27
- }
28
-
29
- export interface MtCmdAreqZdo extends Omit<MtCmdBase, "request" | "response"> {
30
- type: CommandType.AREQ;
31
- }
32
-
33
- export interface MtCmdSreqZdo extends Omit<MtCmdBase, "request" | "response"> {
34
- type: CommandType.SREQ;
35
- }
36
-
37
- export type MtCmd = MtCmdAreq | MtCmdSreq | MtCmdAreqZdo | MtCmdSreqZdo;
38
- // biome-ignore lint/suspicious/noExplicitAny: API
39
- export type ZpiObjectPayload = {[s: string]: any};
40
-
41
- export interface BuffaloZnpOptions {
42
- length?: number;
43
- startIndex?: number;
44
- }
@@ -1,10 +0,0 @@
1
- import {Type} from "../unpi/constants";
2
- import type {MtCmd, MtCmdAreqZdo, MtCmdSreqZdo} from "./tstype";
3
-
4
- export function isMtCmdAreqZdo(cmd: MtCmd): cmd is MtCmdAreqZdo {
5
- return cmd.type === Type.AREQ && "zdoClusterId" in cmd;
6
- }
7
-
8
- export function isMtCmdSreqZdo(cmd: MtCmd): cmd is MtCmdSreqZdo {
9
- return cmd.type === Type.SREQ && "zdoClusterId" in cmd;
10
- }
@@ -1,342 +0,0 @@
1
- import assert from "node:assert";
2
- import events from "node:events";
3
- import {Socket} from "node:net";
4
-
5
- import {Queue, Waitress, wait} from "../../../utils";
6
- import {logger} from "../../../utils/logger";
7
- import {ClusterId as ZdoClusterId} from "../../../zspec/zdo";
8
- import {SerialPort} from "../../serialPort";
9
- import SocketPortUtils from "../../socketPortUtils";
10
- import * as Constants from "../constants";
11
- import {Frame as UnpiFrame, Parser as UnpiParser, Writer as UnpiWriter} from "../unpi";
12
- import {Subsystem, Type} from "../unpi/constants";
13
- import Definition from "./definition";
14
- import type {ZpiObjectPayload} from "./tstype";
15
- import {isMtCmdSreqZdo} from "./utils";
16
- import {ZpiObject} from "./zpiObject";
17
-
18
- const {
19
- COMMON: {ZnpCommandStatus},
20
- Utils: {statusDescription},
21
- } = Constants;
22
-
23
- const timeouts = {
24
- SREQ: 6000,
25
- reset: 30000,
26
- default: 10000,
27
- };
28
-
29
- const NS = "zh:zstack:znp";
30
-
31
- interface WaitressMatcher {
32
- type: Type;
33
- subsystem: Subsystem;
34
- command: string;
35
- target?: number | string;
36
- transid?: number;
37
- state?: number;
38
- }
39
-
40
- export class Znp extends events.EventEmitter {
41
- private path: string;
42
- private baudRate: number;
43
- private rtscts: boolean;
44
-
45
- private serialPort?: SerialPort;
46
- private socketPort?: Socket;
47
- private unpiWriter: UnpiWriter;
48
- private unpiParser: UnpiParser;
49
- private initialized: boolean;
50
- private queue: Queue;
51
- private waitress: Waitress<ZpiObject, WaitressMatcher>;
52
-
53
- public constructor(path: string, baudRate: number, rtscts: boolean) {
54
- super();
55
-
56
- this.path = path;
57
- this.baudRate = typeof baudRate === "number" ? baudRate : 115200;
58
- this.rtscts = typeof rtscts === "boolean" ? rtscts : false;
59
-
60
- this.initialized = false;
61
-
62
- this.queue = new Queue();
63
- this.waitress = new Waitress<ZpiObject, WaitressMatcher>(this.waitressValidator, this.waitressTimeoutFormatter);
64
- this.unpiWriter = new UnpiWriter();
65
- this.unpiParser = new UnpiParser();
66
- }
67
-
68
- private onUnpiParsed(frame: UnpiFrame): void {
69
- try {
70
- const object = ZpiObject.fromUnpiFrame(frame);
71
- logger.debug(() => `<-- ${object.toString(object.subsystem !== Subsystem.ZDO)}`, NS);
72
- this.waitress.resolve(object);
73
- this.emit("received", object);
74
- } catch (error) {
75
- logger.error(`Error while parsing to ZpiObject '${error}'`, NS);
76
- }
77
- }
78
-
79
- public isInitialized(): boolean {
80
- return this.initialized;
81
- }
82
-
83
- private onPortError(error: Error): void {
84
- logger.error(`Port error: ${error}`, NS);
85
- }
86
-
87
- private onPortClose(): void {
88
- logger.info("Port closed", NS);
89
- this.initialized = false;
90
- this.emit("close");
91
- }
92
-
93
- public async open(): Promise<void> {
94
- return SocketPortUtils.isTcpPath(this.path) ? await this.openSocketPort() : await this.openSerialPort();
95
- }
96
-
97
- private async openSerialPort(): Promise<void> {
98
- const options = {path: this.path, baudRate: this.baudRate, rtscts: this.rtscts, autoOpen: false};
99
-
100
- logger.info(`Opening SerialPort with ${JSON.stringify(options)}`, NS);
101
- this.serialPort = new SerialPort(options);
102
-
103
- this.unpiWriter.pipe(this.serialPort);
104
- this.serialPort.pipe(this.unpiParser);
105
- this.unpiParser.on("parsed", this.onUnpiParsed.bind(this));
106
-
107
- try {
108
- await this.serialPort.asyncOpen();
109
- logger.info("Serialport opened", NS);
110
-
111
- this.serialPort.once("close", this.onPortClose.bind(this));
112
- this.serialPort.once("error", this.onPortError.bind(this));
113
-
114
- this.initialized = true;
115
-
116
- await this.skipBootloader();
117
- } catch (error) {
118
- this.initialized = false;
119
-
120
- if (this.serialPort.isOpen) {
121
- this.serialPort.close();
122
- }
123
-
124
- throw error;
125
- }
126
- }
127
-
128
- private async openSocketPort(): Promise<void> {
129
- const info = SocketPortUtils.parseTcpPath(this.path);
130
- logger.info(`Opening TCP socket with ${info.host}:${info.port}`, NS);
131
-
132
- this.socketPort = new Socket();
133
-
134
- this.socketPort.setNoDelay(true);
135
- this.socketPort.setKeepAlive(true, 15000);
136
- this.unpiWriter.pipe(this.socketPort);
137
- this.socketPort.pipe(this.unpiParser);
138
- this.unpiParser.on("parsed", this.onUnpiParsed.bind(this));
139
-
140
- return await new Promise((resolve, reject): void => {
141
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
142
- this.socketPort!.on("connect", () => {
143
- logger.info("Socket connected", NS);
144
- });
145
- const self = this;
146
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
147
- this.socketPort!.on("ready", async () => {
148
- logger.info("Socket ready", NS);
149
- await self.skipBootloader();
150
- self.initialized = true;
151
- resolve();
152
- });
153
-
154
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
155
- this.socketPort!.once("close", this.onPortClose.bind(this));
156
-
157
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
158
- this.socketPort!.on("error", (error) => {
159
- logger.error(`Socket error ${error}`, NS);
160
- reject(new Error("Error while opening socket"));
161
- self.initialized = false;
162
- });
163
-
164
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
165
- this.socketPort!.connect(info.port, info.host);
166
- });
167
- }
168
-
169
- private async skipBootloader(): Promise<void> {
170
- try {
171
- await this.request(Subsystem.SYS, "ping", {capabilities: 1}, undefined, 250);
172
- } catch {
173
- // Skip bootloader on CC2530/CC2531
174
- // Send magic byte: https://github.com/Koenkk/zigbee2mqtt/issues/1343 to bootloader
175
- // and give ZNP 1 second to start.
176
- try {
177
- logger.info("Writing CC2530/CC2531 skip bootloader payload", NS);
178
- this.unpiWriter.writeBuffer(Buffer.from([0xef]));
179
- await wait(1000);
180
- await this.request(Subsystem.SYS, "ping", {capabilities: 1}, undefined, 250 /* v8 ignore next */);
181
- } catch {
182
- // Skip bootloader on some CC2652 devices (e.g. zzh-p)
183
- logger.info("Skip bootloader for CC2652/CC1352", NS);
184
- if (this.serialPort) {
185
- await this.serialPort.asyncSet({dtr: false, rts: false});
186
- await wait(150);
187
- await this.serialPort.asyncSet({dtr: false, rts: true});
188
- await wait(150);
189
- await this.serialPort.asyncSet({dtr: false, rts: false});
190
- await wait(150);
191
- }
192
- }
193
- }
194
- }
195
-
196
- public async close(): Promise<void> {
197
- logger.info("closing", NS);
198
- this.queue.clear();
199
-
200
- if (this.initialized) {
201
- this.initialized = false;
202
-
203
- if (this.serialPort) {
204
- try {
205
- await this.serialPort.asyncFlushAndClose();
206
- } catch (error) {
207
- this.emit("close");
208
-
209
- throw error;
210
- }
211
- } else {
212
- // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress`
213
- this.socketPort!.destroy();
214
- }
215
- }
216
-
217
- this.emit("close");
218
- }
219
-
220
- public async requestWithReply(
221
- subsystem: Subsystem,
222
- command: string,
223
- payload: ZpiObjectPayload,
224
- waiterID?: number,
225
- timeout?: number,
226
- expectedStatuses: Constants.COMMON.ZnpCommandStatus[] = [ZnpCommandStatus.SUCCESS],
227
- ): Promise<ZpiObject> {
228
- const reply = await this.request(subsystem, command, payload, waiterID, timeout, expectedStatuses);
229
- if (reply === undefined) {
230
- throw new Error(`Command ${command} has no reply`);
231
- }
232
- return reply;
233
- }
234
-
235
- public request(
236
- subsystem: Subsystem,
237
- command: string,
238
- payload: ZpiObjectPayload,
239
- waiterID?: number,
240
- timeout?: number,
241
- expectedStatuses: Constants.COMMON.ZnpCommandStatus[] = [ZnpCommandStatus.SUCCESS],
242
- ): Promise<ZpiObject | undefined> {
243
- if (!this.initialized) {
244
- throw new Error("Cannot request when znp has not been initialized yet");
245
- }
246
-
247
- const object = ZpiObject.createRequest(subsystem, command, payload);
248
-
249
- return this.queue.execute<ZpiObject | undefined>(async () => {
250
- logger.debug(() => `--> ${object}`, NS);
251
-
252
- if (object.type === Type.SREQ) {
253
- const t = object.command.name === "bdbStartCommissioning" || object.command.name === "startupFromApp" ? 40000 : timeouts.SREQ;
254
- const waiter = this.waitress.waitFor({type: Type.SRSP, subsystem: object.subsystem, command: object.command.name}, timeout || t);
255
- this.unpiWriter.writeFrame(object.unpiFrame);
256
- const result = await waiter.start().promise;
257
- if (result?.payload.status !== undefined && !expectedStatuses.includes(result.payload.status)) {
258
- if (typeof waiterID === "number") {
259
- this.waitress.remove(waiterID);
260
- }
261
-
262
- throw new Error(
263
- `--> '${object}' failed with status '${statusDescription(
264
- result.payload.status,
265
- )}' (expected '${expectedStatuses.map(statusDescription)}')`,
266
- );
267
- }
268
-
269
- return result;
270
- }
271
-
272
- if (object.type === Type.AREQ && object.isResetCommand()) {
273
- const waiter = this.waitress.waitFor({type: Type.AREQ, subsystem: Subsystem.SYS, command: "resetInd"}, timeout || timeouts.reset);
274
- this.queue.clear();
275
- this.unpiWriter.writeFrame(object.unpiFrame);
276
- return await waiter.start().promise;
277
- }
278
-
279
- if (object.type === Type.AREQ) {
280
- this.unpiWriter.writeFrame(object.unpiFrame);
281
- /* v8 ignore start */
282
- } else {
283
- throw new Error(`Unknown type '${object.type}'`);
284
- }
285
- /* v8 ignore stop */
286
- });
287
- }
288
-
289
- public requestZdo(clusterId: ZdoClusterId, payload: Buffer, waiterID?: number): Promise<void> {
290
- return this.queue.execute(async () => {
291
- const cmd = Definition[Subsystem.ZDO].find((c) => isMtCmdSreqZdo(c) && c.zdoClusterId === clusterId);
292
- assert(cmd, `Command for ZDO cluster ID '${clusterId}' not supported.`);
293
-
294
- const unpiFrame = new UnpiFrame(Type.SREQ, Subsystem.ZDO, cmd.ID, payload);
295
- const waiter = this.waitress.waitFor({type: Type.SRSP, subsystem: Subsystem.ZDO, command: cmd.name}, timeouts.SREQ);
296
-
297
- this.unpiWriter.writeFrame(unpiFrame);
298
-
299
- const result = await waiter.start().promise;
300
-
301
- if (result?.payload.status !== undefined && result.payload.status !== ZnpCommandStatus.SUCCESS) {
302
- if (waiterID !== undefined) {
303
- this.waitress.remove(waiterID);
304
- }
305
-
306
- throw new Error(
307
- `--> 'SREQ: ZDO - ${ZdoClusterId[clusterId]} - ${payload.toString("hex")}' failed with status '${statusDescription(result.payload.status)}'`,
308
- );
309
- }
310
- });
311
- }
312
-
313
- private waitressTimeoutFormatter(matcher: WaitressMatcher, timeout: number): string {
314
- return `${Type[matcher.type]} - ${Subsystem[matcher.subsystem]} - ${matcher.command} after ${timeout}ms`;
315
- }
316
-
317
- public waitFor(
318
- type: Type,
319
- subsystem: Subsystem,
320
- command: string,
321
- target: number | string | undefined,
322
- transid: number | undefined,
323
- state: number | undefined,
324
- timeout: number = timeouts.default,
325
- ): {start: () => {promise: Promise<ZpiObject>; ID: number}; ID: number} {
326
- return this.waitress.waitFor({type, subsystem, command, target, transid, state}, timeout);
327
- }
328
-
329
- private waitressValidator(zpiObject: ZpiObject, matcher: WaitressMatcher): boolean {
330
- return (
331
- matcher.type === zpiObject.type &&
332
- matcher.subsystem === zpiObject.subsystem &&
333
- matcher.command === zpiObject.command.name &&
334
- (matcher.target === undefined ||
335
- (typeof matcher.target === "number"
336
- ? matcher.target === zpiObject.payload.srcaddr
337
- : matcher.target === zpiObject.payload.zdo?.[1]?.eui64)) &&
338
- (matcher.transid === undefined || matcher.transid === zpiObject.payload.transid) &&
339
- (matcher.state === undefined || matcher.state === zpiObject.payload.state)
340
- );
341
- }
342
- }
@@ -1,148 +0,0 @@
1
- import assert from "node:assert";
2
-
3
- import {ClusterId as ZdoClusterId} from "../../../zspec/zdo";
4
- import {BuffaloZdo} from "../../../zspec/zdo/buffaloZdo";
5
- import {Frame as UnpiFrame} from "../unpi";
6
- import {MaxDataSize, Subsystem, Type} from "../unpi/constants";
7
- import BuffaloZnp from "./buffaloZnp";
8
- import Definition from "./definition";
9
- import ParameterType from "./parameterType";
10
- import type {BuffaloZnpOptions, MtCmd, MtParameter, MtType, ZpiObjectPayload} from "./tstype";
11
- import {isMtCmdAreqZdo, isMtCmdSreqZdo} from "./utils";
12
-
13
- const BufferAndListTypes = [
14
- ParameterType.BUFFER,
15
- ParameterType.BUFFER8,
16
- ParameterType.BUFFER16,
17
- ParameterType.BUFFER18,
18
- ParameterType.BUFFER32,
19
- ParameterType.BUFFER42,
20
- ParameterType.BUFFER100,
21
- ParameterType.LIST_UINT16,
22
- ParameterType.LIST_NETWORK,
23
- ParameterType.LIST_UINT8,
24
- ];
25
-
26
- type ZpiObjectType = "Request" | "Response";
27
-
28
- export class ZpiObject<T extends ZpiObjectType = "Response"> {
29
- public readonly type: Type;
30
- public readonly subsystem: Subsystem;
31
- public readonly command: MtCmd;
32
- public readonly payload: ZpiObjectPayload;
33
- public readonly unpiFrame: T extends "Request" ? UnpiFrame : undefined;
34
-
35
- private constructor(
36
- type: Type,
37
- subsystem: Subsystem,
38
- command: MtCmd,
39
- payload: ZpiObjectPayload,
40
- unpiFrame: T extends "Request" ? UnpiFrame : undefined,
41
- ) {
42
- this.type = type;
43
- this.subsystem = subsystem;
44
- this.command = command;
45
- this.payload = payload;
46
- this.unpiFrame = unpiFrame;
47
- }
48
-
49
- public static createRequest(subsystem: Subsystem, command: string, payload: ZpiObjectPayload): ZpiObject<"Request"> {
50
- if (!Definition[subsystem]) {
51
- throw new Error(`Subsystem '${subsystem}' does not exist`);
52
- }
53
-
54
- const cmd = Definition[subsystem].find((c) => c.name === command);
55
-
56
- if (cmd === undefined || isMtCmdAreqZdo(cmd) || isMtCmdSreqZdo(cmd) || cmd.request === undefined) {
57
- throw new Error(`Command request '${command}' from subsystem '${subsystem}' not found`);
58
- }
59
-
60
- // Create the UnpiFrame
61
- const buffalo = new BuffaloZnp(Buffer.alloc(MaxDataSize));
62
-
63
- for (const parameter of cmd.request) {
64
- const value = payload[parameter.name];
65
- buffalo.write(parameter.parameterType, value, {});
66
- }
67
-
68
- const buffer = buffalo.getWritten();
69
- const unpiFrame = new UnpiFrame(cmd.type, subsystem, cmd.ID, buffer);
70
-
71
- return new ZpiObject(cmd.type, subsystem, cmd, payload, unpiFrame);
72
- }
73
-
74
- public static fromUnpiFrame(frame: UnpiFrame): ZpiObject<"Response"> {
75
- const cmd = Definition[frame.subsystem].find((c) => c.ID === frame.commandID);
76
-
77
- if (!cmd) {
78
- throw new Error(`CommandID '${frame.commandID}' from subsystem '${frame.subsystem}' not found`);
79
- }
80
-
81
- let payload: ZpiObjectPayload = {};
82
-
83
- // hotpath AREQ & SREQ ZDO since payload is identical (no need to instantiate BuffaloZnp or parse generically)
84
- if (isMtCmdAreqZdo(cmd)) {
85
- if (cmd.zdoClusterId === ZdoClusterId.NETWORK_ADDRESS_RESPONSE || cmd.zdoClusterId === ZdoClusterId.IEEE_ADDRESS_RESPONSE) {
86
- // ZStack swaps the `startindex` and `numassocdev` compared to ZDO swap them back before handing to ZDO
87
- const startIndex = frame.data[11];
88
- const assocDevCount = frame.data[12];
89
- frame.data[11] = assocDevCount;
90
- frame.data[12] = startIndex;
91
- payload.zdo = BuffaloZdo.readResponse(false, cmd.zdoClusterId, frame.data);
92
- } else {
93
- payload.srcaddr = frame.data.readUInt16LE(0);
94
- payload.zdo = BuffaloZdo.readResponse(false, cmd.zdoClusterId, frame.data.subarray(2));
95
- }
96
- } else if (isMtCmdSreqZdo(cmd)) {
97
- payload.status = frame.data.readUInt8(0);
98
- } else {
99
- const parameters = frame.type === Type.SRSP && cmd.type !== Type.AREQ ? cmd.response : cmd.request;
100
- assert(
101
- parameters,
102
- `CommandID '${frame.commandID}' from subsystem '${frame.subsystem}' cannot be a ` +
103
- `${frame.type === Type.SRSP ? "response" : "request"}`,
104
- );
105
- payload = ZpiObject.readParameters(frame.data, parameters);
106
- }
107
-
108
- // GC UnpiFrame as early as possible, no longer needed
109
- return new ZpiObject(frame.type, frame.subsystem, cmd, payload, undefined);
110
- }
111
-
112
- private static readParameters(buffer: Buffer, parameters: MtParameter[]): ZpiObjectPayload {
113
- const buffalo = new BuffaloZnp(buffer);
114
- const result: ZpiObjectPayload = {};
115
-
116
- for (const parameter of parameters) {
117
- const options: BuffaloZnpOptions = {};
118
-
119
- if (BufferAndListTypes.includes(parameter.parameterType)) {
120
- // When reading a buffer, assume that the previous parsed parameter contains
121
- // the length of the buffer
122
- const lengthParameter = parameters[parameters.indexOf(parameter) - 1];
123
- const length: MtType = result[lengthParameter.name];
124
-
125
- if (typeof length === "number") {
126
- options.length = length;
127
- }
128
- }
129
-
130
- result[parameter.name] = buffalo.read(parameter.parameterType, options);
131
- }
132
-
133
- return result;
134
- }
135
-
136
- public isResetCommand(): boolean {
137
- return (
138
- (this.command.name === "resetReq" && this.subsystem === Subsystem.SYS) ||
139
- /* v8 ignore next */
140
- (this.command.name === "systemReset" && this.subsystem === Subsystem.SAPI)
141
- );
142
- }
143
-
144
- public toString(includePayload = true): string {
145
- const baseStr = `${Type[this.type]}: ${Subsystem[this.subsystem]} - ${this.command.name}`;
146
- return includePayload ? `${baseStr} - ${JSON.stringify(this.payload)}` : baseStr;
147
- }
148
- }