UNPKG

zigbee-herdsman

Version:

An open source ZigBee gateway solution with node.js.

803 lines (659 loc) 28.8 kB
/* v8 ignore start */ import {EventEmitter} from "node:events"; import {Queue, Waitress, wait} from "../../../utils"; import {logger} from "../../../utils/logger"; import type {SerialPortOptions} from "../../tstype"; import { type EZSPFrameDesc, FRAMES, FRAME_NAMES_BY_ID, type ParamsDesc, ZDOREQUESTS, ZDOREQUEST_NAME_BY_ID, ZDORESPONSES, ZDORESPONSE_NAME_BY_ID, } from "./commands"; import * as t from "./types"; import { EmberConcentratorType, type EmberOutgoingMessageType, EmberStatus, EmberZdoConfigurationFlags, EzspConfigId, EzspDecisionBitmask, EzspDecisionId, EzspPolicyId, } from "./types/named"; import type {EmberApsFrame, EmberNetworkParameters} from "./types/struct"; import {SerialDriver} from "./uart"; const NS = "zh:ezsp:ezsp"; const MAX_SERIAL_CONNECT_ATTEMPTS = 4; /** In ms. This is multiplied by tries count (above), e.g. 4 tries = 5000, 10000, 15000 */ const SERIAL_CONNECT_NEW_ATTEMPT_MIN_DELAY = 5000; const MTOR_MIN_INTERVAL = 10; const MTOR_MAX_INTERVAL = 90; const MTOR_ROUTE_ERROR_THRESHOLD = 4; const MTOR_DELIVERY_FAIL_THRESHOLD = 3; const MAX_WATCHDOG_FAILURES = 4; //const RESET_ATTEMPT_BACKOFF_TIME = 5; const WATCHDOG_WAKE_PERIOD = 10; // in sec //const EZSP_COUNTER_CLEAR_INTERVAL = 180; // Clear counters every n * WATCHDOG_WAKE_PERIOD const EZSP_DEFAULT_RADIUS = 0; const EZSP_MULTICAST_NON_MEMBER_RADIUS = 3; const CONFIG_IDS_PRE_V9: number[][] = [ [EzspConfigId.CONFIG_TC_REJOINS_USING_WELL_KNOWN_KEY_TIMEOUT_S, 90], [EzspConfigId.CONFIG_TRUST_CENTER_ADDRESS_CACHE_SIZE, 2], //[EzspConfigId.CONFIG_SUPPORTED_NETWORKS, 1], [EzspConfigId.CONFIG_FRAGMENT_DELAY_MS, 50], [EzspConfigId.CONFIG_PAN_ID_CONFLICT_REPORT_THRESHOLD, 2], //[EzspConfigId.CONFIG_SOURCE_ROUTE_TABLE_SIZE, 16], //[EzspConfigId.CONFIG_ADDRESS_TABLE_SIZE, 16], [ EzspConfigId.CONFIG_APPLICATION_ZDO_FLAGS, EmberZdoConfigurationFlags.APP_HANDLES_UNSUPPORTED_ZDO_REQUESTS | EmberZdoConfigurationFlags.APP_RECEIVES_SUPPORTED_ZDO_REQUESTS, ], [EzspConfigId.CONFIG_INDIRECT_TRANSMISSION_TIMEOUT, 7680], [EzspConfigId.CONFIG_END_DEVICE_POLL_TIMEOUT, 14], [EzspConfigId.CONFIG_SECURITY_LEVEL, 5], [EzspConfigId.CONFIG_STACK_PROFILE, 2], //[EzspConfigId.CONFIG_TX_POWER_MODE, 3], [EzspConfigId.CONFIG_FRAGMENT_WINDOW_SIZE, 1], //[EzspConfigId.CONFIG_NEIGHBOR_TABLE_SIZE, 16], //[EzspConfigId.CONFIG_ROUTE_TABLE_SIZE, 16], //[EzspConfigId.CONFIG_BINDING_TABLE_SIZE, 32], //[EzspConfigId.CONFIG_KEY_TABLE_SIZE, 12], //[EzspConfigId.CONFIG_ZLL_GROUP_ADDRESSES, 0], //[EzspConfigId.CONFIG_ZLL_RSSI_THRESHOLD, 0], //[EzspConfigId.CONFIG_APS_UNICAST_MESSAGE_COUNT, 255], //[EzspConfigId.CONFIG_BROADCAST_TABLE_SIZE, 43], //[EzspConfigId.CONFIG_MAX_HOPS, 30], //[EzspConfigId.CONFIG_MAX_END_DEVICE_CHILDREN, 32], [EzspConfigId.CONFIG_PACKET_BUFFER_COUNT, 255], ]; /** * Can only decrease "NCP Memory Allocation" configs at runtime from V9 on. * @see https://www.silabs.com/documents/public/release-notes/emberznet-release-notes-7.0.1.0.pdf */ const CONFIG_IDS_CURRENT: number[][] = [ [EzspConfigId.CONFIG_TC_REJOINS_USING_WELL_KNOWN_KEY_TIMEOUT_S, 90], [EzspConfigId.CONFIG_TRUST_CENTER_ADDRESS_CACHE_SIZE, 2], [EzspConfigId.CONFIG_FRAGMENT_DELAY_MS, 50], [EzspConfigId.CONFIG_PAN_ID_CONFLICT_REPORT_THRESHOLD, 2], [ EzspConfigId.CONFIG_APPLICATION_ZDO_FLAGS, EmberZdoConfigurationFlags.APP_HANDLES_UNSUPPORTED_ZDO_REQUESTS | EmberZdoConfigurationFlags.APP_RECEIVES_SUPPORTED_ZDO_REQUESTS, ], [EzspConfigId.CONFIG_INDIRECT_TRANSMISSION_TIMEOUT, 7680], [EzspConfigId.CONFIG_END_DEVICE_POLL_TIMEOUT, 14], [EzspConfigId.CONFIG_SECURITY_LEVEL, 5], [EzspConfigId.CONFIG_STACK_PROFILE, 2], [EzspConfigId.CONFIG_FRAGMENT_WINDOW_SIZE, 1], ]; const POLICY_IDS_PRE_V8: number[][] = [ // [EzspPolicyId.BINDING_MODIFICATION_POLICY, // EzspDecisionId.DISALLOW_BINDING_MODIFICATION], // [EzspPolicyId.UNICAST_REPLIES_POLICY, EzspDecisionId.HOST_WILL_NOT_SUPPLY_REPLY], // [EzspPolicyId.POLL_HANDLER_POLICY, EzspDecisionId.POLL_HANDLER_IGNORE], // [EzspPolicyId.MESSAGE_CONTENTS_IN_CALLBACK_POLICY, // EzspDecisionId.MESSAGE_TAG_ONLY_IN_CALLBACK], // [EzspPolicyId.PACKET_VALIDATE_LIBRARY_POLICY, // EzspDecisionId.PACKET_VALIDATE_LIBRARY_CHECKS_DISABLED], // [EzspPolicyId.ZLL_POLICY, EzspDecisionId.ALLOW_JOINS], // [EzspPolicyId.TC_REJOINS_USING_WELL_KNOWN_KEY_POLICY, EzspDecisionId.ALLOW_JOINS], [EzspPolicyId.APP_KEY_REQUEST_POLICY, EzspDecisionId.DENY_APP_KEY_REQUESTS], [EzspPolicyId.TC_KEY_REQUEST_POLICY, EzspDecisionId.ALLOW_TC_KEY_REQUESTS], ]; const POLICY_IDS_CURRENT: number[][] = [ [EzspPolicyId.APP_KEY_REQUEST_POLICY, EzspDecisionId.DENY_APP_KEY_REQUESTS], [EzspPolicyId.TC_KEY_REQUEST_POLICY, EzspDecisionId.ALLOW_TC_KEY_REQUESTS], [EzspPolicyId.TRUST_CENTER_POLICY, EzspDecisionBitmask.ALLOW_UNSECURED_REJOINS | EzspDecisionBitmask.ALLOW_JOINS], ]; type EZSPFrame = { sequence: number; frameId: number; frameName: string; payload: EZSPFrameData; }; type EZSPWaitressMatcher = { sequence: number | null; frameId: number | string; }; export class EZSPFrameData { _cls_: string; _id_: number; _isRequest_: boolean; // biome-ignore lint/suspicious/noExplicitAny: API [name: string]: any; static createFrame(ezspv: number, frameId: number, isRequest: boolean, params: ParamsDesc | Buffer): EZSPFrameData { const names = FRAME_NAMES_BY_ID[frameId]; if (!names) { throw new Error(`Unrecognized frame FrameID ${frameId}`); } let frm: EZSPFrameData; names.every((frameName) => { const frameDesc = EZSPFrameData.getFrame(frameName); if ((frameDesc.maxV && frameDesc.maxV < ezspv) || (frameDesc.minV && frameDesc.minV > ezspv)) { return true; } try { frm = new EZSPFrameData(frameName, isRequest, params); } catch (error) { logger.error(`Frame ${frameName} parsing error: ${error}`, NS); return true; } return false; }); // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress` return frm!; } static getFrame(name: string): EZSPFrameDesc { const frameDesc = FRAMES[name]; if (!frameDesc) throw new Error(`Unrecognized frame from FrameID ${name}`); return frameDesc; } constructor(key: string, isRequest: boolean, params: ParamsDesc | Buffer | undefined) { this._cls_ = key; this._id_ = FRAMES[this._cls_].ID; this._isRequest_ = isRequest; const frame = EZSPFrameData.getFrame(key); const frameDesc = this._isRequest_ ? frame.request || {} : frame.response || {}; if (Buffer.isBuffer(params)) { let data = params; for (const prop of Object.getOwnPropertyNames(frameDesc)) { [this[prop], data] = frameDesc[prop].deserialize(frameDesc[prop], data); } } else { for (const prop of Object.getOwnPropertyNames(frameDesc)) { // biome-ignore lint/style/noNonNullAssertion: ignored using `--suppress` this[prop] = params![prop]; // XXX: assumed defined with logic } } } serialize(): Buffer { const frame = EZSPFrameData.getFrame(this._cls_); const frameDesc = this._isRequest_ ? frame.request || {} : frame.response || {}; const result = []; for (const prop of Object.getOwnPropertyNames(frameDesc)) { result.push(frameDesc[prop].serialize(frameDesc[prop], this[prop])); } return Buffer.concat(result); } get name(): string { return this._cls_; } get id(): number { return this._id_; } } export class EZSPZDORequestFrameData { _cls_: string; _id_: number; _isRequest_: boolean; // biome-ignore lint/suspicious/noExplicitAny: API [name: string]: any; static getFrame(key: string | number): EZSPFrameDesc { const name = typeof key === "string" ? key : ZDOREQUEST_NAME_BY_ID[key]; const frameDesc = ZDOREQUESTS[name]; if (!frameDesc) throw new Error(`Unrecognized ZDOFrame from FrameID ${key}`); return frameDesc; } constructor(key: string | number, isRequest: boolean, params: ParamsDesc | Buffer) { if (typeof key === "string") { this._cls_ = key; this._id_ = ZDOREQUESTS[this._cls_].ID; } else { this._id_ = key; this._cls_ = ZDOREQUEST_NAME_BY_ID[key]; } this._isRequest_ = isRequest; const frame = EZSPZDORequestFrameData.getFrame(key); const frameDesc = this._isRequest_ ? frame.request || {} : frame.response || {}; if (Buffer.isBuffer(params)) { let data = params; for (const prop of Object.getOwnPropertyNames(frameDesc)) { [this[prop], data] = frameDesc[prop].deserialize(frameDesc[prop], data); } } else { for (const prop of Object.getOwnPropertyNames(frameDesc)) { this[prop] = params[prop]; } } } serialize(): Buffer { const frame = EZSPZDORequestFrameData.getFrame(this._cls_); const frameDesc = this._isRequest_ ? frame.request || {} : frame.response || {}; const result = []; for (const prop of Object.getOwnPropertyNames(frameDesc)) { result.push(frameDesc[prop].serialize(frameDesc[prop], this[prop])); } return Buffer.concat(result); } get name(): string { return this._cls_; } get id(): number { return this._id_; } } export class EZSPZDOResponseFrameData { _cls_: string; _id_: number; // biome-ignore lint/suspicious/noExplicitAny: API [name: string]: any; static getFrame(key: string | number): ParamsDesc { const name = typeof key === "string" ? key : ZDORESPONSE_NAME_BY_ID[key]; const frameDesc = ZDORESPONSES[name]; if (!frameDesc) throw new Error(`Unrecognized ZDOFrame from FrameID ${key}`); return frameDesc.params; } constructor(key: string | number, params: ParamsDesc | Buffer) { if (typeof key === "string") { this._cls_ = key; this._id_ = ZDORESPONSES[this._cls_].ID; } else { this._id_ = key; this._cls_ = ZDORESPONSE_NAME_BY_ID[key]; } const frameDesc = EZSPZDOResponseFrameData.getFrame(key); if (Buffer.isBuffer(params)) { let data = params; for (const prop of Object.getOwnPropertyNames(frameDesc)) { [this[prop], data] = frameDesc[prop].deserialize(frameDesc[prop], data); } } else { for (const prop of Object.getOwnPropertyNames(frameDesc)) { this[prop] = params[prop]; } } } serialize(): Buffer { const frameDesc = EZSPZDOResponseFrameData.getFrame(this._cls_); const result = []; for (const prop of Object.getOwnPropertyNames(frameDesc)) { result.push(frameDesc[prop].serialize(frameDesc[prop], this[prop])); } return Buffer.concat(result); } get name(): string { return this._cls_; } get id(): number { return this._id_; } } export class Ezsp extends EventEmitter { ezspV = 4; cmdSeq = 0; // command sequence // COMMANDS_BY_ID = new Map<number, { name: string, inArgs: any[], outArgs: any[] }>(); private serialDriver: SerialDriver; private waitress: Waitress<EZSPFrame, EZSPWaitressMatcher>; private queue: Queue; private watchdogTimer?: NodeJS.Timeout; private failures = 0; private inResetingProcess = false; constructor() { super(); this.queue = new Queue(); this.waitress = new Waitress<EZSPFrame, EZSPWaitressMatcher>(this.waitressValidator, this.waitressTimeoutFormatter); this.serialDriver = new SerialDriver(); this.serialDriver.on("received", this.onFrameReceived.bind(this)); this.serialDriver.on("close", this.onSerialClose.bind(this)); } public async connect(options: SerialPortOptions): Promise<void> { let lastError = null; const resetForReconnect = (): void => { throw new Error("Failure to connect"); }; this.serialDriver.on("reset", resetForReconnect); for (let i = 1; i <= MAX_SERIAL_CONNECT_ATTEMPTS; i++) { try { await this.serialDriver.connect(options); break; } catch (error) { logger.error(`Connection attempt ${i} error: ${error}`, NS); if (i < MAX_SERIAL_CONNECT_ATTEMPTS) { await wait(SERIAL_CONNECT_NEW_ATTEMPT_MIN_DELAY * i); logger.debug(`Next attempt ${i + 1}`, NS); } lastError = error; } } this.serialDriver.off("reset", resetForReconnect); if (!this.serialDriver.isInitialized()) { throw new Error("Failure to connect", {cause: lastError}); } this.inResetingProcess = false; this.serialDriver.on("reset", this.onSerialReset.bind(this)); if (WATCHDOG_WAKE_PERIOD) { this.watchdogTimer = setInterval(this.watchdogHandler.bind(this), WATCHDOG_WAKE_PERIOD * 1000); } } public isInitialized(): boolean { return this.serialDriver?.isInitialized(); } private onSerialReset(): void { logger.debug("onSerialReset()", NS); this.inResetingProcess = true; this.emit("reset"); } private onSerialClose(): void { logger.debug("onSerialClose()", NS); if (!this.inResetingProcess) { this.emit("close"); } } public async close(emitClose: boolean): Promise<void> { logger.debug("Closing Ezsp", NS); clearTimeout(this.watchdogTimer); this.queue.clear(); await this.serialDriver.close(emitClose); } /** * Handle a received EZSP frame * * The protocol has taken care of UART specific framing etc, so we should * just have EZSP application stuff here, with all escaping/stuffing and * data randomization removed. * @param data */ private onFrameReceived(data: Buffer): void { logger.debug(`<== Frame: ${data.toString("hex")}`, NS); let frameId: number; const sequence = data[0]; if (this.ezspV < 8) { [frameId, data] = [data[2], data.subarray(3)]; } else { [[frameId], data] = t.deserialize(data.subarray(3), [t.uint16_t]); } if (frameId === 255) { frameId = 0; if (data.length > 1) { frameId = data[1]; data = data.subarray(2); } } const frm = EZSPFrameData.createFrame(this.ezspV, frameId, false, data); if (!frm) { logger.error(`Unparsed frame 0x${frameId.toString(16)}. Skipped`, NS); return; } logger.debug(() => `<== 0x${frameId.toString(16)}: ${JSON.stringify(frm)}`, NS); const handled = this.waitress.resolve({ frameId, frameName: frm.name, sequence, payload: frm, }); if (!handled) { this.emit("frame", frm.name, frm); } if (frameId === 0) { this.ezspV = frm.protocolVersion; } } async version(): Promise<number> { const version = this.ezspV; const result = await this.execCommand("version", {desiredProtocolVersion: version}); if (result.protocolVersion >= 14) { throw new Error(`'ezsp' driver is not compatible with firmware 8.x.x or above (EZSP v14+). Use 'ember' driver instead.`); } if (result.protocolVersion !== version) { logger.debug(`Switching to eszp version ${result.protocolVersion}`, NS); await this.execCommand("version", {desiredProtocolVersion: result.protocolVersion}); } return result.protocolVersion; } async networkInit(): Promise<boolean> { const waiter = this.waitFor("stackStatusHandler", null); const result = await this.execCommand("networkInit"); logger.debug(`Network init result: ${JSON.stringify(result)}`, NS); if (result.status !== EmberStatus.SUCCESS) { this.waitress.remove(waiter.ID); logger.error("Failure to init network", NS); return false; } const response = await waiter.start().promise; return response.payload.status === EmberStatus.NETWORK_UP; } async leaveNetwork(): Promise<number> { const waiter = this.waitFor("stackStatusHandler", null); const result = await this.execCommand("leaveNetwork"); logger.debug(`Network init result: ${JSON.stringify(result)}`, NS); if (result.status !== EmberStatus.SUCCESS) { this.waitress.remove(waiter.ID); logger.debug("Failure to leave network", NS); throw new Error(`Failure to leave network: ${JSON.stringify(result)}`); } const response = await waiter.start().promise; if (response.payload.status !== EmberStatus.NETWORK_DOWN) { const msg = `Wrong network status: ${JSON.stringify(response.payload)}`; logger.debug(msg, NS); throw new Error(msg); } return response.payload.status; } async setConfigurationValue(configId: number, value: number): Promise<void> { const configName = EzspConfigId.valueToName(EzspConfigId, configId); logger.debug(`Set ${configName} = ${value}`, NS); const ret = await this.execCommand("setConfigurationValue", {configId: configId, value: value}); if (ret.status !== EmberStatus.SUCCESS) { logger.error(`Command (setConfigurationValue(${configName}, ${value})) returned unexpected state: ${JSON.stringify(ret)}`, NS); } } async getConfigurationValue(configId: number): Promise<number> { const configName = EzspConfigId.valueToName(EzspConfigId, configId); logger.debug(`Get ${configName}`, NS); const ret = await this.execCommand("getConfigurationValue", {configId: configId}); if (ret.status !== EmberStatus.SUCCESS) { logger.error(`Command (getConfigurationValue(${configName})) returned unexpected state: ${JSON.stringify(ret)}`, NS); } logger.debug(`Got ${configName} = ${ret.value}`, NS); return ret.value; } async getMulticastTableEntry(index: number): Promise<t.EmberMulticastTableEntry> { const ret = await this.execCommand("getMulticastTableEntry", {index: index}); return ret.value; } async setMulticastTableEntry(index: number, entry: t.EmberMulticastTableEntry): Promise<EmberStatus> { const ret = await this.execCommand("setMulticastTableEntry", {index: index, value: entry}); if (ret.status !== EmberStatus.SUCCESS) { logger.error(`Command (setMulticastTableEntry) returned unexpected state: ${JSON.stringify(ret)}`, NS); } return ret.status; } async setInitialSecurityState(entry: t.EmberInitialSecurityState): Promise<EmberStatus> { const ret = await this.execCommand("setInitialSecurityState", {state: entry}); if (ret.success !== EmberStatus.SUCCESS) { logger.error(`Command (setInitialSecurityState) returned unexpected state: ${JSON.stringify(ret)}`, NS); } return ret.success; } async getCurrentSecurityState(): Promise<EZSPFrameData> { const ret = await this.execCommand("getCurrentSecurityState"); if (ret.status !== EmberStatus.SUCCESS) { logger.error(`Command (getCurrentSecurityState) returned unexpected state: ${JSON.stringify(ret)}`, NS); } return ret; } async setValue(valueId: t.EzspValueId, value: number): Promise<EZSPFrameData> { const valueName = t.EzspValueId.valueToName(t.EzspValueId, valueId); logger.debug(`Set ${valueName} = ${value}`, NS); const ret = await this.execCommand("setValue", {valueId, value}); if (ret.status !== EmberStatus.SUCCESS) { logger.error(`Command (setValue(${valueName}, ${value})) returned unexpected state: ${JSON.stringify(ret)}`, NS); } return ret; } async getValue(valueId: t.EzspValueId): Promise<Buffer> { const valueName = t.EzspValueId.valueToName(t.EzspValueId, valueId); logger.debug(`Get ${valueName}`, NS); const ret = await this.execCommand("getValue", {valueId}); if (ret.status !== EmberStatus.SUCCESS) { logger.error(`Command (getValue(${valueName})) returned unexpected state: ${JSON.stringify(ret)}`, NS); } logger.debug(`Got ${valueName} = ${ret.value}`, NS); return ret.value; } async setPolicy(policyId: EzspPolicyId, value: number): Promise<EZSPFrameData> { const policyName = EzspPolicyId.valueToName(EzspPolicyId, policyId); logger.debug(`Set ${policyName} = ${value}`, NS); const ret = await this.execCommand("setPolicy", {policyId: policyId, decisionId: value}); if (ret.status !== EmberStatus.SUCCESS) { logger.error(`Command (setPolicy(${policyName}, ${value})) returned unexpected state: ${JSON.stringify(ret)}`, NS); } return ret; } async updateConfig(): Promise<void> { const config = this.ezspV < 9 ? CONFIG_IDS_PRE_V9 : CONFIG_IDS_CURRENT; for (const [confName, value] of config) { try { await this.setConfigurationValue(confName, value); } catch (error) { logger.error(`setConfigurationValue(${confName}, ${value}) error: ${error}`, NS); } } } async updatePolicies(): Promise<void> { // Set up the policies for what the NCP should do. const policies = this.ezspV < 8 ? POLICY_IDS_PRE_V8 : POLICY_IDS_CURRENT; for (const [policy, value] of policies) { try { await this.setPolicy(policy, value); } catch (error) { logger.error(`setPolicy(${policy}, ${value}) error: ${error}`, NS); } } } public makeZDOframe(name: string | number, params: ParamsDesc): Buffer { const frmData = new EZSPZDORequestFrameData(name, true, params); return frmData.serialize(); } private makeFrame(name: string, params: ParamsDesc | undefined, seq: number): Buffer { const frmData = new EZSPFrameData(name, true, params); logger.debug(() => `==> ${JSON.stringify(frmData)}`, NS); const frame = [seq & 255]; if (this.ezspV < 8) { if (this.ezspV >= 5) { frame.push(0x00, 0xff, 0x00, frmData.id); } else { frame.push(0x00, frmData.id); } } else { const cmd_id = t.serialize([frmData.id], [t.uint16_t]); frame.push(0x00, 0x01, ...cmd_id); } return Buffer.concat([Buffer.from(frame), frmData.serialize()]); } public async execCommand(name: string, params?: ParamsDesc): Promise<EZSPFrameData> { logger.debug(() => `==> ${name}: ${JSON.stringify(params)}`, NS); if (!this.serialDriver.isInitialized()) { throw new Error("Connection not initialized"); } return await this.queue.execute<EZSPFrameData>(async (): Promise<EZSPFrameData> => { const data = this.makeFrame(name, params, this.cmdSeq); const waiter = this.waitFor(name, this.cmdSeq); this.cmdSeq = (this.cmdSeq + 1) & 255; try { await this.serialDriver.sendDATA(data); const response = await waiter.start().promise; return response.payload; } catch { this.waitress.remove(waiter.ID); throw new Error(`Failure send ${name}:${JSON.stringify(data)}`); } }); } async formNetwork(params: EmberNetworkParameters): Promise<number> { const waiter = this.waitFor("stackStatusHandler", null); const v = await this.execCommand("formNetwork", {parameters: params}); if (v.status !== EmberStatus.SUCCESS) { this.waitress.remove(waiter.ID); logger.error(`Failure forming network: ${JSON.stringify(v)}`, NS); throw new Error(`Failure forming network: ${JSON.stringify(v)}`); } const response = await waiter.start().promise; if (response.payload.status !== EmberStatus.NETWORK_UP) { logger.error(`Wrong network status: ${JSON.stringify(response.payload)}`, NS); throw new Error(`Wrong network status: ${JSON.stringify(response.payload)}`); } return response.payload.status; } public sendUnicast(direct: EmberOutgoingMessageType, nwk: number, apsFrame: EmberApsFrame, seq: number, data: Buffer): Promise<EZSPFrameData> { return this.execCommand("sendUnicast", { type: direct, indexOrDestination: nwk, apsFrame: apsFrame, messageTag: seq, message: data, }); } public sendMulticast(apsFrame: EmberApsFrame, seq: number, data: Buffer): Promise<EZSPFrameData> { return this.execCommand("sendMulticast", { apsFrame: apsFrame, hops: EZSP_DEFAULT_RADIUS, nonmemberRadius: EZSP_MULTICAST_NON_MEMBER_RADIUS, messageTag: seq, message: data, }); } public async setSourceRouting(): Promise<void> { const res = await this.execCommand("setConcentrator", { on: true, concentratorType: EmberConcentratorType.HIGH_RAM_CONCENTRATOR, minTime: MTOR_MIN_INTERVAL, maxTime: MTOR_MAX_INTERVAL, routeErrorThreshold: MTOR_ROUTE_ERROR_THRESHOLD, deliveryFailureThreshold: MTOR_DELIVERY_FAIL_THRESHOLD, maxHops: 0, }); logger.debug(`Set concentrator type: ${JSON.stringify(res)}`, NS); if (res.status !== EmberStatus.SUCCESS) { logger.error(`Couldn't set concentrator ${JSON.stringify(res)}`, NS); } if (this.ezspV >= 8) { await this.execCommand("setSourceRouteDiscoveryMode", {mode: 1}); } } public sendBroadcast(destination: number, apsFrame: EmberApsFrame, seq: number, data: Buffer): Promise<EZSPFrameData> { return this.execCommand("sendBroadcast", { destination: destination, apsFrame: apsFrame, radius: EZSP_DEFAULT_RADIUS, messageTag: seq, message: data, }); } public waitFor( frameId: string | number, sequence: number | null, timeout = 10000, ): {start: () => {promise: Promise<EZSPFrame>; ID: number}; ID: number} { return this.waitress.waitFor({frameId, sequence}, timeout); } private waitressTimeoutFormatter(matcher: EZSPWaitressMatcher, timeout: number): string { return `${JSON.stringify(matcher)} after ${timeout}ms`; } private waitressValidator(payload: EZSPFrame, matcher: EZSPWaitressMatcher): boolean { const frameNames = typeof matcher.frameId === "string" ? [matcher.frameId] : FRAME_NAMES_BY_ID[matcher.frameId]; return (matcher.sequence == null || payload.sequence === matcher.sequence) && frameNames.includes(payload.frameName); } private async watchdogHandler(): Promise<void> { logger.debug(`Time to watchdog ... ${this.failures}`, NS); if (this.inResetingProcess) { logger.debug("The reset process is in progress...", NS); return; } try { await this.execCommand("nop"); } catch (error) { logger.error(`Watchdog heartbeat timeout ${error}`, NS); if (!this.inResetingProcess) { this.failures += 1; if (this.failures > MAX_WATCHDOG_FAILURES) { this.failures = 0; this.emit("reset"); } } } } }