@willieee802/zigbee-herdsman 0.18.2 → 0.19.2
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/.release-please-manifest.json +1 -1
- package/CHANGELOG.md +67 -0
- package/dist/adapter/adapter.d.ts +61 -61
- package/dist/adapter/adapter.d.ts.map +1 -1
- package/dist/adapter/adapter.js +146 -146
- package/dist/adapter/deconz/adapter/deconzAdapter.d.ts +68 -68
- package/dist/adapter/deconz/adapter/deconzAdapter.js +1060 -1060
- package/dist/adapter/deconz/adapter/index.d.ts +2 -2
- package/dist/adapter/deconz/adapter/index.js +10 -10
- package/dist/adapter/deconz/driver/constants.d.ts +104 -104
- package/dist/adapter/deconz/driver/constants.js +55 -55
- package/dist/adapter/deconz/driver/driver.d.ts +81 -81
- package/dist/adapter/deconz/driver/driver.js +732 -732
- package/dist/adapter/deconz/driver/frame.d.ts +6 -6
- package/dist/adapter/deconz/driver/frame.js +13 -13
- package/dist/adapter/deconz/driver/frameParser.d.ts +2 -2
- package/dist/adapter/deconz/driver/frameParser.js +443 -443
- package/dist/adapter/deconz/driver/parser.d.ts +12 -12
- package/dist/adapter/deconz/driver/parser.js +61 -61
- package/dist/adapter/deconz/driver/writer.d.ts +8 -8
- package/dist/adapter/deconz/driver/writer.js +44 -44
- package/dist/adapter/events.d.ts +47 -47
- package/dist/adapter/events.js +14 -14
- package/dist/adapter/ezsp/adapter/backup.d.ts +9 -9
- package/dist/adapter/ezsp/adapter/backup.js +53 -53
- package/dist/adapter/ezsp/adapter/ezspAdapter.d.ts +59 -59
- package/dist/adapter/ezsp/adapter/ezspAdapter.d.ts.map +1 -1
- package/dist/adapter/ezsp/adapter/ezspAdapter.js +603 -585
- package/dist/adapter/ezsp/adapter/ezspAdapter.js.map +1 -1
- package/dist/adapter/ezsp/adapter/index.d.ts +2 -2
- package/dist/adapter/ezsp/adapter/index.js +10 -10
- package/dist/adapter/ezsp/driver/commands.d.ts +36 -36
- package/dist/adapter/ezsp/driver/commands.js +2359 -2359
- package/dist/adapter/ezsp/driver/consts.d.ts +10 -10
- package/dist/adapter/ezsp/driver/consts.js +13 -13
- package/dist/adapter/ezsp/driver/driver.d.ts +103 -103
- package/dist/adapter/ezsp/driver/driver.js +639 -639
- package/dist/adapter/ezsp/driver/ezsp.d.ts +96 -96
- package/dist/adapter/ezsp/driver/ezsp.js +586 -586
- package/dist/adapter/ezsp/driver/index.d.ts +3 -3
- package/dist/adapter/ezsp/driver/index.js +8 -8
- package/dist/adapter/ezsp/driver/multicast.d.ts +12 -12
- package/dist/adapter/ezsp/driver/multicast.js +74 -74
- package/dist/adapter/ezsp/driver/parser.d.ts +12 -12
- package/dist/adapter/ezsp/driver/parser.js +111 -111
- package/dist/adapter/ezsp/driver/types/basic.d.ts +62 -62
- package/dist/adapter/ezsp/driver/types/basic.js +208 -208
- package/dist/adapter/ezsp/driver/types/basic.js.map +1 -1
- package/dist/adapter/ezsp/driver/types/index.d.ts +9 -9
- package/dist/adapter/ezsp/driver/types/index.js +133 -133
- package/dist/adapter/ezsp/driver/types/named.d.ts +697 -697
- package/dist/adapter/ezsp/driver/types/named.js +1726 -1726
- package/dist/adapter/ezsp/driver/types/named.js.map +1 -1
- package/dist/adapter/ezsp/driver/types/struct.d.ts +251 -251
- package/dist/adapter/ezsp/driver/types/struct.js +708 -708
- package/dist/adapter/ezsp/driver/types/struct.js.map +1 -1
- package/dist/adapter/ezsp/driver/uart.d.ts +44 -44
- package/dist/adapter/ezsp/driver/uart.js +368 -368
- package/dist/adapter/ezsp/driver/utils/crc16ccitt.d.ts +2 -2
- package/dist/adapter/ezsp/driver/utils/crc16ccitt.js +55 -55
- package/dist/adapter/ezsp/driver/utils/index.d.ts +18 -18
- package/dist/adapter/ezsp/driver/utils/index.js +67 -67
- package/dist/adapter/ezsp/driver/writer.d.ts +13 -13
- package/dist/adapter/ezsp/driver/writer.js +88 -88
- package/dist/adapter/index.d.ts +4 -4
- package/dist/adapter/index.js +35 -35
- package/dist/adapter/serialPort.d.ts +8 -8
- package/dist/adapter/serialPort.js +22 -22
- package/dist/adapter/serialPort.js.map +1 -1
- package/dist/adapter/serialPortUtils.d.ts +12 -12
- package/dist/adapter/serialPortUtils.js +18 -18
- package/dist/adapter/socketPortUtils.d.ts +10 -10
- package/dist/adapter/socketPortUtils.js +16 -16
- package/dist/adapter/tstype.d.ts +85 -85
- package/dist/adapter/tstype.js +2 -2
- package/dist/adapter/z-stack/adapter/adapter-backup.d.ts +62 -62
- package/dist/adapter/z-stack/adapter/adapter-backup.d.ts.map +1 -1
- package/dist/adapter/z-stack/adapter/adapter-backup.js +460 -441
- package/dist/adapter/z-stack/adapter/adapter-backup.js.map +1 -1
- package/dist/adapter/z-stack/adapter/adapter-nv-memory.d.ts +150 -150
- package/dist/adapter/z-stack/adapter/adapter-nv-memory.js +258 -258
- package/dist/adapter/z-stack/adapter/endpoints.d.ts +11 -11
- package/dist/adapter/z-stack/adapter/endpoints.js +73 -73
- package/dist/adapter/z-stack/adapter/index.d.ts +2 -2
- package/dist/adapter/z-stack/adapter/index.js +8 -8
- package/dist/adapter/z-stack/adapter/manager.d.ts +86 -86
- package/dist/adapter/z-stack/adapter/manager.js +476 -476
- package/dist/adapter/z-stack/adapter/tstype.d.ts +6 -6
- package/dist/adapter/z-stack/adapter/tstype.js +10 -10
- package/dist/adapter/z-stack/adapter/zStackAdapter.d.ts +81 -81
- package/dist/adapter/z-stack/adapter/zStackAdapter.d.ts.map +1 -1
- package/dist/adapter/z-stack/adapter/zStackAdapter.js +868 -868
- package/dist/adapter/z-stack/adapter/zStackAdapter.js.map +1 -1
- package/dist/adapter/z-stack/constants/af.d.ts +23 -23
- package/dist/adapter/z-stack/constants/af.js +27 -27
- package/dist/adapter/z-stack/constants/common.d.ts +278 -278
- package/dist/adapter/z-stack/constants/common.js +289 -289
- package/dist/adapter/z-stack/constants/dbg.d.ts +22 -22
- package/dist/adapter/z-stack/constants/dbg.js +24 -24
- package/dist/adapter/z-stack/constants/index.d.ts +10 -10
- package/dist/adapter/z-stack/constants/index.js +47 -47
- package/dist/adapter/z-stack/constants/mac.d.ts +127 -127
- package/dist/adapter/z-stack/constants/mac.js +129 -129
- package/dist/adapter/z-stack/constants/sapi.d.ts +24 -24
- package/dist/adapter/z-stack/constants/sapi.js +26 -26
- package/dist/adapter/z-stack/constants/sys.d.ts +71 -71
- package/dist/adapter/z-stack/constants/sys.js +73 -73
- package/dist/adapter/z-stack/constants/util.d.ts +81 -81
- package/dist/adapter/z-stack/constants/util.js +83 -83
- package/dist/adapter/z-stack/constants/utils.d.ts +4 -4
- package/dist/adapter/z-stack/constants/utils.js +14 -14
- package/dist/adapter/z-stack/constants/zdo.d.ts +102 -102
- package/dist/adapter/z-stack/constants/zdo.js +104 -104
- package/dist/adapter/z-stack/models/index.d.ts +1 -1
- package/dist/adapter/z-stack/models/index.js +17 -17
- package/dist/adapter/z-stack/models/startup-options.d.ts +12 -12
- package/dist/adapter/z-stack/models/startup-options.js +2 -2
- package/dist/adapter/z-stack/structs/entries/address-manager-entry.d.ts +23 -23
- package/dist/adapter/z-stack/structs/entries/address-manager-entry.js +45 -45
- package/dist/adapter/z-stack/structs/entries/address-manager-table.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/address-manager-table.js +22 -22
- package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.js +21 -21
- package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.js +23 -23
- package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.js +24 -24
- package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.js +23 -23
- package/dist/adapter/z-stack/structs/entries/channel-list.d.ts +8 -8
- package/dist/adapter/z-stack/structs/entries/channel-list.js +15 -15
- package/dist/adapter/z-stack/structs/entries/has-configured.d.ts +8 -8
- package/dist/adapter/z-stack/structs/entries/has-configured.js +16 -16
- package/dist/adapter/z-stack/structs/entries/index.d.ts +16 -16
- package/dist/adapter/z-stack/structs/entries/index.js +32 -32
- package/dist/adapter/z-stack/structs/entries/nib.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/nib.js +68 -68
- package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.js +18 -18
- package/dist/adapter/z-stack/structs/entries/nwk-key.d.ts +8 -8
- package/dist/adapter/z-stack/structs/entries/nwk-key.js +15 -15
- package/dist/adapter/z-stack/structs/entries/nwk-pan-id.d.ts +8 -8
- package/dist/adapter/z-stack/structs/entries/nwk-pan-id.js +15 -15
- package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.d.ts +13 -13
- package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.js +23 -23
- package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.js +22 -22
- package/dist/adapter/z-stack/structs/entries/security-manager-entry.d.ts +20 -20
- package/dist/adapter/z-stack/structs/entries/security-manager-entry.js +36 -36
- package/dist/adapter/z-stack/structs/entries/security-manager-table.d.ts +10 -10
- package/dist/adapter/z-stack/structs/entries/security-manager-table.js +24 -24
- package/dist/adapter/z-stack/structs/index.d.ts +4 -4
- package/dist/adapter/z-stack/structs/index.js +20 -20
- package/dist/adapter/z-stack/structs/serializable-memory-object.d.ts +13 -13
- package/dist/adapter/z-stack/structs/serializable-memory-object.js +2 -2
- package/dist/adapter/z-stack/structs/struct.d.ts +99 -99
- package/dist/adapter/z-stack/structs/struct.js +295 -295
- package/dist/adapter/z-stack/structs/table.d.ts +94 -94
- package/dist/adapter/z-stack/structs/table.js +161 -161
- package/dist/adapter/z-stack/unpi/constants.d.ts +28 -28
- package/dist/adapter/z-stack/unpi/constants.js +41 -41
- package/dist/adapter/z-stack/unpi/frame.d.ts +16 -16
- package/dist/adapter/z-stack/unpi/frame.js +48 -48
- package/dist/adapter/z-stack/unpi/index.d.ts +5 -5
- package/dist/adapter/z-stack/unpi/index.js +37 -37
- package/dist/adapter/z-stack/unpi/parser.d.ts +10 -10
- package/dist/adapter/z-stack/unpi/parser.js +74 -74
- package/dist/adapter/z-stack/unpi/writer.d.ts +10 -10
- package/dist/adapter/z-stack/unpi/writer.js +44 -44
- package/dist/adapter/z-stack/utils/channel-list.d.ts +20 -20
- package/dist/adapter/z-stack/utils/channel-list.js +40 -40
- package/dist/adapter/z-stack/utils/index.d.ts +2 -2
- package/dist/adapter/z-stack/utils/index.js +18 -18
- package/dist/adapter/z-stack/utils/network-options.d.ts +8 -8
- package/dist/adapter/z-stack/utils/network-options.js +22 -22
- package/dist/adapter/z-stack/znp/buffaloZnp.d.ts +11 -11
- package/dist/adapter/z-stack/znp/buffaloZnp.js +113 -113
- package/dist/adapter/z-stack/znp/definition.d.ts +5 -5
- package/dist/adapter/z-stack/znp/definition.js +3050 -3050
- package/dist/adapter/z-stack/znp/index.d.ts +3 -3
- package/dist/adapter/z-stack/znp/index.js +10 -10
- package/dist/adapter/z-stack/znp/parameterType.d.ts +22 -22
- package/dist/adapter/z-stack/znp/parameterType.js +25 -25
- package/dist/adapter/z-stack/znp/tstype.d.ts +21 -21
- package/dist/adapter/z-stack/znp/tstype.js +2 -2
- package/dist/adapter/z-stack/znp/znp.d.ts +43 -43
- package/dist/adapter/z-stack/znp/znp.js +325 -325
- package/dist/adapter/z-stack/znp/zpiObject.d.ts +19 -19
- package/dist/adapter/z-stack/znp/zpiObject.js +96 -96
- package/dist/adapter/zigate/adapter/index.d.ts +2 -2
- package/dist/adapter/zigate/adapter/index.js +10 -10
- package/dist/adapter/zigate/adapter/zigateAdapter.d.ts +69 -69
- package/dist/adapter/zigate/adapter/zigateAdapter.js +678 -678
- package/dist/adapter/zigate/debug.d.ts +7 -7
- package/dist/adapter/zigate/debug.js +22 -22
- package/dist/adapter/zigate/driver/buffaloZiGate.d.ts +18 -18
- package/dist/adapter/zigate/driver/buffaloZiGate.js +139 -139
- package/dist/adapter/zigate/driver/commandType.d.ts +41 -41
- package/dist/adapter/zigate/driver/commandType.js +385 -385
- package/dist/adapter/zigate/driver/constants.d.ts +276 -276
- package/dist/adapter/zigate/driver/constants.js +371 -371
- package/dist/adapter/zigate/driver/frame.d.ts +26 -26
- package/dist/adapter/zigate/driver/frame.js +172 -172
- package/dist/adapter/zigate/driver/frame.js.map +1 -1
- package/dist/adapter/zigate/driver/messageType.d.ts +11 -11
- package/dist/adapter/zigate/driver/messageType.js +278 -278
- package/dist/adapter/zigate/driver/parameterType.d.ts +20 -20
- package/dist/adapter/zigate/driver/parameterType.js +23 -23
- package/dist/adapter/zigate/driver/ziGateObject.d.ts +23 -23
- package/dist/adapter/zigate/driver/ziGateObject.js +106 -106
- package/dist/adapter/zigate/driver/zigate.d.ts +49 -49
- package/dist/adapter/zigate/driver/zigate.js +303 -303
- package/dist/buffalo/buffalo.d.ts +50 -50
- package/dist/buffalo/buffalo.js +322 -322
- package/dist/buffalo/index.d.ts +3 -3
- package/dist/buffalo/index.js +33 -33
- package/dist/buffalo/tstype.d.ts +8 -8
- package/dist/buffalo/tstype.js +2 -2
- package/dist/controller/controller.d.ts +113 -110
- package/dist/controller/controller.d.ts.map +1 -1
- package/dist/controller/controller.js +619 -607
- package/dist/controller/controller.js.map +1 -1
- package/dist/controller/database.d.ts +18 -18
- package/dist/controller/database.js +93 -93
- package/dist/controller/events.d.ts +58 -55
- package/dist/controller/events.d.ts.map +1 -1
- package/dist/controller/events.js +101 -101
- package/dist/controller/events.js.map +1 -1
- package/dist/controller/greenPower.d.ts +12 -12
- package/dist/controller/greenPower.js +220 -220
- package/dist/controller/helpers/index.d.ts +2 -2
- package/dist/controller/helpers/index.js +28 -28
- package/dist/controller/helpers/request.d.ts +22 -22
- package/dist/controller/helpers/request.js +71 -71
- package/dist/controller/helpers/zclFrameConverter.d.ts +7 -7
- package/dist/controller/helpers/zclFrameConverter.js +31 -31
- package/dist/controller/helpers/zclTransactionSequenceNumber.d.ts +5 -5
- package/dist/controller/helpers/zclTransactionSequenceNumber.js +13 -13
- package/dist/controller/index.d.ts +5 -5
- package/dist/controller/index.js +8 -8
- package/dist/controller/logger-stub.d.ts +6 -6
- package/dist/controller/logger-stub.js +2 -2
- package/dist/controller/model/device.d.ts +133 -132
- package/dist/controller/model/device.d.ts.map +1 -1
- package/dist/controller/model/device.js +717 -708
- package/dist/controller/model/device.js.map +1 -1
- package/dist/controller/model/endpoint.d.ts +131 -131
- package/dist/controller/model/endpoint.d.ts.map +1 -1
- package/dist/controller/model/endpoint.js +816 -817
- package/dist/controller/model/endpoint.js.map +1 -1
- package/dist/controller/model/entity.d.ts +14 -14
- package/dist/controller/model/entity.js +26 -26
- package/dist/controller/model/group.d.ts +38 -38
- package/dist/controller/model/group.js +221 -221
- package/dist/controller/model/index.d.ts +5 -5
- package/dist/controller/model/index.js +14 -14
- package/dist/controller/touchlink.d.ts +19 -19
- package/dist/controller/touchlink.js +157 -157
- package/dist/controller/tstype.d.ts +21 -21
- package/dist/controller/tstype.js +9 -9
- package/dist/index.d.ts +3 -3
- package/dist/index.js +33 -33
- package/dist/models/backup-storage-legacy.d.ts +26 -26
- package/dist/models/backup-storage-legacy.js +2 -2
- package/dist/models/backup-storage-unified.d.ts +49 -49
- package/dist/models/backup-storage-unified.js +2 -2
- package/dist/models/backup.d.ts +37 -37
- package/dist/models/backup.js +2 -2
- package/dist/models/index.d.ts +4 -4
- package/dist/models/index.js +20 -20
- package/dist/models/network-options.d.ts +12 -12
- package/dist/models/network-options.js +2 -2
- package/dist/utils/assertString.d.ts +2 -2
- package/dist/utils/assertString.js +8 -8
- package/dist/utils/backup.d.ts +20 -20
- package/dist/utils/backup.js +187 -187
- package/dist/utils/equalsPartial.d.ts +2 -2
- package/dist/utils/equalsPartial.js +11 -11
- package/dist/utils/index.d.ts +9 -9
- package/dist/utils/index.js +45 -45
- package/dist/utils/isNumberArray.d.ts +2 -2
- package/dist/utils/isNumberArray.js +6 -6
- package/dist/utils/queue.d.ts +11 -11
- package/dist/utils/queue.js +50 -50
- package/dist/utils/realpathSync.d.ts +2 -2
- package/dist/utils/realpathSync.js +12 -12
- package/dist/utils/wait.d.ts +2 -2
- package/dist/utils/wait.js +8 -8
- package/dist/utils/waitress.d.ts +21 -21
- package/dist/utils/waitress.js +61 -61
- package/dist/zcl/buffaloZcl.d.ts +41 -41
- package/dist/zcl/buffaloZcl.js +591 -591
- package/dist/zcl/definition/buffaloZclDataType.d.ts +17 -17
- package/dist/zcl/definition/buffaloZclDataType.js +20 -20
- package/dist/zcl/definition/cluster.d.ts +29 -29
- package/dist/zcl/definition/cluster.js +5335 -5335
- package/dist/zcl/definition/dataType.d.ts +59 -59
- package/dist/zcl/definition/dataType.js +64 -64
- package/dist/zcl/definition/direction.d.ts +5 -5
- package/dist/zcl/definition/direction.js +8 -8
- package/dist/zcl/definition/endpointDeviceType.d.ts +4 -4
- package/dist/zcl/definition/endpointDeviceType.js +15 -15
- package/dist/zcl/definition/foundation.d.ts +11 -11
- package/dist/zcl/definition/foundation.js +167 -167
- package/dist/zcl/definition/frameControl.d.ts +10 -10
- package/dist/zcl/definition/frameControl.js +2 -2
- package/dist/zcl/definition/frameType.d.ts +5 -5
- package/dist/zcl/definition/frameType.js +8 -8
- package/dist/zcl/definition/index.d.ts +13 -13
- package/dist/zcl/definition/index.js +51 -51
- package/dist/zcl/definition/manufacturerCode.d.ts +1074 -1074
- package/dist/zcl/definition/manufacturerCode.js +1079 -1079
- package/dist/zcl/definition/powerSource.d.ts +4 -4
- package/dist/zcl/definition/powerSource.js +12 -12
- package/dist/zcl/definition/status.d.ts +38 -38
- package/dist/zcl/definition/status.js +41 -41
- package/dist/zcl/definition/tstype.d.ts +16 -16
- package/dist/zcl/definition/tstype.js +2 -2
- package/dist/zcl/index.d.ts +15 -15
- package/dist/zcl/index.js +55 -55
- package/dist/zcl/tstype.d.ts +56 -56
- package/dist/zcl/tstype.js +10 -10
- package/dist/zcl/utils.d.ts +6 -6
- package/dist/zcl/utils.js +165 -165
- package/dist/zcl/zclFrame.d.ts +45 -45
- package/dist/zcl/zclFrame.js +347 -347
- package/dist/zcl/zclStatusError.d.ts +5 -5
- package/dist/zcl/zclStatusError.js +13 -13
- package/package.json +6 -6
|
@@ -1,640 +1,640 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
-
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
-
};
|
|
5
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
-
exports.Driver = void 0;
|
|
7
|
-
const ezsp_1 = require("./ezsp");
|
|
8
|
-
const types_1 = require("./types");
|
|
9
|
-
const events_1 = require("events");
|
|
10
|
-
const struct_1 = require("./types/struct");
|
|
11
|
-
const utils_1 = require("./utils");
|
|
12
|
-
const named_1 = require("./types/named");
|
|
13
|
-
const multicast_1 = require("./multicast");
|
|
14
|
-
const utils_2 = require("../../../utils");
|
|
15
|
-
const debug_1 = __importDefault(require("debug"));
|
|
16
|
-
const es6_1 = __importDefault(require("fast-deep-equal/es6"));
|
|
17
|
-
const debug = {
|
|
18
|
-
error: (0, debug_1.default)('zigbee-herdsman:adapter:ezsp:erro'),
|
|
19
|
-
log: (0, debug_1.default)('zigbee-herdsman:adapter:ezsp:driv'),
|
|
20
|
-
};
|
|
21
|
-
const IEEE_PREFIX_MFG_ID = [
|
|
22
|
-
{ mfgId: 0x115F, prefix: [0x04, 0xcf, 0xfc] },
|
|
23
|
-
{ mfgId: 0x115F, prefix: [0x54, 0xef, 0x44] },
|
|
24
|
-
];
|
|
25
|
-
const DEFAULT_MFG_ID = 0x1049;
|
|
26
|
-
class Driver extends events_1.EventEmitter {
|
|
27
|
-
constructor() {
|
|
28
|
-
super();
|
|
29
|
-
this.direct = named_1.EmberOutgoingMessageType.OUTGOING_DIRECT;
|
|
30
|
-
this.eui64ToNodeId = new Map();
|
|
31
|
-
this.eui64ToRelays = new Map();
|
|
32
|
-
this.transactionID = 1;
|
|
33
|
-
this.queue = new utils_2.Queue(8);
|
|
34
|
-
this.waitress = new utils_2.Waitress(this.waitressValidator, this.waitressTimeoutFormatter);
|
|
35
|
-
}
|
|
36
|
-
async onReset() {
|
|
37
|
-
let attempts = 0;
|
|
38
|
-
const pauses = [10, 30, 60];
|
|
39
|
-
let pause = 0;
|
|
40
|
-
while (true) {
|
|
41
|
-
debug.log(`Reset connection. Try ${attempts}`);
|
|
42
|
-
try {
|
|
43
|
-
await this.stop();
|
|
44
|
-
await (0, utils_2.Wait)(1000);
|
|
45
|
-
await this.startup(this.port, this.serialOpt, this.nwkOpt, this.greenPowerGroup);
|
|
46
|
-
break;
|
|
47
|
-
}
|
|
48
|
-
catch (e) {
|
|
49
|
-
debug.error(`Reset error ${e.stack}`);
|
|
50
|
-
attempts += 1;
|
|
51
|
-
if (pauses.length) {
|
|
52
|
-
pause = pauses.shift();
|
|
53
|
-
}
|
|
54
|
-
debug.log(`Pause ${pause}sec before try ${attempts}`);
|
|
55
|
-
await (0, utils_2.Wait)(pause * 1000);
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
}
|
|
59
|
-
/* eslint-disable-next-line @typescript-eslint/no-explicit-any*/
|
|
60
|
-
async startup(port, serialOpt, nwkOpt, greenPowerGroup) {
|
|
61
|
-
let result = 'resumed';
|
|
62
|
-
this.nwkOpt = nwkOpt;
|
|
63
|
-
this.port = port;
|
|
64
|
-
this.serialOpt = serialOpt;
|
|
65
|
-
this.greenPowerGroup = greenPowerGroup;
|
|
66
|
-
this.transactionID = 1;
|
|
67
|
-
this.ezsp = undefined;
|
|
68
|
-
this.ezsp = new ezsp_1.Ezsp();
|
|
69
|
-
this.ezsp.on('reset', this.onReset.bind(this));
|
|
70
|
-
this.ezsp.on('close', this.onClose.bind(this));
|
|
71
|
-
await this.ezsp.connect(port, serialOpt);
|
|
72
|
-
await this.ezsp.version();
|
|
73
|
-
await this.ezsp.updateConfig();
|
|
74
|
-
await this.ezsp.updatePolicies();
|
|
75
|
-
//await this.ezsp.setValue(EzspValueId.VALUE_MAXIMUM_OUTGOING_TRANSFER_SIZE, 82);
|
|
76
|
-
//await this.ezsp.setValue(EzspValueId.VALUE_MAXIMUM_INCOMING_TRANSFER_SIZE, 82);
|
|
77
|
-
await this.ezsp.setValue(named_1.EzspValueId.VALUE_END_DEVICE_KEEP_ALIVE_SUPPORT_MODE, 3);
|
|
78
|
-
await this.ezsp.setValue(named_1.EzspValueId.VALUE_CCA_THRESHOLD, 0);
|
|
79
|
-
await this.ezsp.setSourceRouting();
|
|
80
|
-
//const count = await ezsp.getConfigurationValue(EzspConfigId.CONFIG_APS_UNICAST_MESSAGE_COUNT);
|
|
81
|
-
//debug.log("APS_UNICAST_MESSAGE_COUNT is set to %s", count);
|
|
82
|
-
await this.addEndpoint({
|
|
83
|
-
inputClusters: [0x0000, 0x0003, 0x0006, 0x000A, 0x0019, 0x001A, 0x0300],
|
|
84
|
-
outputClusters: [0x0000, 0x0003, 0x0004, 0x0005, 0x0006, 0x0008, 0x0020,
|
|
85
|
-
0x0300, 0x0400, 0x0402, 0x0405, 0x0406, 0x0500, 0x0B01, 0x0B03,
|
|
86
|
-
0x0B04, 0x0702, 0x1000, 0xFC01, 0xFC02]
|
|
87
|
-
});
|
|
88
|
-
await this.addEndpoint({
|
|
89
|
-
endpoint: 242, profileId: 0xA1E0, deviceId: 0x61,
|
|
90
|
-
outputClusters: [0x0021]
|
|
91
|
-
});
|
|
92
|
-
// getting MFG_STRING token
|
|
93
|
-
//const mfgName = await ezsp.execCommand('getMfgToken', EzspMfgTokenId.MFG_STRING);
|
|
94
|
-
// getting MFG_BOARD_NAME token
|
|
95
|
-
//const boardName = await ezsp.execCommand('getMfgToken', EzspMfgTokenId.MFG_BOARD_NAME);
|
|
96
|
-
/* eslint-disable prefer-const */
|
|
97
|
-
let verInfo = await this.ezsp.getValue(named_1.EzspValueId.VALUE_VERSION_INFO);
|
|
98
|
-
let build, major, minor, patch, special;
|
|
99
|
-
[build, verInfo] = types_1.uint16_t.deserialize(types_1.uint16_t, verInfo);
|
|
100
|
-
[major, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
101
|
-
[minor, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
102
|
-
[patch, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
103
|
-
[special, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
104
|
-
/* eslint-enable prefer-const */
|
|
105
|
-
const vers = `${major}.${minor}.${patch}.${special} build ${build}`;
|
|
106
|
-
debug.log(`EmberZNet version: ${vers}`);
|
|
107
|
-
this.version = {
|
|
108
|
-
product: this.ezsp.ezspV,
|
|
109
|
-
majorrel: `${major}`,
|
|
110
|
-
minorrel: `${minor}`,
|
|
111
|
-
maintrel: `${patch} `,
|
|
112
|
-
revision: vers
|
|
113
|
-
};
|
|
114
|
-
if (await this.needsToBeInitialised(nwkOpt)) {
|
|
115
|
-
const res = await this.ezsp.execCommand('networkState');
|
|
116
|
-
debug.log(`Network state ${res.status}`);
|
|
117
|
-
if (res.status == named_1.EmberNetworkStatus.JOINED_NETWORK) {
|
|
118
|
-
debug.log(`Leaving current network and forming new network`);
|
|
119
|
-
const st = await this.ezsp.leaveNetwork();
|
|
120
|
-
console.assert(st == types_1.EmberStatus.NETWORK_DOWN, `leaveNetwork returned unexpected status: ${st}`);
|
|
121
|
-
}
|
|
122
|
-
await this.form_network();
|
|
123
|
-
result = 'reset';
|
|
124
|
-
}
|
|
125
|
-
const state = (await this.ezsp.execCommand('networkState')).status;
|
|
126
|
-
debug.log(`Network state ${state}`);
|
|
127
|
-
const netParams = await this.ezsp.execCommand('getNetworkParameters');
|
|
128
|
-
console.assert(netParams.status == types_1.EmberStatus.SUCCESS, `Command (getNetworkParameters) returned unexpected state: ${netParams.status}`);
|
|
129
|
-
this.networkParams = netParams.parameters;
|
|
130
|
-
debug.log("Node type: %s, Network parameters: %s", netParams.nodeType, this.networkParams);
|
|
131
|
-
const nwk = (await this.ezsp.execCommand('getNodeId')).nodeId;
|
|
132
|
-
const ieee = (await this.ezsp.execCommand('getEui64')).eui64;
|
|
133
|
-
this.ieee = new named_1.EmberEUI64(ieee);
|
|
134
|
-
debug.log('Network ready');
|
|
135
|
-
this.ezsp.on('frame', this.handleFrame.bind(this));
|
|
136
|
-
this.handleNodeJoined(nwk, this.ieee);
|
|
137
|
-
debug.log(`EZSP nwk=${nwk}, IEEE=0x${this.ieee}`);
|
|
138
|
-
const linkResult = await this.ezsp.execCommand('getKey', { keyType: named_1.EmberKeyType.TRUST_CENTER_LINK_KEY });
|
|
139
|
-
debug.log(`TRUST_CENTER_LINK_KEY: ${JSON.stringify(linkResult)}`);
|
|
140
|
-
const netResult = await this.ezsp.execCommand('getKey', { keyType: named_1.EmberKeyType.CURRENT_NETWORK_KEY });
|
|
141
|
-
debug.log(`CURRENT_NETWORK_KEY: ${JSON.stringify(netResult)}`);
|
|
142
|
-
await (0, utils_2.Wait)(1000);
|
|
143
|
-
await this.ezsp.execCommand('setManufacturerCode', { code: DEFAULT_MFG_ID });
|
|
144
|
-
this.multicast = new multicast_1.Multicast(this);
|
|
145
|
-
await this.multicast.startup([]);
|
|
146
|
-
await this.multicast.subscribe(greenPowerGroup, 242);
|
|
147
|
-
// await this.multicast.subscribe(1, 901);
|
|
148
|
-
return result;
|
|
149
|
-
}
|
|
150
|
-
async needsToBeInitialised(options) {
|
|
151
|
-
let valid = true;
|
|
152
|
-
valid = valid && (await this.ezsp.networkInit());
|
|
153
|
-
const netParams = await this.ezsp.execCommand('getNetworkParameters');
|
|
154
|
-
const networkParams = netParams.parameters;
|
|
155
|
-
debug.log("Current Node type: %s, Network parameters: %s", netParams.nodeType, networkParams);
|
|
156
|
-
valid = valid && (netParams.status == types_1.EmberStatus.SUCCESS);
|
|
157
|
-
valid = valid && (netParams.nodeType == types_1.EmberNodeType.COORDINATOR);
|
|
158
|
-
valid = valid && (options.panID == networkParams.panId);
|
|
159
|
-
valid = valid && (options.channelList.includes(networkParams.radioChannel));
|
|
160
|
-
valid = valid && ((0, es6_1.default)(options.extendedPanID, networkParams.extendedPanId));
|
|
161
|
-
return !valid;
|
|
162
|
-
}
|
|
163
|
-
async form_network() {
|
|
164
|
-
let status;
|
|
165
|
-
status = (await this.ezsp.execCommand('clearKeyTable')).status;
|
|
166
|
-
console.assert(status == types_1.EmberStatus.SUCCESS, `Command clearKeyTable returned unexpected state: ${status}`);
|
|
167
|
-
await this.ezsp.execCommand('clearTransientLinkKeys');
|
|
168
|
-
const panID = this.nwkOpt.panID;
|
|
169
|
-
const extendedPanID = this.nwkOpt.extendedPanID;
|
|
170
|
-
const initial_security_state = (0, utils_1.ember_security)(this.nwkOpt);
|
|
171
|
-
status = await this.ezsp.setInitialSecurityState(initial_security_state);
|
|
172
|
-
const parameters = new struct_1.EmberNetworkParameters();
|
|
173
|
-
parameters.panId = panID;
|
|
174
|
-
parameters.extendedPanId = extendedPanID;
|
|
175
|
-
parameters.radioTxPower = 5;
|
|
176
|
-
parameters.radioChannel = this.nwkOpt.channelList[0];
|
|
177
|
-
parameters.joinMethod = named_1.EmberJoinMethod.USE_MAC_ASSOCIATION;
|
|
178
|
-
parameters.nwkManagerId = 0;
|
|
179
|
-
parameters.nwkUpdateId = 0;
|
|
180
|
-
parameters.channels = 0x07FFF800; // all channels
|
|
181
|
-
await this.ezsp.formNetwork(parameters);
|
|
182
|
-
await this.ezsp.setValue(named_1.EzspValueId.VALUE_STACK_TOKEN_WRITING, 1);
|
|
183
|
-
}
|
|
184
|
-
handleFrame(frameName, frame) {
|
|
185
|
-
switch (true) {
|
|
186
|
-
case (frameName === 'incomingMessageHandler'): {
|
|
187
|
-
const eui64 = this.eui64ToNodeId.get(frame.sender);
|
|
188
|
-
const handled = this.waitress.resolve({ address: frame.sender, payload: frame.message,
|
|
189
|
-
frame: frame.apsFrame });
|
|
190
|
-
if (!handled) {
|
|
191
|
-
this.emit('incomingMessage', {
|
|
192
|
-
messageType: frame.type,
|
|
193
|
-
apsFrame: frame.apsFrame,
|
|
194
|
-
lqi: frame.lastHopLqi,
|
|
195
|
-
rssi: frame.lastHopRssi,
|
|
196
|
-
sender: frame.sender,
|
|
197
|
-
bindingIndex: frame.bindingIndex,
|
|
198
|
-
addressIndex: frame.addressIndex,
|
|
199
|
-
message: frame.message,
|
|
200
|
-
senderEui64: eui64
|
|
201
|
-
});
|
|
202
|
-
}
|
|
203
|
-
break;
|
|
204
|
-
}
|
|
205
|
-
case (frameName === 'trustCenterJoinHandler'): {
|
|
206
|
-
if (frame.status === named_1.EmberDeviceUpdate.DEVICE_LEFT) {
|
|
207
|
-
this.handleNodeLeft(frame.newNodeId, frame.newNodeEui64);
|
|
208
|
-
}
|
|
209
|
-
else {
|
|
210
|
-
if (frame.status === named_1.EmberDeviceUpdate.STANDARD_SECURITY_UNSECURED_JOIN) {
|
|
211
|
-
this.cleanupTClinkKey(frame.newNodeEui64);
|
|
212
|
-
}
|
|
213
|
-
if (frame.policyDecision !== types_1.EmberJoinDecision.DENY_JOIN) {
|
|
214
|
-
this.handleNodeJoined(frame.newNodeId, frame.newNodeEui64);
|
|
215
|
-
}
|
|
216
|
-
}
|
|
217
|
-
break;
|
|
218
|
-
}
|
|
219
|
-
case (frameName === 'incomingRouteRecordHandler'): {
|
|
220
|
-
this.handleRouteRecord(frame.source, frame.longId, frame.lastHopLqi, frame.lastHopRssi, frame.relay);
|
|
221
|
-
break;
|
|
222
|
-
}
|
|
223
|
-
case (frameName === 'incomingRouteErrorHandler'): {
|
|
224
|
-
this.handleRouteError(frame.status, frame.target);
|
|
225
|
-
break;
|
|
226
|
-
}
|
|
227
|
-
case (frameName === 'messageSentHandler'): {
|
|
228
|
-
// todo
|
|
229
|
-
const status = frame.status;
|
|
230
|
-
if (status != 0) {
|
|
231
|
-
// send failure
|
|
232
|
-
}
|
|
233
|
-
else {
|
|
234
|
-
// send success
|
|
235
|
-
// If there was a message to the group and this group is not known,
|
|
236
|
-
// then we will register the coordinator in this group
|
|
237
|
-
// Applicable for IKEA remotes
|
|
238
|
-
const msgType = frame.type;
|
|
239
|
-
if (msgType == named_1.EmberOutgoingMessageType.OUTGOING_MULTICAST) {
|
|
240
|
-
const apsFrame = frame.apsFrame;
|
|
241
|
-
if (apsFrame.destinationEndpoint == 255) {
|
|
242
|
-
this.multicast.subscribe(apsFrame.groupId, 1);
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
}
|
|
246
|
-
break;
|
|
247
|
-
}
|
|
248
|
-
case (frameName === 'macFilterMatchMessageHandler'): {
|
|
249
|
-
const [rawFrame, data] = struct_1.EmberIeeeRawFrame.deserialize(struct_1.EmberIeeeRawFrame, frame.message);
|
|
250
|
-
debug.log(`macFilterMatchMessageHandler frame message: ${rawFrame}`);
|
|
251
|
-
this.emit('incomingMessage', {
|
|
252
|
-
messageType: null,
|
|
253
|
-
apsFrame: rawFrame,
|
|
254
|
-
lqi: frame.lastHopLqi,
|
|
255
|
-
rssi: frame.lastHopRssi,
|
|
256
|
-
sender: null,
|
|
257
|
-
bindingIndex: null,
|
|
258
|
-
addressIndex: null,
|
|
259
|
-
message: data,
|
|
260
|
-
senderEui64: new named_1.EmberEUI64(rawFrame.sourceAddress)
|
|
261
|
-
});
|
|
262
|
-
break;
|
|
263
|
-
}
|
|
264
|
-
case (frameName === 'stackStatusHandler'): {
|
|
265
|
-
debug.log(`stackStatusHandler: ${types_1.EmberStatus.valueToName(types_1.EmberStatus, frame.status)}`);
|
|
266
|
-
break;
|
|
267
|
-
}
|
|
268
|
-
// case (frameName === 'childJoinHandler'): {
|
|
269
|
-
// if (!frame.joining) {
|
|
270
|
-
// this.handleNodeLeft(frame.childId, frame.childEui64);
|
|
271
|
-
// } else {
|
|
272
|
-
// this.handleNodeJoined(frame.childId, frame.childEui64);
|
|
273
|
-
// }
|
|
274
|
-
// break;
|
|
275
|
-
// }
|
|
276
|
-
case (frameName == 'gpepIncomingMessageHandler'): {
|
|
277
|
-
this.handleGPMessage(frame);
|
|
278
|
-
break;
|
|
279
|
-
}
|
|
280
|
-
default:
|
|
281
|
-
// <=== Application frame 35 (childJoinHandler) received: 00013e9c2ebd08feff9ffd9004 +1ms
|
|
282
|
-
// <=== Application frame 35 (childJoinHandler) parsed: 0,1,39998,144,253,159,255,254,8,189,46,4 +1ms
|
|
283
|
-
// Unhandled frame childJoinHandler +2s
|
|
284
|
-
// <=== Application frame 98 (incomingSenderEui64Handler) received: 2ebd08feff9ffd90 +2ms
|
|
285
|
-
// <=== Application frame 98 (incomingSenderEui64Handler) parsed: 144,253,159,255,254,8,189,46 +1ms
|
|
286
|
-
// Unhandled frame incomingSenderEui64Handler
|
|
287
|
-
// <=== Application frame 155 (zigbeeKeyEstablishmentHandler) received: 2ebd08feff9ffd9006 +2ms
|
|
288
|
-
// <=== Application frame 155 (zigbeeKeyEstablishmentHandler) parsed: 144,253,159,255,254,8,189,46,6 +2ms
|
|
289
|
-
// Unhandled frame zigbeeKeyEstablishmentHandler
|
|
290
|
-
debug.log(`Unhandled frame ${frameName}`);
|
|
291
|
-
}
|
|
292
|
-
}
|
|
293
|
-
async cleanupTClinkKey(ieee) {
|
|
294
|
-
// Remove tc link_key for the given device.
|
|
295
|
-
const index = (await this.ezsp.execCommand('findKeyTableEntry', { address: ieee, linkKey: true })).index;
|
|
296
|
-
if (index != 0xFF) {
|
|
297
|
-
await this.ezsp.execCommand('eraseKeyTableEntry', { index: index });
|
|
298
|
-
}
|
|
299
|
-
}
|
|
300
|
-
handleRouteRecord(nwk, ieee, lqi, rssi, relays) {
|
|
301
|
-
// todo
|
|
302
|
-
debug.log(`handleRouteRecord: nwk=${nwk}, ieee=${ieee}, lqi=${lqi}, rssi=${rssi}, relays=${relays}`);
|
|
303
|
-
this.setNode(nwk, ieee);
|
|
304
|
-
// if (ieee && !(ieee instanceof EmberEUI64)) {
|
|
305
|
-
// ieee = new EmberEUI64(ieee);
|
|
306
|
-
// }
|
|
307
|
-
// this.eui64ToRelays.set(ieee.toString(), relays);
|
|
308
|
-
}
|
|
309
|
-
async handleRouteError(status, nwk) {
|
|
310
|
-
// todo
|
|
311
|
-
debug.log(`handleRouteError: nwk=${nwk}, status=${status}`);
|
|
312
|
-
//this.waitress.reject({address: nwk, payload: null, frame: null}, 'Route error');
|
|
313
|
-
// const ieee = await this.networkIdToEUI64(nwk);
|
|
314
|
-
// this.eui64ToRelays.set(ieee.toString(), null);
|
|
315
|
-
}
|
|
316
|
-
handleNodeLeft(nwk, ieee) {
|
|
317
|
-
if (ieee && !(ieee instanceof named_1.EmberEUI64)) {
|
|
318
|
-
ieee = new named_1.EmberEUI64(ieee);
|
|
319
|
-
}
|
|
320
|
-
this.eui64ToNodeId.delete(ieee.toString());
|
|
321
|
-
this.emit('deviceLeft', [nwk, ieee]);
|
|
322
|
-
}
|
|
323
|
-
async resetMfgId(mfgId) {
|
|
324
|
-
await this.ezsp.execCommand('setManufacturerCode', { code: mfgId });
|
|
325
|
-
// 60 sec for waiting
|
|
326
|
-
await (0, utils_2.Wait)(60000);
|
|
327
|
-
await this.ezsp.execCommand('setManufacturerCode', { code: DEFAULT_MFG_ID });
|
|
328
|
-
}
|
|
329
|
-
handleNodeJoined(nwk, ieee) {
|
|
330
|
-
if (ieee && !(ieee instanceof named_1.EmberEUI64)) {
|
|
331
|
-
ieee = new named_1.EmberEUI64(ieee);
|
|
332
|
-
}
|
|
333
|
-
for (const rec of IEEE_PREFIX_MFG_ID) {
|
|
334
|
-
if ((Buffer.from(ieee.value)).indexOf(Buffer.from(rec.prefix)) == 0) {
|
|
335
|
-
// set ManufacturerCode
|
|
336
|
-
debug.log(`handleNodeJoined: change ManufacturerCode for ieee ${ieee} to ${rec.mfgId}`);
|
|
337
|
-
this.resetMfgId(rec.mfgId);
|
|
338
|
-
break;
|
|
339
|
-
}
|
|
340
|
-
}
|
|
341
|
-
this.eui64ToNodeId.set(ieee.toString(), nwk);
|
|
342
|
-
this.emit('deviceJoined', [nwk, ieee]);
|
|
343
|
-
}
|
|
344
|
-
setNode(nwk, ieee) {
|
|
345
|
-
if (ieee && !(ieee instanceof named_1.EmberEUI64)) {
|
|
346
|
-
ieee = new named_1.EmberEUI64(ieee);
|
|
347
|
-
}
|
|
348
|
-
this.eui64ToNodeId.set(ieee.toString(), nwk);
|
|
349
|
-
}
|
|
350
|
-
async request(nwk, apsFrame,
|
|
351
|
-
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
352
|
-
data, timeout = 30000) {
|
|
353
|
-
try {
|
|
354
|
-
const seq = (apsFrame.sequence + 1) & 0xFF;
|
|
355
|
-
let eui64;
|
|
356
|
-
if (typeof nwk !== 'number') {
|
|
357
|
-
eui64 = nwk;
|
|
358
|
-
const strEui64 = eui64.toString();
|
|
359
|
-
let nodeId = this.eui64ToNodeId.get(strEui64);
|
|
360
|
-
if (nodeId === undefined) {
|
|
361
|
-
nodeId = (await this.ezsp.execCommand('lookupNodeIdByEui64', { eui64: eui64 })).nodeId;
|
|
362
|
-
if (nodeId && nodeId !== 0xFFFF) {
|
|
363
|
-
this.eui64ToNodeId.set(strEui64, nodeId);
|
|
364
|
-
}
|
|
365
|
-
else {
|
|
366
|
-
throw new Error('Unknown EUI64:' + strEui64);
|
|
367
|
-
}
|
|
368
|
-
}
|
|
369
|
-
nwk = nodeId;
|
|
370
|
-
}
|
|
371
|
-
else {
|
|
372
|
-
eui64 = await this.networkIdToEUI64(nwk);
|
|
373
|
-
}
|
|
374
|
-
if (this.ezsp.ezspV < 8) {
|
|
375
|
-
// const route = this.eui64ToRelays.get(eui64.toString());
|
|
376
|
-
// if (route) {
|
|
377
|
-
// const = await this.ezsp.execCommand('setSourceRoute', {eui64});
|
|
378
|
-
// // }
|
|
379
|
-
}
|
|
380
|
-
await this.ezsp.execCommand('setExtendedTimeout', { remoteEui64: eui64, extendedTimeout: true });
|
|
381
|
-
const result = await this.ezsp.sendUnicast(this.direct, nwk, apsFrame, seq, data);
|
|
382
|
-
return result.status == types_1.EmberStatus.SUCCESS;
|
|
383
|
-
}
|
|
384
|
-
catch (e) {
|
|
385
|
-
debug.error(`Request error ${e}: ${e.stack}`);
|
|
386
|
-
return false;
|
|
387
|
-
}
|
|
388
|
-
}
|
|
389
|
-
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
390
|
-
async mrequest(apsFrame, data, timeout = 30000) {
|
|
391
|
-
try {
|
|
392
|
-
const seq = (apsFrame.sequence + 1) & 0xFF;
|
|
393
|
-
await this.ezsp.sendMulticast(apsFrame, seq, data);
|
|
394
|
-
return true;
|
|
395
|
-
}
|
|
396
|
-
catch (e) {
|
|
397
|
-
return false;
|
|
398
|
-
}
|
|
399
|
-
}
|
|
400
|
-
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
401
|
-
async rawrequest(rawFrame, data, timeout = 10000) {
|
|
402
|
-
try {
|
|
403
|
-
const msgData = Buffer.concat([struct_1.EmberRawFrame.serialize(struct_1.EmberRawFrame, rawFrame), data]);
|
|
404
|
-
await this.ezsp.execCommand('sendRawMessage', { message: msgData });
|
|
405
|
-
return true;
|
|
406
|
-
}
|
|
407
|
-
catch (e) {
|
|
408
|
-
debug.error(`Request error ${e}: ${e.stack}`);
|
|
409
|
-
return false;
|
|
410
|
-
}
|
|
411
|
-
}
|
|
412
|
-
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
413
|
-
async ieeerawrequest(rawFrame, data, timeout = 10000) {
|
|
414
|
-
try {
|
|
415
|
-
const msgData = Buffer.concat([struct_1.EmberIeeeRawFrame.serialize(struct_1.EmberIeeeRawFrame, rawFrame), data]);
|
|
416
|
-
await this.ezsp.execCommand('sendRawMessage', { message: msgData });
|
|
417
|
-
return true;
|
|
418
|
-
}
|
|
419
|
-
catch (e) {
|
|
420
|
-
debug.error(`Request error ${e}: ${e.stack}`);
|
|
421
|
-
return false;
|
|
422
|
-
}
|
|
423
|
-
}
|
|
424
|
-
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
425
|
-
async brequest(destination, apsFrame, data) {
|
|
426
|
-
try {
|
|
427
|
-
const seq = (apsFrame.sequence + 1) & 0xFF;
|
|
428
|
-
await this.ezsp.sendBroadcast(destination, apsFrame, seq, data);
|
|
429
|
-
return true;
|
|
430
|
-
}
|
|
431
|
-
catch (e) {
|
|
432
|
-
return false;
|
|
433
|
-
}
|
|
434
|
-
}
|
|
435
|
-
nextTransactionID() {
|
|
436
|
-
this.transactionID = (this.transactionID + 1) & 0xFF;
|
|
437
|
-
return this.transactionID;
|
|
438
|
-
}
|
|
439
|
-
makeApsFrame(clusterId, disableResponse) {
|
|
440
|
-
const frame = new struct_1.EmberApsFrame();
|
|
441
|
-
frame.clusterId = clusterId;
|
|
442
|
-
frame.profileId = 0;
|
|
443
|
-
frame.sequence = this.nextTransactionID();
|
|
444
|
-
frame.sourceEndpoint = 0;
|
|
445
|
-
frame.destinationEndpoint = 0;
|
|
446
|
-
frame.groupId = 0;
|
|
447
|
-
frame.options = (types_1.EmberApsOption.APS_OPTION_ENABLE_ROUTE_DISCOVERY ||
|
|
448
|
-
types_1.EmberApsOption.APS_OPTION_ENABLE_ADDRESS_DISCOVERY);
|
|
449
|
-
if (!disableResponse) {
|
|
450
|
-
frame.options || (frame.options = types_1.EmberApsOption.APS_OPTION_RETRY);
|
|
451
|
-
}
|
|
452
|
-
return frame;
|
|
453
|
-
}
|
|
454
|
-
makeEmberRawFrame() {
|
|
455
|
-
const frame = new struct_1.EmberRawFrame();
|
|
456
|
-
frame.sequence = this.nextTransactionID();
|
|
457
|
-
return frame;
|
|
458
|
-
}
|
|
459
|
-
makeEmberIeeeRawFrame() {
|
|
460
|
-
const frame = new struct_1.EmberIeeeRawFrame();
|
|
461
|
-
frame.sequence = this.nextTransactionID();
|
|
462
|
-
return frame;
|
|
463
|
-
}
|
|
464
|
-
async zdoRequest(networkAddress, requestCmd, responseCmd, params) {
|
|
465
|
-
const requestName = types_1.EmberZDOCmd.valueName(types_1.EmberZDOCmd, requestCmd);
|
|
466
|
-
const responseName = types_1.EmberZDOCmd.valueName(types_1.EmberZDOCmd, responseCmd);
|
|
467
|
-
debug.log(`ZDO ${requestName} params: ${JSON.stringify(params)}`);
|
|
468
|
-
const frame = this.makeApsFrame(requestCmd, false);
|
|
469
|
-
const payload = this.makeZDOframe(requestCmd, { transId: frame.sequence, ...params });
|
|
470
|
-
const waiter = this.waitFor(networkAddress, responseCmd, frame.sequence).start();
|
|
471
|
-
const res = await this.request(networkAddress, frame, payload);
|
|
472
|
-
if (!res) {
|
|
473
|
-
debug.error(`zdoRequest error`);
|
|
474
|
-
this.waitress.remove(waiter.ID);
|
|
475
|
-
throw Error('ZdoRequest error');
|
|
476
|
-
}
|
|
477
|
-
const message = await waiter.promise;
|
|
478
|
-
debug.log(`${responseName} frame: ${JSON.stringify(message.payload)}`);
|
|
479
|
-
const result = this.parse_frame_payload(responseCmd, message.payload);
|
|
480
|
-
debug.log(`${responseName} parsed: ${JSON.stringify(result)}`);
|
|
481
|
-
return result;
|
|
482
|
-
}
|
|
483
|
-
onClose() {
|
|
484
|
-
debug.log('Close driver');
|
|
485
|
-
}
|
|
486
|
-
async stop() {
|
|
487
|
-
if (this.ezsp) {
|
|
488
|
-
debug.log('Stop driver');
|
|
489
|
-
return this.ezsp.close(true);
|
|
490
|
-
}
|
|
491
|
-
}
|
|
492
|
-
async networkIdToEUI64(nwk) {
|
|
493
|
-
for (const [eUI64, value] of this.eui64ToNodeId) {
|
|
494
|
-
if (value === nwk)
|
|
495
|
-
return new named_1.EmberEUI64(eUI64);
|
|
496
|
-
}
|
|
497
|
-
const value = await this.ezsp.execCommand('lookupEui64ByNodeId', { nodeId: nwk });
|
|
498
|
-
if (value.status === types_1.EmberStatus.SUCCESS) {
|
|
499
|
-
const eUI64 = new named_1.EmberEUI64(value.eui64);
|
|
500
|
-
this.eui64ToNodeId.set(eUI64.toString(), nwk);
|
|
501
|
-
return eUI64;
|
|
502
|
-
}
|
|
503
|
-
else {
|
|
504
|
-
throw new Error('Unrecognized nodeId:' + nwk);
|
|
505
|
-
}
|
|
506
|
-
}
|
|
507
|
-
async preJoining() {
|
|
508
|
-
const ieee = new named_1.EmberEUI64('0xFFFFFFFFFFFFFFFF');
|
|
509
|
-
const linkKey = new types_1.EmberKeyData();
|
|
510
|
-
linkKey.contents = Buffer.from("ZigBeeAlliance09");
|
|
511
|
-
const result = await this.addTransientLinkKey(ieee, linkKey);
|
|
512
|
-
if (result.status !== types_1.EmberStatus.SUCCESS) {
|
|
513
|
-
throw new Error(`Add Transient Link Key for '${ieee}' failed`);
|
|
514
|
-
}
|
|
515
|
-
if (this.ezsp.ezspV >= 8) {
|
|
516
|
-
await this.ezsp.setPolicy(named_1.EzspPolicyId.TRUST_CENTER_POLICY, named_1.EzspDecisionBitmask.ALLOW_UNSECURED_REJOINS | named_1.EzspDecisionBitmask.ALLOW_JOINS);
|
|
517
|
-
//| EzspDecisionBitmask.JOINS_USE_INSTALL_CODE_KEY
|
|
518
|
-
}
|
|
519
|
-
}
|
|
520
|
-
async permitJoining(seconds) {
|
|
521
|
-
return this.ezsp.execCommand('permitJoining', { duration: seconds });
|
|
522
|
-
}
|
|
523
|
-
makeZDOframe(name, params) {
|
|
524
|
-
return this.ezsp.makeZDOframe(name, params);
|
|
525
|
-
}
|
|
526
|
-
parse_frame_payload(name, obj) {
|
|
527
|
-
return this.ezsp.parse_frame_payload(name, obj);
|
|
528
|
-
}
|
|
529
|
-
async addEndpoint({ endpoint = 1, profileId = 260, deviceId = 0xBEEF, appFlags = 0, inputClusters = [], outputClusters = [] }) {
|
|
530
|
-
const res = await this.ezsp.execCommand('addEndpoint', {
|
|
531
|
-
endpoint: endpoint,
|
|
532
|
-
profileId: profileId,
|
|
533
|
-
deviceId: deviceId,
|
|
534
|
-
appFlags: appFlags,
|
|
535
|
-
inputClusterCount: inputClusters.length,
|
|
536
|
-
outputClusterCount: outputClusters.length,
|
|
537
|
-
inputClusterList: inputClusters,
|
|
538
|
-
outputClusterList: outputClusters,
|
|
539
|
-
});
|
|
540
|
-
debug.log(`Ezsp adding endpoint: ${JSON.stringify(res)}`);
|
|
541
|
-
}
|
|
542
|
-
waitFor(address, clusterId, sequence, timeout = 10000) {
|
|
543
|
-
return this.waitress.waitFor({ address, clusterId, sequence }, timeout);
|
|
544
|
-
}
|
|
545
|
-
waitressTimeoutFormatter(matcher, timeout) {
|
|
546
|
-
return `${JSON.stringify(matcher)} after ${timeout}ms`;
|
|
547
|
-
}
|
|
548
|
-
waitressValidator(payload, matcher) {
|
|
549
|
-
return (!matcher.address || payload.address === matcher.address) &&
|
|
550
|
-
(!payload.frame || payload.frame.clusterId === matcher.clusterId) &&
|
|
551
|
-
(!payload.frame || payload.payload[0] === matcher.sequence);
|
|
552
|
-
}
|
|
553
|
-
setRadioPower(value) {
|
|
554
|
-
return this.ezsp.execCommand('setRadioPower', { power: value });
|
|
555
|
-
}
|
|
556
|
-
setChannel(channel) {
|
|
557
|
-
return this.ezsp.execCommand('setLogicalAndRadioChannel', { radioChannel: channel });
|
|
558
|
-
}
|
|
559
|
-
addTransientLinkKey(partner, transientKey) {
|
|
560
|
-
return this.ezsp.execCommand('addTransientLinkKey', { partner, transientKey });
|
|
561
|
-
}
|
|
562
|
-
async addInstallCode(ieeeAddress, key) {
|
|
563
|
-
// Key need to be converted to aes hash string
|
|
564
|
-
const hc = new struct_1.EmberAesMmoHashContext();
|
|
565
|
-
hc.result = Buffer.from([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]);
|
|
566
|
-
hc.length = 0;
|
|
567
|
-
const hash = await this.ezsp.execCommand('aesMmoHash', { context: hc, finalize: true, data: key });
|
|
568
|
-
if (hash.status == types_1.EmberStatus.SUCCESS) {
|
|
569
|
-
const ieee = new named_1.EmberEUI64(ieeeAddress);
|
|
570
|
-
const linkKey = new types_1.EmberKeyData();
|
|
571
|
-
linkKey.contents = hash.returnContext.result;
|
|
572
|
-
const result = await this.addTransientLinkKey(ieee, linkKey);
|
|
573
|
-
if (result.status !== types_1.EmberStatus.SUCCESS) {
|
|
574
|
-
throw new Error(`Add install code for '${ieeeAddress}' failed`);
|
|
575
|
-
}
|
|
576
|
-
}
|
|
577
|
-
else {
|
|
578
|
-
throw new Error(`Add install code for '${ieeeAddress}' failed`);
|
|
579
|
-
}
|
|
580
|
-
}
|
|
581
|
-
async handleGPMessage(frame) {
|
|
582
|
-
// Commissioning
|
|
583
|
-
if (frame.gpdCommandId == 0xE0) {
|
|
584
|
-
let data = frame.payload.subarray(5);
|
|
585
|
-
/* eslint-disable */
|
|
586
|
-
let st, deviceId, options, extOptions, key, mic, counter;
|
|
587
|
-
[st, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
588
|
-
[deviceId, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
589
|
-
[options, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
590
|
-
[extOptions, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
591
|
-
[key, data] = types_1.EmberKeyData.deserialize(types_1.EmberKeyData, data);
|
|
592
|
-
[mic, data] = types_1.uint32_t.deserialize(types_1.uint32_t, data);
|
|
593
|
-
[counter, data] = types_1.uint32_t.deserialize(types_1.uint32_t, data);
|
|
594
|
-
/* eslint-enable */
|
|
595
|
-
const gpdMessage = {
|
|
596
|
-
messageType: frame.gpdCommandId,
|
|
597
|
-
apsFrame: {
|
|
598
|
-
profileId: 0xA1E0,
|
|
599
|
-
sourceEndpoint: 242,
|
|
600
|
-
clusterId: 0x0021,
|
|
601
|
-
sequence: frame.sequenceNumber,
|
|
602
|
-
},
|
|
603
|
-
lqi: frame.gpdLink,
|
|
604
|
-
message: {
|
|
605
|
-
commandID: frame.gpdCommandId,
|
|
606
|
-
commandFrame: {
|
|
607
|
-
options: options,
|
|
608
|
-
securityKey: Buffer.from(key.contents),
|
|
609
|
-
deviceID: deviceId,
|
|
610
|
-
outgoingCounter: counter,
|
|
611
|
-
},
|
|
612
|
-
srcID: frame.srcId,
|
|
613
|
-
},
|
|
614
|
-
sender: frame.addr,
|
|
615
|
-
};
|
|
616
|
-
this.emit('incomingMessage', gpdMessage);
|
|
617
|
-
}
|
|
618
|
-
else {
|
|
619
|
-
const gpdMessage = {
|
|
620
|
-
messageType: frame.gpdCommandId,
|
|
621
|
-
apsFrame: {
|
|
622
|
-
profileId: 0xA1E0,
|
|
623
|
-
sourceEndpoint: 242,
|
|
624
|
-
clusterId: 0x0021,
|
|
625
|
-
sequence: frame.sequenceNumber,
|
|
626
|
-
},
|
|
627
|
-
lqi: frame.gpdLink,
|
|
628
|
-
message: {
|
|
629
|
-
commandID: frame.gpdCommandId,
|
|
630
|
-
frameCounter: frame.sequenceNumber,
|
|
631
|
-
srcID: frame.srcId,
|
|
632
|
-
},
|
|
633
|
-
sender: frame.addr,
|
|
634
|
-
};
|
|
635
|
-
this.emit('incomingMessage', gpdMessage);
|
|
636
|
-
}
|
|
637
|
-
}
|
|
638
|
-
}
|
|
639
|
-
exports.Driver = Driver;
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.Driver = void 0;
|
|
7
|
+
const ezsp_1 = require("./ezsp");
|
|
8
|
+
const types_1 = require("./types");
|
|
9
|
+
const events_1 = require("events");
|
|
10
|
+
const struct_1 = require("./types/struct");
|
|
11
|
+
const utils_1 = require("./utils");
|
|
12
|
+
const named_1 = require("./types/named");
|
|
13
|
+
const multicast_1 = require("./multicast");
|
|
14
|
+
const utils_2 = require("../../../utils");
|
|
15
|
+
const debug_1 = __importDefault(require("debug"));
|
|
16
|
+
const es6_1 = __importDefault(require("fast-deep-equal/es6"));
|
|
17
|
+
const debug = {
|
|
18
|
+
error: (0, debug_1.default)('zigbee-herdsman:adapter:ezsp:erro'),
|
|
19
|
+
log: (0, debug_1.default)('zigbee-herdsman:adapter:ezsp:driv'),
|
|
20
|
+
};
|
|
21
|
+
const IEEE_PREFIX_MFG_ID = [
|
|
22
|
+
{ mfgId: 0x115F, prefix: [0x04, 0xcf, 0xfc] },
|
|
23
|
+
{ mfgId: 0x115F, prefix: [0x54, 0xef, 0x44] },
|
|
24
|
+
];
|
|
25
|
+
const DEFAULT_MFG_ID = 0x1049;
|
|
26
|
+
class Driver extends events_1.EventEmitter {
|
|
27
|
+
constructor() {
|
|
28
|
+
super();
|
|
29
|
+
this.direct = named_1.EmberOutgoingMessageType.OUTGOING_DIRECT;
|
|
30
|
+
this.eui64ToNodeId = new Map();
|
|
31
|
+
this.eui64ToRelays = new Map();
|
|
32
|
+
this.transactionID = 1;
|
|
33
|
+
this.queue = new utils_2.Queue(8);
|
|
34
|
+
this.waitress = new utils_2.Waitress(this.waitressValidator, this.waitressTimeoutFormatter);
|
|
35
|
+
}
|
|
36
|
+
async onReset() {
|
|
37
|
+
let attempts = 0;
|
|
38
|
+
const pauses = [10, 30, 60];
|
|
39
|
+
let pause = 0;
|
|
40
|
+
while (true) {
|
|
41
|
+
debug.log(`Reset connection. Try ${attempts}`);
|
|
42
|
+
try {
|
|
43
|
+
await this.stop();
|
|
44
|
+
await (0, utils_2.Wait)(1000);
|
|
45
|
+
await this.startup(this.port, this.serialOpt, this.nwkOpt, this.greenPowerGroup);
|
|
46
|
+
break;
|
|
47
|
+
}
|
|
48
|
+
catch (e) {
|
|
49
|
+
debug.error(`Reset error ${e.stack}`);
|
|
50
|
+
attempts += 1;
|
|
51
|
+
if (pauses.length) {
|
|
52
|
+
pause = pauses.shift();
|
|
53
|
+
}
|
|
54
|
+
debug.log(`Pause ${pause}sec before try ${attempts}`);
|
|
55
|
+
await (0, utils_2.Wait)(pause * 1000);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
/* eslint-disable-next-line @typescript-eslint/no-explicit-any*/
|
|
60
|
+
async startup(port, serialOpt, nwkOpt, greenPowerGroup) {
|
|
61
|
+
let result = 'resumed';
|
|
62
|
+
this.nwkOpt = nwkOpt;
|
|
63
|
+
this.port = port;
|
|
64
|
+
this.serialOpt = serialOpt;
|
|
65
|
+
this.greenPowerGroup = greenPowerGroup;
|
|
66
|
+
this.transactionID = 1;
|
|
67
|
+
this.ezsp = undefined;
|
|
68
|
+
this.ezsp = new ezsp_1.Ezsp();
|
|
69
|
+
this.ezsp.on('reset', this.onReset.bind(this));
|
|
70
|
+
this.ezsp.on('close', this.onClose.bind(this));
|
|
71
|
+
await this.ezsp.connect(port, serialOpt);
|
|
72
|
+
await this.ezsp.version();
|
|
73
|
+
await this.ezsp.updateConfig();
|
|
74
|
+
await this.ezsp.updatePolicies();
|
|
75
|
+
//await this.ezsp.setValue(EzspValueId.VALUE_MAXIMUM_OUTGOING_TRANSFER_SIZE, 82);
|
|
76
|
+
//await this.ezsp.setValue(EzspValueId.VALUE_MAXIMUM_INCOMING_TRANSFER_SIZE, 82);
|
|
77
|
+
await this.ezsp.setValue(named_1.EzspValueId.VALUE_END_DEVICE_KEEP_ALIVE_SUPPORT_MODE, 3);
|
|
78
|
+
await this.ezsp.setValue(named_1.EzspValueId.VALUE_CCA_THRESHOLD, 0);
|
|
79
|
+
await this.ezsp.setSourceRouting();
|
|
80
|
+
//const count = await ezsp.getConfigurationValue(EzspConfigId.CONFIG_APS_UNICAST_MESSAGE_COUNT);
|
|
81
|
+
//debug.log("APS_UNICAST_MESSAGE_COUNT is set to %s", count);
|
|
82
|
+
await this.addEndpoint({
|
|
83
|
+
inputClusters: [0x0000, 0x0003, 0x0006, 0x000A, 0x0019, 0x001A, 0x0300],
|
|
84
|
+
outputClusters: [0x0000, 0x0003, 0x0004, 0x0005, 0x0006, 0x0008, 0x0020,
|
|
85
|
+
0x0300, 0x0400, 0x0402, 0x0405, 0x0406, 0x0500, 0x0B01, 0x0B03,
|
|
86
|
+
0x0B04, 0x0702, 0x1000, 0xFC01, 0xFC02]
|
|
87
|
+
});
|
|
88
|
+
await this.addEndpoint({
|
|
89
|
+
endpoint: 242, profileId: 0xA1E0, deviceId: 0x61,
|
|
90
|
+
outputClusters: [0x0021]
|
|
91
|
+
});
|
|
92
|
+
// getting MFG_STRING token
|
|
93
|
+
//const mfgName = await ezsp.execCommand('getMfgToken', EzspMfgTokenId.MFG_STRING);
|
|
94
|
+
// getting MFG_BOARD_NAME token
|
|
95
|
+
//const boardName = await ezsp.execCommand('getMfgToken', EzspMfgTokenId.MFG_BOARD_NAME);
|
|
96
|
+
/* eslint-disable prefer-const */
|
|
97
|
+
let verInfo = await this.ezsp.getValue(named_1.EzspValueId.VALUE_VERSION_INFO);
|
|
98
|
+
let build, major, minor, patch, special;
|
|
99
|
+
[build, verInfo] = types_1.uint16_t.deserialize(types_1.uint16_t, verInfo);
|
|
100
|
+
[major, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
101
|
+
[minor, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
102
|
+
[patch, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
103
|
+
[special, verInfo] = types_1.uint8_t.deserialize(types_1.uint8_t, verInfo);
|
|
104
|
+
/* eslint-enable prefer-const */
|
|
105
|
+
const vers = `${major}.${minor}.${patch}.${special} build ${build}`;
|
|
106
|
+
debug.log(`EmberZNet version: ${vers}`);
|
|
107
|
+
this.version = {
|
|
108
|
+
product: this.ezsp.ezspV,
|
|
109
|
+
majorrel: `${major}`,
|
|
110
|
+
minorrel: `${minor}`,
|
|
111
|
+
maintrel: `${patch} `,
|
|
112
|
+
revision: vers
|
|
113
|
+
};
|
|
114
|
+
if (await this.needsToBeInitialised(nwkOpt)) {
|
|
115
|
+
const res = await this.ezsp.execCommand('networkState');
|
|
116
|
+
debug.log(`Network state ${res.status}`);
|
|
117
|
+
if (res.status == named_1.EmberNetworkStatus.JOINED_NETWORK) {
|
|
118
|
+
debug.log(`Leaving current network and forming new network`);
|
|
119
|
+
const st = await this.ezsp.leaveNetwork();
|
|
120
|
+
console.assert(st == types_1.EmberStatus.NETWORK_DOWN, `leaveNetwork returned unexpected status: ${st}`);
|
|
121
|
+
}
|
|
122
|
+
await this.form_network();
|
|
123
|
+
result = 'reset';
|
|
124
|
+
}
|
|
125
|
+
const state = (await this.ezsp.execCommand('networkState')).status;
|
|
126
|
+
debug.log(`Network state ${state}`);
|
|
127
|
+
const netParams = await this.ezsp.execCommand('getNetworkParameters');
|
|
128
|
+
console.assert(netParams.status == types_1.EmberStatus.SUCCESS, `Command (getNetworkParameters) returned unexpected state: ${netParams.status}`);
|
|
129
|
+
this.networkParams = netParams.parameters;
|
|
130
|
+
debug.log("Node type: %s, Network parameters: %s", netParams.nodeType, this.networkParams);
|
|
131
|
+
const nwk = (await this.ezsp.execCommand('getNodeId')).nodeId;
|
|
132
|
+
const ieee = (await this.ezsp.execCommand('getEui64')).eui64;
|
|
133
|
+
this.ieee = new named_1.EmberEUI64(ieee);
|
|
134
|
+
debug.log('Network ready');
|
|
135
|
+
this.ezsp.on('frame', this.handleFrame.bind(this));
|
|
136
|
+
this.handleNodeJoined(nwk, this.ieee);
|
|
137
|
+
debug.log(`EZSP nwk=${nwk}, IEEE=0x${this.ieee}`);
|
|
138
|
+
const linkResult = await this.ezsp.execCommand('getKey', { keyType: named_1.EmberKeyType.TRUST_CENTER_LINK_KEY });
|
|
139
|
+
debug.log(`TRUST_CENTER_LINK_KEY: ${JSON.stringify(linkResult)}`);
|
|
140
|
+
const netResult = await this.ezsp.execCommand('getKey', { keyType: named_1.EmberKeyType.CURRENT_NETWORK_KEY });
|
|
141
|
+
debug.log(`CURRENT_NETWORK_KEY: ${JSON.stringify(netResult)}`);
|
|
142
|
+
await (0, utils_2.Wait)(1000);
|
|
143
|
+
await this.ezsp.execCommand('setManufacturerCode', { code: DEFAULT_MFG_ID });
|
|
144
|
+
this.multicast = new multicast_1.Multicast(this);
|
|
145
|
+
await this.multicast.startup([]);
|
|
146
|
+
await this.multicast.subscribe(greenPowerGroup, 242);
|
|
147
|
+
// await this.multicast.subscribe(1, 901);
|
|
148
|
+
return result;
|
|
149
|
+
}
|
|
150
|
+
async needsToBeInitialised(options) {
|
|
151
|
+
let valid = true;
|
|
152
|
+
valid = valid && (await this.ezsp.networkInit());
|
|
153
|
+
const netParams = await this.ezsp.execCommand('getNetworkParameters');
|
|
154
|
+
const networkParams = netParams.parameters;
|
|
155
|
+
debug.log("Current Node type: %s, Network parameters: %s", netParams.nodeType, networkParams);
|
|
156
|
+
valid = valid && (netParams.status == types_1.EmberStatus.SUCCESS);
|
|
157
|
+
valid = valid && (netParams.nodeType == types_1.EmberNodeType.COORDINATOR);
|
|
158
|
+
valid = valid && (options.panID == networkParams.panId);
|
|
159
|
+
valid = valid && (options.channelList.includes(networkParams.radioChannel));
|
|
160
|
+
valid = valid && ((0, es6_1.default)(options.extendedPanID, networkParams.extendedPanId));
|
|
161
|
+
return !valid;
|
|
162
|
+
}
|
|
163
|
+
async form_network() {
|
|
164
|
+
let status;
|
|
165
|
+
status = (await this.ezsp.execCommand('clearKeyTable')).status;
|
|
166
|
+
console.assert(status == types_1.EmberStatus.SUCCESS, `Command clearKeyTable returned unexpected state: ${status}`);
|
|
167
|
+
await this.ezsp.execCommand('clearTransientLinkKeys');
|
|
168
|
+
const panID = this.nwkOpt.panID;
|
|
169
|
+
const extendedPanID = this.nwkOpt.extendedPanID;
|
|
170
|
+
const initial_security_state = (0, utils_1.ember_security)(this.nwkOpt);
|
|
171
|
+
status = await this.ezsp.setInitialSecurityState(initial_security_state);
|
|
172
|
+
const parameters = new struct_1.EmberNetworkParameters();
|
|
173
|
+
parameters.panId = panID;
|
|
174
|
+
parameters.extendedPanId = extendedPanID;
|
|
175
|
+
parameters.radioTxPower = 5;
|
|
176
|
+
parameters.radioChannel = this.nwkOpt.channelList[0];
|
|
177
|
+
parameters.joinMethod = named_1.EmberJoinMethod.USE_MAC_ASSOCIATION;
|
|
178
|
+
parameters.nwkManagerId = 0;
|
|
179
|
+
parameters.nwkUpdateId = 0;
|
|
180
|
+
parameters.channels = 0x07FFF800; // all channels
|
|
181
|
+
await this.ezsp.formNetwork(parameters);
|
|
182
|
+
await this.ezsp.setValue(named_1.EzspValueId.VALUE_STACK_TOKEN_WRITING, 1);
|
|
183
|
+
}
|
|
184
|
+
handleFrame(frameName, frame) {
|
|
185
|
+
switch (true) {
|
|
186
|
+
case (frameName === 'incomingMessageHandler'): {
|
|
187
|
+
const eui64 = this.eui64ToNodeId.get(frame.sender);
|
|
188
|
+
const handled = this.waitress.resolve({ address: frame.sender, payload: frame.message,
|
|
189
|
+
frame: frame.apsFrame });
|
|
190
|
+
if (!handled) {
|
|
191
|
+
this.emit('incomingMessage', {
|
|
192
|
+
messageType: frame.type,
|
|
193
|
+
apsFrame: frame.apsFrame,
|
|
194
|
+
lqi: frame.lastHopLqi,
|
|
195
|
+
rssi: frame.lastHopRssi,
|
|
196
|
+
sender: frame.sender,
|
|
197
|
+
bindingIndex: frame.bindingIndex,
|
|
198
|
+
addressIndex: frame.addressIndex,
|
|
199
|
+
message: frame.message,
|
|
200
|
+
senderEui64: eui64
|
|
201
|
+
});
|
|
202
|
+
}
|
|
203
|
+
break;
|
|
204
|
+
}
|
|
205
|
+
case (frameName === 'trustCenterJoinHandler'): {
|
|
206
|
+
if (frame.status === named_1.EmberDeviceUpdate.DEVICE_LEFT) {
|
|
207
|
+
this.handleNodeLeft(frame.newNodeId, frame.newNodeEui64);
|
|
208
|
+
}
|
|
209
|
+
else {
|
|
210
|
+
if (frame.status === named_1.EmberDeviceUpdate.STANDARD_SECURITY_UNSECURED_JOIN) {
|
|
211
|
+
this.cleanupTClinkKey(frame.newNodeEui64);
|
|
212
|
+
}
|
|
213
|
+
if (frame.policyDecision !== types_1.EmberJoinDecision.DENY_JOIN) {
|
|
214
|
+
this.handleNodeJoined(frame.newNodeId, frame.newNodeEui64);
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
break;
|
|
218
|
+
}
|
|
219
|
+
case (frameName === 'incomingRouteRecordHandler'): {
|
|
220
|
+
this.handleRouteRecord(frame.source, frame.longId, frame.lastHopLqi, frame.lastHopRssi, frame.relay);
|
|
221
|
+
break;
|
|
222
|
+
}
|
|
223
|
+
case (frameName === 'incomingRouteErrorHandler'): {
|
|
224
|
+
this.handleRouteError(frame.status, frame.target);
|
|
225
|
+
break;
|
|
226
|
+
}
|
|
227
|
+
case (frameName === 'messageSentHandler'): {
|
|
228
|
+
// todo
|
|
229
|
+
const status = frame.status;
|
|
230
|
+
if (status != 0) {
|
|
231
|
+
// send failure
|
|
232
|
+
}
|
|
233
|
+
else {
|
|
234
|
+
// send success
|
|
235
|
+
// If there was a message to the group and this group is not known,
|
|
236
|
+
// then we will register the coordinator in this group
|
|
237
|
+
// Applicable for IKEA remotes
|
|
238
|
+
const msgType = frame.type;
|
|
239
|
+
if (msgType == named_1.EmberOutgoingMessageType.OUTGOING_MULTICAST) {
|
|
240
|
+
const apsFrame = frame.apsFrame;
|
|
241
|
+
if (apsFrame.destinationEndpoint == 255) {
|
|
242
|
+
this.multicast.subscribe(apsFrame.groupId, 1);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
break;
|
|
247
|
+
}
|
|
248
|
+
case (frameName === 'macFilterMatchMessageHandler'): {
|
|
249
|
+
const [rawFrame, data] = struct_1.EmberIeeeRawFrame.deserialize(struct_1.EmberIeeeRawFrame, frame.message);
|
|
250
|
+
debug.log(`macFilterMatchMessageHandler frame message: ${rawFrame}`);
|
|
251
|
+
this.emit('incomingMessage', {
|
|
252
|
+
messageType: null,
|
|
253
|
+
apsFrame: rawFrame,
|
|
254
|
+
lqi: frame.lastHopLqi,
|
|
255
|
+
rssi: frame.lastHopRssi,
|
|
256
|
+
sender: null,
|
|
257
|
+
bindingIndex: null,
|
|
258
|
+
addressIndex: null,
|
|
259
|
+
message: data,
|
|
260
|
+
senderEui64: new named_1.EmberEUI64(rawFrame.sourceAddress)
|
|
261
|
+
});
|
|
262
|
+
break;
|
|
263
|
+
}
|
|
264
|
+
case (frameName === 'stackStatusHandler'): {
|
|
265
|
+
debug.log(`stackStatusHandler: ${types_1.EmberStatus.valueToName(types_1.EmberStatus, frame.status)}`);
|
|
266
|
+
break;
|
|
267
|
+
}
|
|
268
|
+
// case (frameName === 'childJoinHandler'): {
|
|
269
|
+
// if (!frame.joining) {
|
|
270
|
+
// this.handleNodeLeft(frame.childId, frame.childEui64);
|
|
271
|
+
// } else {
|
|
272
|
+
// this.handleNodeJoined(frame.childId, frame.childEui64);
|
|
273
|
+
// }
|
|
274
|
+
// break;
|
|
275
|
+
// }
|
|
276
|
+
case (frameName == 'gpepIncomingMessageHandler'): {
|
|
277
|
+
this.handleGPMessage(frame);
|
|
278
|
+
break;
|
|
279
|
+
}
|
|
280
|
+
default:
|
|
281
|
+
// <=== Application frame 35 (childJoinHandler) received: 00013e9c2ebd08feff9ffd9004 +1ms
|
|
282
|
+
// <=== Application frame 35 (childJoinHandler) parsed: 0,1,39998,144,253,159,255,254,8,189,46,4 +1ms
|
|
283
|
+
// Unhandled frame childJoinHandler +2s
|
|
284
|
+
// <=== Application frame 98 (incomingSenderEui64Handler) received: 2ebd08feff9ffd90 +2ms
|
|
285
|
+
// <=== Application frame 98 (incomingSenderEui64Handler) parsed: 144,253,159,255,254,8,189,46 +1ms
|
|
286
|
+
// Unhandled frame incomingSenderEui64Handler
|
|
287
|
+
// <=== Application frame 155 (zigbeeKeyEstablishmentHandler) received: 2ebd08feff9ffd9006 +2ms
|
|
288
|
+
// <=== Application frame 155 (zigbeeKeyEstablishmentHandler) parsed: 144,253,159,255,254,8,189,46,6 +2ms
|
|
289
|
+
// Unhandled frame zigbeeKeyEstablishmentHandler
|
|
290
|
+
debug.log(`Unhandled frame ${frameName}`);
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
async cleanupTClinkKey(ieee) {
|
|
294
|
+
// Remove tc link_key for the given device.
|
|
295
|
+
const index = (await this.ezsp.execCommand('findKeyTableEntry', { address: ieee, linkKey: true })).index;
|
|
296
|
+
if (index != 0xFF) {
|
|
297
|
+
await this.ezsp.execCommand('eraseKeyTableEntry', { index: index });
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
handleRouteRecord(nwk, ieee, lqi, rssi, relays) {
|
|
301
|
+
// todo
|
|
302
|
+
debug.log(`handleRouteRecord: nwk=${nwk}, ieee=${ieee}, lqi=${lqi}, rssi=${rssi}, relays=${relays}`);
|
|
303
|
+
this.setNode(nwk, ieee);
|
|
304
|
+
// if (ieee && !(ieee instanceof EmberEUI64)) {
|
|
305
|
+
// ieee = new EmberEUI64(ieee);
|
|
306
|
+
// }
|
|
307
|
+
// this.eui64ToRelays.set(ieee.toString(), relays);
|
|
308
|
+
}
|
|
309
|
+
async handleRouteError(status, nwk) {
|
|
310
|
+
// todo
|
|
311
|
+
debug.log(`handleRouteError: nwk=${nwk}, status=${status}`);
|
|
312
|
+
//this.waitress.reject({address: nwk, payload: null, frame: null}, 'Route error');
|
|
313
|
+
// const ieee = await this.networkIdToEUI64(nwk);
|
|
314
|
+
// this.eui64ToRelays.set(ieee.toString(), null);
|
|
315
|
+
}
|
|
316
|
+
handleNodeLeft(nwk, ieee) {
|
|
317
|
+
if (ieee && !(ieee instanceof named_1.EmberEUI64)) {
|
|
318
|
+
ieee = new named_1.EmberEUI64(ieee);
|
|
319
|
+
}
|
|
320
|
+
this.eui64ToNodeId.delete(ieee.toString());
|
|
321
|
+
this.emit('deviceLeft', [nwk, ieee]);
|
|
322
|
+
}
|
|
323
|
+
async resetMfgId(mfgId) {
|
|
324
|
+
await this.ezsp.execCommand('setManufacturerCode', { code: mfgId });
|
|
325
|
+
// 60 sec for waiting
|
|
326
|
+
await (0, utils_2.Wait)(60000);
|
|
327
|
+
await this.ezsp.execCommand('setManufacturerCode', { code: DEFAULT_MFG_ID });
|
|
328
|
+
}
|
|
329
|
+
handleNodeJoined(nwk, ieee) {
|
|
330
|
+
if (ieee && !(ieee instanceof named_1.EmberEUI64)) {
|
|
331
|
+
ieee = new named_1.EmberEUI64(ieee);
|
|
332
|
+
}
|
|
333
|
+
for (const rec of IEEE_PREFIX_MFG_ID) {
|
|
334
|
+
if ((Buffer.from(ieee.value)).indexOf(Buffer.from(rec.prefix)) == 0) {
|
|
335
|
+
// set ManufacturerCode
|
|
336
|
+
debug.log(`handleNodeJoined: change ManufacturerCode for ieee ${ieee} to ${rec.mfgId}`);
|
|
337
|
+
this.resetMfgId(rec.mfgId);
|
|
338
|
+
break;
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
this.eui64ToNodeId.set(ieee.toString(), nwk);
|
|
342
|
+
this.emit('deviceJoined', [nwk, ieee]);
|
|
343
|
+
}
|
|
344
|
+
setNode(nwk, ieee) {
|
|
345
|
+
if (ieee && !(ieee instanceof named_1.EmberEUI64)) {
|
|
346
|
+
ieee = new named_1.EmberEUI64(ieee);
|
|
347
|
+
}
|
|
348
|
+
this.eui64ToNodeId.set(ieee.toString(), nwk);
|
|
349
|
+
}
|
|
350
|
+
async request(nwk, apsFrame,
|
|
351
|
+
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
352
|
+
data, timeout = 30000) {
|
|
353
|
+
try {
|
|
354
|
+
const seq = (apsFrame.sequence + 1) & 0xFF;
|
|
355
|
+
let eui64;
|
|
356
|
+
if (typeof nwk !== 'number') {
|
|
357
|
+
eui64 = nwk;
|
|
358
|
+
const strEui64 = eui64.toString();
|
|
359
|
+
let nodeId = this.eui64ToNodeId.get(strEui64);
|
|
360
|
+
if (nodeId === undefined) {
|
|
361
|
+
nodeId = (await this.ezsp.execCommand('lookupNodeIdByEui64', { eui64: eui64 })).nodeId;
|
|
362
|
+
if (nodeId && nodeId !== 0xFFFF) {
|
|
363
|
+
this.eui64ToNodeId.set(strEui64, nodeId);
|
|
364
|
+
}
|
|
365
|
+
else {
|
|
366
|
+
throw new Error('Unknown EUI64:' + strEui64);
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
nwk = nodeId;
|
|
370
|
+
}
|
|
371
|
+
else {
|
|
372
|
+
eui64 = await this.networkIdToEUI64(nwk);
|
|
373
|
+
}
|
|
374
|
+
if (this.ezsp.ezspV < 8) {
|
|
375
|
+
// const route = this.eui64ToRelays.get(eui64.toString());
|
|
376
|
+
// if (route) {
|
|
377
|
+
// const = await this.ezsp.execCommand('setSourceRoute', {eui64});
|
|
378
|
+
// // }
|
|
379
|
+
}
|
|
380
|
+
await this.ezsp.execCommand('setExtendedTimeout', { remoteEui64: eui64, extendedTimeout: true });
|
|
381
|
+
const result = await this.ezsp.sendUnicast(this.direct, nwk, apsFrame, seq, data);
|
|
382
|
+
return result.status == types_1.EmberStatus.SUCCESS;
|
|
383
|
+
}
|
|
384
|
+
catch (e) {
|
|
385
|
+
debug.error(`Request error ${e}: ${e.stack}`);
|
|
386
|
+
return false;
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
390
|
+
async mrequest(apsFrame, data, timeout = 30000) {
|
|
391
|
+
try {
|
|
392
|
+
const seq = (apsFrame.sequence + 1) & 0xFF;
|
|
393
|
+
await this.ezsp.sendMulticast(apsFrame, seq, data);
|
|
394
|
+
return true;
|
|
395
|
+
}
|
|
396
|
+
catch (e) {
|
|
397
|
+
return false;
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
401
|
+
async rawrequest(rawFrame, data, timeout = 10000) {
|
|
402
|
+
try {
|
|
403
|
+
const msgData = Buffer.concat([struct_1.EmberRawFrame.serialize(struct_1.EmberRawFrame, rawFrame), data]);
|
|
404
|
+
await this.ezsp.execCommand('sendRawMessage', { message: msgData });
|
|
405
|
+
return true;
|
|
406
|
+
}
|
|
407
|
+
catch (e) {
|
|
408
|
+
debug.error(`Request error ${e}: ${e.stack}`);
|
|
409
|
+
return false;
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
413
|
+
async ieeerawrequest(rawFrame, data, timeout = 10000) {
|
|
414
|
+
try {
|
|
415
|
+
const msgData = Buffer.concat([struct_1.EmberIeeeRawFrame.serialize(struct_1.EmberIeeeRawFrame, rawFrame), data]);
|
|
416
|
+
await this.ezsp.execCommand('sendRawMessage', { message: msgData });
|
|
417
|
+
return true;
|
|
418
|
+
}
|
|
419
|
+
catch (e) {
|
|
420
|
+
debug.error(`Request error ${e}: ${e.stack}`);
|
|
421
|
+
return false;
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
/* eslint-disable-next-line @typescript-eslint/no-unused-vars */
|
|
425
|
+
async brequest(destination, apsFrame, data) {
|
|
426
|
+
try {
|
|
427
|
+
const seq = (apsFrame.sequence + 1) & 0xFF;
|
|
428
|
+
await this.ezsp.sendBroadcast(destination, apsFrame, seq, data);
|
|
429
|
+
return true;
|
|
430
|
+
}
|
|
431
|
+
catch (e) {
|
|
432
|
+
return false;
|
|
433
|
+
}
|
|
434
|
+
}
|
|
435
|
+
nextTransactionID() {
|
|
436
|
+
this.transactionID = (this.transactionID + 1) & 0xFF;
|
|
437
|
+
return this.transactionID;
|
|
438
|
+
}
|
|
439
|
+
makeApsFrame(clusterId, disableResponse) {
|
|
440
|
+
const frame = new struct_1.EmberApsFrame();
|
|
441
|
+
frame.clusterId = clusterId;
|
|
442
|
+
frame.profileId = 0;
|
|
443
|
+
frame.sequence = this.nextTransactionID();
|
|
444
|
+
frame.sourceEndpoint = 0;
|
|
445
|
+
frame.destinationEndpoint = 0;
|
|
446
|
+
frame.groupId = 0;
|
|
447
|
+
frame.options = (types_1.EmberApsOption.APS_OPTION_ENABLE_ROUTE_DISCOVERY ||
|
|
448
|
+
types_1.EmberApsOption.APS_OPTION_ENABLE_ADDRESS_DISCOVERY);
|
|
449
|
+
if (!disableResponse) {
|
|
450
|
+
frame.options || (frame.options = types_1.EmberApsOption.APS_OPTION_RETRY);
|
|
451
|
+
}
|
|
452
|
+
return frame;
|
|
453
|
+
}
|
|
454
|
+
makeEmberRawFrame() {
|
|
455
|
+
const frame = new struct_1.EmberRawFrame();
|
|
456
|
+
frame.sequence = this.nextTransactionID();
|
|
457
|
+
return frame;
|
|
458
|
+
}
|
|
459
|
+
makeEmberIeeeRawFrame() {
|
|
460
|
+
const frame = new struct_1.EmberIeeeRawFrame();
|
|
461
|
+
frame.sequence = this.nextTransactionID();
|
|
462
|
+
return frame;
|
|
463
|
+
}
|
|
464
|
+
async zdoRequest(networkAddress, requestCmd, responseCmd, params) {
|
|
465
|
+
const requestName = types_1.EmberZDOCmd.valueName(types_1.EmberZDOCmd, requestCmd);
|
|
466
|
+
const responseName = types_1.EmberZDOCmd.valueName(types_1.EmberZDOCmd, responseCmd);
|
|
467
|
+
debug.log(`ZDO ${requestName} params: ${JSON.stringify(params)}`);
|
|
468
|
+
const frame = this.makeApsFrame(requestCmd, false);
|
|
469
|
+
const payload = this.makeZDOframe(requestCmd, { transId: frame.sequence, ...params });
|
|
470
|
+
const waiter = this.waitFor(networkAddress, responseCmd, frame.sequence).start();
|
|
471
|
+
const res = await this.request(networkAddress, frame, payload);
|
|
472
|
+
if (!res) {
|
|
473
|
+
debug.error(`zdoRequest error`);
|
|
474
|
+
this.waitress.remove(waiter.ID);
|
|
475
|
+
throw Error('ZdoRequest error');
|
|
476
|
+
}
|
|
477
|
+
const message = await waiter.promise;
|
|
478
|
+
debug.log(`${responseName} frame: ${JSON.stringify(message.payload)}`);
|
|
479
|
+
const result = this.parse_frame_payload(responseCmd, message.payload);
|
|
480
|
+
debug.log(`${responseName} parsed: ${JSON.stringify(result)}`);
|
|
481
|
+
return result;
|
|
482
|
+
}
|
|
483
|
+
onClose() {
|
|
484
|
+
debug.log('Close driver');
|
|
485
|
+
}
|
|
486
|
+
async stop() {
|
|
487
|
+
if (this.ezsp) {
|
|
488
|
+
debug.log('Stop driver');
|
|
489
|
+
return this.ezsp.close(true);
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
async networkIdToEUI64(nwk) {
|
|
493
|
+
for (const [eUI64, value] of this.eui64ToNodeId) {
|
|
494
|
+
if (value === nwk)
|
|
495
|
+
return new named_1.EmberEUI64(eUI64);
|
|
496
|
+
}
|
|
497
|
+
const value = await this.ezsp.execCommand('lookupEui64ByNodeId', { nodeId: nwk });
|
|
498
|
+
if (value.status === types_1.EmberStatus.SUCCESS) {
|
|
499
|
+
const eUI64 = new named_1.EmberEUI64(value.eui64);
|
|
500
|
+
this.eui64ToNodeId.set(eUI64.toString(), nwk);
|
|
501
|
+
return eUI64;
|
|
502
|
+
}
|
|
503
|
+
else {
|
|
504
|
+
throw new Error('Unrecognized nodeId:' + nwk);
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
async preJoining() {
|
|
508
|
+
const ieee = new named_1.EmberEUI64('0xFFFFFFFFFFFFFFFF');
|
|
509
|
+
const linkKey = new types_1.EmberKeyData();
|
|
510
|
+
linkKey.contents = Buffer.from("ZigBeeAlliance09");
|
|
511
|
+
const result = await this.addTransientLinkKey(ieee, linkKey);
|
|
512
|
+
if (result.status !== types_1.EmberStatus.SUCCESS) {
|
|
513
|
+
throw new Error(`Add Transient Link Key for '${ieee}' failed`);
|
|
514
|
+
}
|
|
515
|
+
if (this.ezsp.ezspV >= 8) {
|
|
516
|
+
await this.ezsp.setPolicy(named_1.EzspPolicyId.TRUST_CENTER_POLICY, named_1.EzspDecisionBitmask.ALLOW_UNSECURED_REJOINS | named_1.EzspDecisionBitmask.ALLOW_JOINS);
|
|
517
|
+
//| EzspDecisionBitmask.JOINS_USE_INSTALL_CODE_KEY
|
|
518
|
+
}
|
|
519
|
+
}
|
|
520
|
+
async permitJoining(seconds) {
|
|
521
|
+
return this.ezsp.execCommand('permitJoining', { duration: seconds });
|
|
522
|
+
}
|
|
523
|
+
makeZDOframe(name, params) {
|
|
524
|
+
return this.ezsp.makeZDOframe(name, params);
|
|
525
|
+
}
|
|
526
|
+
parse_frame_payload(name, obj) {
|
|
527
|
+
return this.ezsp.parse_frame_payload(name, obj);
|
|
528
|
+
}
|
|
529
|
+
async addEndpoint({ endpoint = 1, profileId = 260, deviceId = 0xBEEF, appFlags = 0, inputClusters = [], outputClusters = [] }) {
|
|
530
|
+
const res = await this.ezsp.execCommand('addEndpoint', {
|
|
531
|
+
endpoint: endpoint,
|
|
532
|
+
profileId: profileId,
|
|
533
|
+
deviceId: deviceId,
|
|
534
|
+
appFlags: appFlags,
|
|
535
|
+
inputClusterCount: inputClusters.length,
|
|
536
|
+
outputClusterCount: outputClusters.length,
|
|
537
|
+
inputClusterList: inputClusters,
|
|
538
|
+
outputClusterList: outputClusters,
|
|
539
|
+
});
|
|
540
|
+
debug.log(`Ezsp adding endpoint: ${JSON.stringify(res)}`);
|
|
541
|
+
}
|
|
542
|
+
waitFor(address, clusterId, sequence, timeout = 10000) {
|
|
543
|
+
return this.waitress.waitFor({ address, clusterId, sequence }, timeout);
|
|
544
|
+
}
|
|
545
|
+
waitressTimeoutFormatter(matcher, timeout) {
|
|
546
|
+
return `${JSON.stringify(matcher)} after ${timeout}ms`;
|
|
547
|
+
}
|
|
548
|
+
waitressValidator(payload, matcher) {
|
|
549
|
+
return (!matcher.address || payload.address === matcher.address) &&
|
|
550
|
+
(!payload.frame || payload.frame.clusterId === matcher.clusterId) &&
|
|
551
|
+
(!payload.frame || payload.payload[0] === matcher.sequence);
|
|
552
|
+
}
|
|
553
|
+
setRadioPower(value) {
|
|
554
|
+
return this.ezsp.execCommand('setRadioPower', { power: value });
|
|
555
|
+
}
|
|
556
|
+
setChannel(channel) {
|
|
557
|
+
return this.ezsp.execCommand('setLogicalAndRadioChannel', { radioChannel: channel });
|
|
558
|
+
}
|
|
559
|
+
addTransientLinkKey(partner, transientKey) {
|
|
560
|
+
return this.ezsp.execCommand('addTransientLinkKey', { partner, transientKey });
|
|
561
|
+
}
|
|
562
|
+
async addInstallCode(ieeeAddress, key) {
|
|
563
|
+
// Key need to be converted to aes hash string
|
|
564
|
+
const hc = new struct_1.EmberAesMmoHashContext();
|
|
565
|
+
hc.result = Buffer.from([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]);
|
|
566
|
+
hc.length = 0;
|
|
567
|
+
const hash = await this.ezsp.execCommand('aesMmoHash', { context: hc, finalize: true, data: key });
|
|
568
|
+
if (hash.status == types_1.EmberStatus.SUCCESS) {
|
|
569
|
+
const ieee = new named_1.EmberEUI64(ieeeAddress);
|
|
570
|
+
const linkKey = new types_1.EmberKeyData();
|
|
571
|
+
linkKey.contents = hash.returnContext.result;
|
|
572
|
+
const result = await this.addTransientLinkKey(ieee, linkKey);
|
|
573
|
+
if (result.status !== types_1.EmberStatus.SUCCESS) {
|
|
574
|
+
throw new Error(`Add install code for '${ieeeAddress}' failed`);
|
|
575
|
+
}
|
|
576
|
+
}
|
|
577
|
+
else {
|
|
578
|
+
throw new Error(`Add install code for '${ieeeAddress}' failed`);
|
|
579
|
+
}
|
|
580
|
+
}
|
|
581
|
+
async handleGPMessage(frame) {
|
|
582
|
+
// Commissioning
|
|
583
|
+
if (frame.gpdCommandId == 0xE0) {
|
|
584
|
+
let data = frame.payload.subarray(5);
|
|
585
|
+
/* eslint-disable */
|
|
586
|
+
let st, deviceId, options, extOptions, key, mic, counter;
|
|
587
|
+
[st, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
588
|
+
[deviceId, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
589
|
+
[options, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
590
|
+
[extOptions, data] = types_1.uint8_t.deserialize(types_1.uint8_t, data);
|
|
591
|
+
[key, data] = types_1.EmberKeyData.deserialize(types_1.EmberKeyData, data);
|
|
592
|
+
[mic, data] = types_1.uint32_t.deserialize(types_1.uint32_t, data);
|
|
593
|
+
[counter, data] = types_1.uint32_t.deserialize(types_1.uint32_t, data);
|
|
594
|
+
/* eslint-enable */
|
|
595
|
+
const gpdMessage = {
|
|
596
|
+
messageType: frame.gpdCommandId,
|
|
597
|
+
apsFrame: {
|
|
598
|
+
profileId: 0xA1E0,
|
|
599
|
+
sourceEndpoint: 242,
|
|
600
|
+
clusterId: 0x0021,
|
|
601
|
+
sequence: frame.sequenceNumber,
|
|
602
|
+
},
|
|
603
|
+
lqi: frame.gpdLink,
|
|
604
|
+
message: {
|
|
605
|
+
commandID: frame.gpdCommandId,
|
|
606
|
+
commandFrame: {
|
|
607
|
+
options: options,
|
|
608
|
+
securityKey: Buffer.from(key.contents),
|
|
609
|
+
deviceID: deviceId,
|
|
610
|
+
outgoingCounter: counter,
|
|
611
|
+
},
|
|
612
|
+
srcID: frame.srcId,
|
|
613
|
+
},
|
|
614
|
+
sender: frame.addr,
|
|
615
|
+
};
|
|
616
|
+
this.emit('incomingMessage', gpdMessage);
|
|
617
|
+
}
|
|
618
|
+
else {
|
|
619
|
+
const gpdMessage = {
|
|
620
|
+
messageType: frame.gpdCommandId,
|
|
621
|
+
apsFrame: {
|
|
622
|
+
profileId: 0xA1E0,
|
|
623
|
+
sourceEndpoint: 242,
|
|
624
|
+
clusterId: 0x0021,
|
|
625
|
+
sequence: frame.sequenceNumber,
|
|
626
|
+
},
|
|
627
|
+
lqi: frame.gpdLink,
|
|
628
|
+
message: {
|
|
629
|
+
commandID: frame.gpdCommandId,
|
|
630
|
+
frameCounter: frame.sequenceNumber,
|
|
631
|
+
srcID: frame.srcId,
|
|
632
|
+
},
|
|
633
|
+
sender: frame.addr,
|
|
634
|
+
};
|
|
635
|
+
this.emit('incomingMessage', gpdMessage);
|
|
636
|
+
}
|
|
637
|
+
}
|
|
638
|
+
}
|
|
639
|
+
exports.Driver = Driver;
|
|
640
640
|
//# sourceMappingURL=driver.js.map
|