UNPKG

zigbee2mqtt

Version:

Zigbee to MQTT bridge using Zigbee-herdsman

214 lines 19.8 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const node_assert_1 = __importDefault(require("node:assert")); const bind_decorator_1 = __importDefault(require("bind-decorator")); const debounce_1 = __importDefault(require("debounce")); const json_stable_stringify_without_jsonify_1 = __importDefault(require("json-stable-stringify-without-jsonify")); const throttleit_1 = __importDefault(require("throttleit")); const zhc = __importStar(require("zigbee-herdsman-converters")); const logger_1 = __importDefault(require("../util/logger")); const settings = __importStar(require("../util/settings")); const utils_1 = __importDefault(require("../util/utils")); const extension_1 = __importDefault(require("./extension")); class Receive extends extension_1.default { // TODO: move all to `Map` elapsed = {}; debouncers = {}; throttlers = {}; // biome-ignore lint/suspicious/useAwait: API async start() { this.eventBus.onPublishEntityState(this, this.onPublishEntityState); this.eventBus.onDeviceMessage(this, this.onDeviceMessage); } onPublishEntityState(data) { /** * Prevent that outdated properties are being published. * In case that e.g. the state is currently held back by a debounce and a new state is published * remove it from the to be send debounced message. */ if (data.entity.isDevice() && this.debouncers[data.entity.ieeeAddr] && data.stateChangeReason !== "publishDebounce" && data.stateChangeReason !== "lastSeenChanged") { for (const key of Object.keys(data.payload)) { delete this.debouncers[data.entity.ieeeAddr].payload[key]; } } } publishDebounce(device, payload, time, debounceIgnore) { if (!this.debouncers[device.ieeeAddr]) { this.debouncers[device.ieeeAddr] = { payload: {}, publish: (0, debounce_1.default)(async () => { await this.publishEntityState(device, this.debouncers[device.ieeeAddr].payload, "publishDebounce"); this.debouncers[device.ieeeAddr].payload = {}; }, time * 1000), }; } if (this.isPayloadConflicted(payload, this.debouncers[device.ieeeAddr].payload, debounceIgnore)) { // publish previous payload immediately this.debouncers[device.ieeeAddr].publish.flush(); } // extend debounced payload with current this.debouncers[device.ieeeAddr].payload = { ...this.debouncers[device.ieeeAddr].payload, ...payload }; // Update state cache right away. This makes sure that during debouncing cached state is always up to date. // ( Update right away as "lastSeenChanged" event might occur while debouncer is still active. // And if that happens it would cause old message to be published from cache. // By updating cache we make sure that state cache is always up-to-date. this.state.set(device, this.debouncers[device.ieeeAddr].payload); this.debouncers[device.ieeeAddr].publish(); } async publishThrottle(device, payload, time) { if (!this.throttlers[device.ieeeAddr]) { this.throttlers[device.ieeeAddr] = { publish: (0, throttleit_1.default)(this.publishEntityState, time * 1000), }; } // Update state cache right away. This makes sure that during throttling cached state is always up to date. // By updating cache we make sure that state cache is always up-to-date. this.state.set(device, payload); await this.throttlers[device.ieeeAddr].publish(device, payload, "publishThrottle"); } // if debounce_ignore are specified (Array of strings) // then all newPayload values with key present in debounce_ignore // should equal or be undefined in oldPayload // otherwise payload is conflicted isPayloadConflicted(newPayload, oldPayload, debounceIgnore) { let result = false; for (const key in oldPayload) { if (debounceIgnore?.includes(key) && typeof newPayload[key] !== "undefined" && newPayload[key] !== oldPayload[key]) { result = true; } } return result; } async onDeviceMessage(data) { /* v8 ignore next */ if (!data.device) return; if (!data.device.definition || !data.device.interviewed) { logger_1.default.debug("Skipping message, still interviewing"); await utils_1.default.publishLastSeen({ device: data.device, reason: "messageEmitted" }, settings.get(), true, this.publishEntityState); return; } const converters = data.device.definition.fromZigbee.filter((c) => { const type = Array.isArray(c.type) ? c.type.includes(data.type) : c.type === data.type; return c.cluster === data.cluster && type; }); // Check if there is an available converter, genOta messages are not interesting. const ignoreClusters = ["genOta", "genTime", "genBasic", "genPollCtrl"]; if (converters.length === 0 && !ignoreClusters.includes(data.cluster)) { logger_1.default.debug(`No converter available for '${data.device.definition.model}' with ` + `cluster '${data.cluster}' and type '${data.type}' and data '${(0, json_stable_stringify_without_jsonify_1.default)(data.data)}'`); await utils_1.default.publishLastSeen({ device: data.device, reason: "messageEmitted" }, settings.get(), true, this.publishEntityState); return; } // Convert this Zigbee message to a MQTT message. // Get payload for the message. // - If a payload is returned publish it to the MQTT broker // - If NO payload is returned do nothing. This is for non-standard behaviour // for e.g. click switches where we need to count number of clicks and detect long presses. const publish = async (payload) => { (0, node_assert_1.default)(data.device.definition); const options = data.device.options; zhc.postProcessConvertedFromZigbeeMessage(data.device.definition, payload, options, data.device.zh); if (settings.get().advanced.elapsed) { const now = Date.now(); if (this.elapsed[data.device.ieeeAddr]) { payload.elapsed = now - this.elapsed[data.device.ieeeAddr]; } this.elapsed[data.device.ieeeAddr] = now; } // Check if we have to debounce or throttle if (data.device.options.debounce) { this.publishDebounce(data.device, payload, data.device.options.debounce, data.device.options.debounce_ignore); } else if (data.device.options.throttle) { await this.publishThrottle(data.device, payload, data.device.options.throttle); } else { await this.publishEntityState(data.device, payload); } }; const meta = { device: data.device.zh, logger: logger_1.default, state: this.state.get(data.device), deviceExposesChanged: () => this.eventBus.emitExposesAndDevicesChanged(data.device), }; let payload = {}; for (const converter of converters) { try { const convertData = { ...data, device: data.device.zh }; const options = data.device.options; const converted = await converter.convert(data.device.definition, convertData, publish, options, meta); if (converted) { payload = { ...payload, ...converted }; } /* v8 ignore start */ } catch (error) { logger_1.default.error(`Exception while calling fromZigbee converter: ${error.message}}`); // biome-ignore lint/style/noNonNullAssertion: always Error logger_1.default.debug(error.stack); } /* v8 ignore stop */ } if (!utils_1.default.objectIsEmpty(payload)) { await publish(payload); } else { await utils_1.default.publishLastSeen({ device: data.device, reason: "messageEmitted" }, settings.get(), true, this.publishEntityState); } } } exports.default = Receive; __decorate([ bind_decorator_1.default ], Receive.prototype, "onPublishEntityState", null); __decorate([ bind_decorator_1.default ], Receive.prototype, "onDeviceMessage", null); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"receive.js","sourceRoot":"","sources":["../../lib/extension/receive.ts"],"names":[],"mappings":"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"}