zigbee2mqtt
Version:
Zigbee to MQTT bridge using Zigbee-herdsman
328 lines • 33.5 kB
JavaScript
"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 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"));
const RETRIEVE_ON_RECONNECT = [
{ keys: ["state"] },
{ keys: ["brightness"], condition: (state) => state.state === "ON" },
{ keys: ["color", "color_temp"], condition: (state) => state.state === "ON" },
];
class Availability extends extension_1.default {
/** Mapped by IEEE address */
timers = new Map();
/** Mapped by IEEE address or Group ID */
lastPublishedAvailabilities = new Map();
/** Mapped by IEEE address */
pingBackoffs = new Map();
/** IEEE addresses, waiting for last seen changes to take them out of "availability sleep" */
backoffPausedDevices = new Set();
/** Mapped by IEEE address */
retrieveStateDebouncers = new Map();
pingQueue = [];
pingQueueExecuting = false;
stopped = false;
getTimeout(device) {
if (typeof device.options.availability === "object" && device.options.availability?.timeout != null) {
return utils_1.default.minutes(device.options.availability.timeout);
}
return utils_1.default.minutes(this.isActiveDevice(device) ? settings.get().availability.active.timeout : settings.get().availability.passive.timeout);
}
getMaxJitter(device) {
if (typeof device.options.availability === "object" && device.options.availability?.max_jitter != null) {
return device.options.availability.max_jitter;
}
return settings.get().availability.active.max_jitter;
}
getBackoff(device) {
if (typeof device.options.availability === "object" && device.options.availability?.backoff != null) {
return device.options.availability.backoff;
}
return settings.get().availability.active.backoff;
}
getPauseOnBackoffGt(device) {
if (typeof device.options.availability === "object" && device.options.availability?.pause_on_backoff_gt != null) {
return device.options.availability.pause_on_backoff_gt;
}
return settings.get().availability.active.pause_on_backoff_gt;
}
isActiveDevice(device) {
return ((device.zh.type === "Router" && device.zh.powerSource !== "Battery") ||
(device.zh.powerSource !== undefined && device.zh.powerSource !== "Unknown" && device.zh.powerSource !== "Battery"));
}
isAvailable(entity) {
if (entity.isDevice()) {
const lastSeen = entity.zh.lastSeen ?? /* v8 ignore next */ 0;
return Date.now() - lastSeen < this.getTimeout(entity) + this.getMaxJitter(entity);
}
for (const memberDevice of entity.membersDevices()) {
if (this.lastPublishedAvailabilities.get(memberDevice.ieeeAddr) === true) {
return true;
}
}
return false;
}
resetTimer(device, resetBackoff = false) {
clearTimeout(this.timers.get(device.ieeeAddr));
this.removeFromPingQueue(device);
// If the timer triggers, the device is not available anymore otherwise resetTimer already has been called
if (this.isActiveDevice(device)) {
const backoffEnabled = this.getBackoff(device);
const jitter = Math.random() * this.getMaxJitter(device);
let backoff = 1;
if (resetBackoff) {
// always cleanup even if backoff disabled (ensures proper state if changed at runtime)
this.backoffPausedDevices.delete(device.ieeeAddr);
this.pingBackoffs.delete(device.ieeeAddr);
}
else if (backoffEnabled) {
backoff = this.pingBackoffs.get(device.ieeeAddr) ?? 1;
}
// never paused if was reset (just deleted) or backoff disabled, might as well skip the Set lookup
if (!backoffEnabled || resetBackoff || !this.backoffPausedDevices.has(device.ieeeAddr)) {
// If device did not check in, ping it, if that fails it will be marked as offline
this.timers.set(device.ieeeAddr, setTimeout(this.addToPingQueue.bind(this, device), (this.getTimeout(device) + utils_1.default.seconds(1) + jitter) * backoff));
}
}
else {
this.timers.set(device.ieeeAddr, setTimeout(this.publishAvailability.bind(this, device, true), this.getTimeout(device) + utils_1.default.seconds(1)));
}
}
clearTimer(ieeeAddress) {
clearTimeout(this.timers.get(ieeeAddress));
this.timers.delete(ieeeAddress);
}
addToPingQueue(device) {
this.pingQueue.push(device);
this.pingQueueExecuteNext().catch(utils_1.default.noop);
}
removeFromPingQueue(device) {
const index = this.pingQueue.findIndex((d) => d.ieeeAddr === device.ieeeAddr);
if (index !== -1) {
this.pingQueue.splice(index, 1);
}
}
async pingQueueExecuteNext() {
if (this.pingQueue.length === 0 || this.pingQueueExecuting) {
return;
}
this.pingQueueExecuting = true;
const device = this.pingQueue[0];
let pingSuccess = false;
const available = this.lastPublishedAvailabilities.get(device.ieeeAddr) || this.isAvailable(device);
const attempts = available ? 2 : 1;
for (let i = 1; i <= attempts; i++) {
try {
// Enable recovery if device is marked as available and first ping fails.
await device.zh.ping(!available || i !== 2);
pingSuccess = true;
logger_1.default.debug(`Successfully pinged '${device.name}' (attempt ${i}/${attempts})`);
break;
}
catch (error) {
logger_1.default.warning(`Failed to ping '${device.name}' (attempt ${i}/${attempts}, ${error.message})`);
// Try again in 3 seconds.
if (i !== attempts) {
await utils_1.default.sleep(3);
}
}
}
if (this.stopped) {
// Exit here to avoid triggering any follow-up activity (e.g., re-queuing another ping attempt).
return;
}
if (!pingSuccess && this.getBackoff(device)) {
const currentBackoff = this.pingBackoffs.get(device.ieeeAddr) ?? 1;
// setting is "greater than" but since we already did the ping, we use ">=" for comparison below (pause next)
const pauseOnBackoff = this.getPauseOnBackoffGt(device);
if (pauseOnBackoff > 0 && currentBackoff >= pauseOnBackoff) {
this.backoffPausedDevices.add(device.ieeeAddr);
}
else {
// results in backoffs: *1.5, *3, *6, *12... (with default timeout: 10, 15, 30, 60, 120)
this.pingBackoffs.set(device.ieeeAddr, currentBackoff * (available ? 1.5 : 2));
}
}
await this.publishAvailability(device, !pingSuccess);
this.resetTimer(device, pingSuccess);
this.removeFromPingQueue(device);
// Sleep 2 seconds before executing next ping
await utils_1.default.sleep(2);
this.pingQueueExecuting = false;
await this.pingQueueExecuteNext();
}
async start() {
if (this.stopped) {
throw new Error("This extension cannot be restarted.");
}
this.eventBus.onEntityRenamed(this, async (data) => {
if (utils_1.default.isAvailabilityEnabledForEntity(data.entity, settings.get())) {
await this.mqtt.publish(`${data.from}/availability`, "", { clientOptions: { retain: true, qos: 1 } });
await this.publishAvailability(data.entity, false, true);
}
});
this.eventBus.onEntityRemoved(this, (data) => data.type === "device" && this.clearTimer(data.id));
this.eventBus.onDeviceLeave(this, (data) => this.clearTimer(data.ieeeAddr));
this.eventBus.onDeviceAnnounce(this, (data) => this.retrieveState(data.device));
this.eventBus.onLastSeenChanged(this, this.onLastSeenChanged);
this.eventBus.onGroupMembersChanged(this, (data) => this.publishAvailability(data.group, false));
// Publish initial availability
await this.publishAvailabilityForAllEntities();
// Start availability for the devices
for (const device of this.zigbee.devicesIterator(utils_1.default.deviceNotCoordinator)) {
if (utils_1.default.isAvailabilityEnabledForEntity(device, settings.get())) {
this.resetTimer(device);
// If an active device is unavailable on start, add it to the pingqueue immediately.
if (this.isActiveDevice(device) && !this.isAvailable(device)) {
this.addToPingQueue(device);
}
}
}
}
async publishAvailabilityForAllEntities() {
for (const entity of this.zigbee.devicesAndGroupsIterator(utils_1.default.deviceNotCoordinator)) {
if (utils_1.default.isAvailabilityEnabledForEntity(entity, settings.get())) {
await this.publishAvailability(entity, true, false, true);
}
}
}
async publishAvailability(entity, logLastSeen, forcePublish = false, skipGroups = false) {
if (logLastSeen && entity.isDevice()) {
const ago = Date.now() - (entity.zh.lastSeen ?? /* v8 ignore next */ 0);
if (this.isActiveDevice(entity)) {
logger_1.default.debug(`Active device '${entity.name}' was last seen '${(ago / utils_1.default.minutes(1)).toFixed(2)}' minutes ago.`);
}
else {
logger_1.default.debug(`Passive device '${entity.name}' was last seen '${(ago / utils_1.default.hours(1)).toFixed(2)}' hours ago.`);
}
}
const available = this.isAvailable(entity);
if (!forcePublish && this.lastPublishedAvailabilities.get(entity.ID) === available) {
return;
}
if (entity.isDevice() && available && this.lastPublishedAvailabilities.get(entity.ieeeAddr) === false) {
logger_1.default.debug(`Device '${entity.name}' reconnected`);
this.retrieveState(entity);
}
const topic = `${entity.name}/availability`;
const payload = { state: available ? "online" : "offline" };
this.lastPublishedAvailabilities.set(entity.ID, available);
await this.mqtt.publish(topic, JSON.stringify(payload), { clientOptions: { retain: true, qos: 1 } });
if (!skipGroups && entity.isDevice()) {
for (const group of this.zigbee.groupsIterator()) {
if (group.hasMember(entity) && utils_1.default.isAvailabilityEnabledForEntity(group, settings.get())) {
await this.publishAvailability(group, false, forcePublish);
}
}
}
}
async onLastSeenChanged(data) {
if (utils_1.default.isAvailabilityEnabledForEntity(data.device, settings.get())) {
// Remove from ping queue, not necessary anymore since we know the device is online.
this.removeFromPingQueue(data.device);
this.resetTimer(data.device, true);
await this.publishAvailability(data.device, false);
}
}
async stop() {
this.stopped = true;
this.pingQueue = [];
for (const [, t] of this.timers) {
clearTimeout(t);
}
await super.stop();
}
retrieveState(device) {
/**
* Retrieve state of a device in a debounced manner, this function is called on a 'deviceAnnounce' which a
* device can send multiple times after each other.
*/
if (device.definition && device.interviewed && !this.retrieveStateDebouncers.get(device.ieeeAddr)) {
this.retrieveStateDebouncers.set(device.ieeeAddr, (0, debounce_1.default)(async () => {
logger_1.default.debug(`Retrieving state of '${device.name}' after reconnect`);
// Color and color temperature converters do both, only needs to be called once.
for (const item of RETRIEVE_ON_RECONNECT) {
if (item.condition && this.state.get(device) && !item.condition(this.state.get(device))) {
continue;
}
// biome-ignore lint/style/noNonNullAssertion: doesn't change once valid
const converter = device.definition.toZigbee.find((c) => !c.key || c.key.find((k) => item.keys.includes(k)));
const options = device.options;
const state = this.state.get(device);
const meta = {
message: this.state.get(device),
// biome-ignore lint/style/noNonNullAssertion: doesn't change once valid
mapped: device.definition,
endpoint_name: undefined,
options,
state,
device: device.zh,
/* v8 ignore next */
publish: (payload) => this.publishEntityState(device, payload),
};
try {
const endpoint = device.endpoint();
(0, node_assert_1.default)(endpoint);
await converter?.convertGet?.(endpoint, item.keys[0], meta);
}
catch (error) {
logger_1.default.error(`Failed to read state of '${device.name}' after reconnect (${error.message})`);
}
await utils_1.default.sleep(500);
}
}, utils_1.default.seconds(2)));
}
this.retrieveStateDebouncers.get(device.ieeeAddr)?.();
}
}
exports.default = Availability;
__decorate([
bind_decorator_1.default
], Availability.prototype, "onLastSeenChanged", null);
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"availability.js","sourceRoot":"","sources":["../../lib/extension/availability.ts"],"names":[],"mappings":"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"}