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