mirror of
https://github.com/Koenkk/zigbee2mqtt.git
synced 2026-07-04 11:01:40 +00:00
30227a13ae
* chore: Implement prettier * Run prettier * fix lint * process feedback * process feedback
449 lines
18 KiB
TypeScript
449 lines
18 KiB
TypeScript
import bind from 'bind-decorator';
|
|
import equals from 'fast-deep-equal/es6';
|
|
import stringify from 'json-stable-stringify-without-jsonify';
|
|
import * as zhc from 'zigbee-herdsman-converters';
|
|
|
|
import Device from '../model/device';
|
|
import Group from '../model/group';
|
|
import logger from '../util/logger';
|
|
import * as settings from '../util/settings';
|
|
import utils from '../util/utils';
|
|
import Extension from './extension';
|
|
|
|
const TOPIC_REGEX = new RegExp(`^${settings.get().mqtt.base_topic}/bridge/request/group/members/(remove|add|remove_all)$`);
|
|
const LEGACY_TOPIC_REGEX = new RegExp(`^${settings.get().mqtt.base_topic}/bridge/group/(.+)/(remove|add|remove_all)$`);
|
|
const LEGACY_TOPIC_REGEX_REMOVE_ALL = new RegExp(`^${settings.get().mqtt.base_topic}/bridge/group/remove_all$`);
|
|
|
|
const STATE_PROPERTIES: Readonly<Record<string, (value: string, exposes: zhc.Expose[]) => boolean>> = {
|
|
state: () => true,
|
|
brightness: (value, exposes) => exposes.some((e) => e.type === 'light' && e.features.some((f) => f.name === 'brightness')),
|
|
color_temp: (value, exposes) => exposes.some((e) => e.type === 'light' && e.features.some((f) => f.name === 'color_temp')),
|
|
color: (value, exposes) => exposes.some((e) => e.type === 'light' && e.features.some((f) => f.name === 'color_xy' || f.name === 'color_hs')),
|
|
color_mode: (value, exposes) =>
|
|
exposes.some(
|
|
(e) =>
|
|
e.type === 'light' &&
|
|
(e.features.some((f) => f.name === `color_${value}`) || (value === 'color_temp' && e.features.some((f) => f.name === 'color_temp'))),
|
|
),
|
|
};
|
|
|
|
interface ParsedMQTTMessage {
|
|
type: 'remove' | 'add' | 'remove_all';
|
|
resolvedEntityGroup: Group;
|
|
resolvedEntityDevice: Device;
|
|
error: string;
|
|
groupKey: string;
|
|
deviceKey: string;
|
|
triggeredViaLegacyApi: boolean;
|
|
skipDisableReporting: boolean;
|
|
resolvedEntityEndpoint: zh.Endpoint;
|
|
}
|
|
|
|
export default class Groups extends Extension {
|
|
private legacyApi = settings.get().advanced.legacy_api;
|
|
private lastOptimisticState: {[s: string]: KeyValue} = {};
|
|
|
|
override async start(): Promise<void> {
|
|
this.eventBus.onStateChange(this, this.onStateChange);
|
|
this.eventBus.onMQTTMessage(this, this.onMQTTMessage);
|
|
await this.syncGroupsWithSettings();
|
|
}
|
|
|
|
private async syncGroupsWithSettings(): Promise<void> {
|
|
const settingsGroups = settings.getGroups();
|
|
const zigbeeGroups = this.zigbee.groups();
|
|
|
|
const addRemoveFromGroup = async (
|
|
action: 'add' | 'remove',
|
|
deviceName: string,
|
|
groupName: string | number,
|
|
endpoint: zh.Endpoint,
|
|
group: Group,
|
|
): Promise<void> => {
|
|
try {
|
|
logger.info(`${action === 'add' ? 'Adding' : 'Removing'} '${deviceName}' to group '${groupName}'`);
|
|
|
|
if (action === 'remove') {
|
|
await endpoint.removeFromGroup(group.zh);
|
|
} else {
|
|
await endpoint.addToGroup(group.zh);
|
|
}
|
|
} catch (error) {
|
|
logger.error(`Failed to ${action} '${deviceName}' from '${groupName}'`);
|
|
logger.debug(error.stack);
|
|
}
|
|
};
|
|
|
|
for (const settingGroup of settingsGroups) {
|
|
const groupID = settingGroup.ID;
|
|
const zigbeeGroup = zigbeeGroups.find((g) => g.ID === groupID) || this.zigbee.createGroup(groupID);
|
|
const settingsEndpoints: zh.Endpoint[] = [];
|
|
|
|
for (const d of settingGroup.devices) {
|
|
const parsed = this.zigbee.resolveEntityAndEndpoint(d);
|
|
const device = parsed.entity as Device;
|
|
|
|
if (!device) {
|
|
logger.error(`Cannot find '${d}' of group '${settingGroup.friendly_name}'`);
|
|
}
|
|
|
|
if (!parsed.endpoint) {
|
|
if (parsed.endpointID) {
|
|
logger.error(`Cannot find endpoint '${parsed.endpointID}' of device '${parsed.ID}'`);
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
// In settings but not in zigbee
|
|
if (!zigbeeGroup.zh.hasMember(parsed.endpoint)) {
|
|
await addRemoveFromGroup('add', device?.name, settingGroup.friendly_name, parsed.endpoint, zigbeeGroup);
|
|
}
|
|
|
|
settingsEndpoints.push(parsed.endpoint);
|
|
}
|
|
|
|
// In zigbee but not in settings
|
|
for (const endpoint of zigbeeGroup.zh.members) {
|
|
if (!settingsEndpoints.includes(endpoint)) {
|
|
const deviceName = settings.getDevice(endpoint.getDevice().ieeeAddr).friendly_name;
|
|
|
|
await addRemoveFromGroup('remove', deviceName, settingGroup.friendly_name, endpoint, zigbeeGroup);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const zigbeeGroup of zigbeeGroups) {
|
|
if (!settingsGroups.some((g) => g.ID === zigbeeGroup.ID)) {
|
|
for (const endpoint of zigbeeGroup.zh.members) {
|
|
const deviceName = settings.getDevice(endpoint.getDevice().ieeeAddr).friendly_name;
|
|
|
|
await addRemoveFromGroup('remove', deviceName, zigbeeGroup.ID, endpoint, zigbeeGroup);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@bind async onStateChange(data: eventdata.StateChange): Promise<void> {
|
|
const reason = 'groupOptimistic';
|
|
|
|
if (data.reason === reason || data.reason === 'publishCached') {
|
|
return;
|
|
}
|
|
|
|
const payload: KeyValue = {};
|
|
let endpointName: string = null;
|
|
const endpointNames: string[] = data.entity instanceof Device ? data.entity.getEndpointNames() : [];
|
|
|
|
for (let prop of Object.keys(data.update)) {
|
|
const value = data.update[prop];
|
|
const endpointNameMatch = endpointNames.find((n) => prop.endsWith(`_${n}`));
|
|
|
|
if (endpointNameMatch) {
|
|
prop = prop.substring(0, prop.length - endpointNameMatch.length - 1);
|
|
endpointName = endpointNameMatch;
|
|
}
|
|
|
|
if (prop in STATE_PROPERTIES) {
|
|
payload[prop] = value;
|
|
}
|
|
}
|
|
|
|
const payloadKeys = Object.keys(payload);
|
|
|
|
if (payloadKeys.length) {
|
|
const entity = data.entity;
|
|
const groups = this.zigbee.groups().filter((g) => g.options && (g.options.optimistic == undefined || g.options.optimistic));
|
|
|
|
if (entity instanceof Device) {
|
|
for (const group of groups) {
|
|
if (
|
|
group.zh.hasMember(entity.endpoint(endpointName)) &&
|
|
!equals(this.lastOptimisticState[group.ID], payload) &&
|
|
this.shouldPublishPayloadForGroup(group, payload)
|
|
) {
|
|
this.lastOptimisticState[group.ID] = payload;
|
|
|
|
await this.publishEntityState(group, payload, reason);
|
|
}
|
|
}
|
|
} else {
|
|
// Invalidate the last optimistic group state when group state is changed directly.
|
|
delete this.lastOptimisticState[entity.ID];
|
|
|
|
const groupsToPublish: Set<Group> = new Set();
|
|
|
|
for (const member of entity.zh.members) {
|
|
const device = this.zigbee.resolveEntity(member.getDevice()) as Device;
|
|
|
|
if (device.options.disabled) {
|
|
continue;
|
|
}
|
|
|
|
const exposes = device.exposes();
|
|
const memberPayload: KeyValue = {};
|
|
|
|
for (const key of payloadKeys) {
|
|
if (STATE_PROPERTIES[key](payload[key], exposes)) {
|
|
memberPayload[key] = payload[key];
|
|
}
|
|
}
|
|
|
|
const endpointName = device.endpointName(member);
|
|
|
|
if (endpointName) {
|
|
for (const key of Object.keys(memberPayload)) {
|
|
memberPayload[`${key}_${endpointName}`] = memberPayload[key];
|
|
delete memberPayload[key];
|
|
}
|
|
}
|
|
|
|
await this.publishEntityState(device, memberPayload, reason);
|
|
|
|
for (const zigbeeGroup of groups) {
|
|
if (zigbeeGroup.zh.hasMember(member) && this.shouldPublishPayloadForGroup(zigbeeGroup, payload)) {
|
|
groupsToPublish.add(zigbeeGroup);
|
|
}
|
|
}
|
|
}
|
|
|
|
groupsToPublish.delete(entity);
|
|
|
|
for (const group of groupsToPublish) {
|
|
await this.publishEntityState(group, payload, reason);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private shouldPublishPayloadForGroup(group: Group, payload: KeyValue): boolean {
|
|
return group.options.off_state === 'last_member_state' || !payload || payload.state !== 'OFF' || this.areAllMembersOff(group);
|
|
}
|
|
|
|
private areAllMembersOff(group: Group): boolean {
|
|
for (const member of group.zh.members) {
|
|
const device = this.zigbee.resolveEntity(member.getDevice());
|
|
|
|
if (this.state.exists(device)) {
|
|
const state = this.state.get(device);
|
|
|
|
if (state.state === 'ON') {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private async parseMQTTMessage(data: eventdata.MQTTMessage): Promise<ParsedMQTTMessage> {
|
|
let type: ParsedMQTTMessage['type'] = null;
|
|
let resolvedEntityGroup: ParsedMQTTMessage['resolvedEntityGroup'] = null;
|
|
let resolvedEntityDevice: ParsedMQTTMessage['resolvedEntityDevice'] = null;
|
|
let resolvedEntityEndpoint: ParsedMQTTMessage['resolvedEntityEndpoint'] = null;
|
|
let error: ParsedMQTTMessage['error'] = null;
|
|
let groupKey: ParsedMQTTMessage['groupKey'] = null;
|
|
let deviceKey: ParsedMQTTMessage['deviceKey'] = null;
|
|
let triggeredViaLegacyApi: ParsedMQTTMessage['triggeredViaLegacyApi'] = false;
|
|
let skipDisableReporting: ParsedMQTTMessage['skipDisableReporting'] = false;
|
|
|
|
/* istanbul ignore else */
|
|
const topicRegexMatch = data.topic.match(TOPIC_REGEX);
|
|
const legacyTopicRegexRemoveAllMatch = data.topic.match(LEGACY_TOPIC_REGEX_REMOVE_ALL);
|
|
const legacyTopicRegexMatch = data.topic.match(LEGACY_TOPIC_REGEX);
|
|
|
|
if (this.legacyApi && (legacyTopicRegexMatch || legacyTopicRegexRemoveAllMatch)) {
|
|
triggeredViaLegacyApi = true;
|
|
|
|
if (legacyTopicRegexMatch) {
|
|
resolvedEntityGroup = this.zigbee.resolveEntity(legacyTopicRegexMatch[1]) as Group;
|
|
type = legacyTopicRegexMatch[2] as ParsedMQTTMessage['type'];
|
|
|
|
if (!resolvedEntityGroup || !(resolvedEntityGroup instanceof Group)) {
|
|
logger.error(`Group '${legacyTopicRegexMatch[1]}' does not exist`);
|
|
|
|
/* istanbul ignore else */
|
|
if (settings.get().advanced.legacy_api) {
|
|
const message = {friendly_name: data.message, group: legacyTopicRegexMatch[1], error: `group doesn't exists`};
|
|
|
|
await this.mqtt.publish('bridge/log', stringify({type: `device_group_${type}_failed`, message}));
|
|
}
|
|
|
|
return null;
|
|
}
|
|
} else {
|
|
type = 'remove_all';
|
|
}
|
|
|
|
const parsedEntity = this.zigbee.resolveEntityAndEndpoint(data.message);
|
|
resolvedEntityDevice = parsedEntity.entity as Device;
|
|
|
|
if (!resolvedEntityDevice || !(resolvedEntityDevice instanceof Device)) {
|
|
logger.error(`Device '${data.message}' does not exist`);
|
|
|
|
/* istanbul ignore else */
|
|
if (settings.get().advanced.legacy_api) {
|
|
const message = {friendly_name: data.message, group: legacyTopicRegexMatch[1], error: "entity doesn't exists"};
|
|
|
|
await this.mqtt.publish('bridge/log', stringify({type: `device_group_${type}_failed`, message}));
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
resolvedEntityEndpoint = parsedEntity.endpoint;
|
|
|
|
if (parsedEntity.endpointID && !resolvedEntityEndpoint) {
|
|
logger.error(`Device '${parsedEntity.ID}' does not have endpoint '${parsedEntity.endpointID}'`);
|
|
return null;
|
|
}
|
|
} else if (topicRegexMatch) {
|
|
type = topicRegexMatch[1] as 'remove' | 'add' | 'remove_all';
|
|
const message = JSON.parse(data.message);
|
|
deviceKey = message.device;
|
|
skipDisableReporting = 'skip_disable_reporting' in message ? message.skip_disable_reporting : false;
|
|
|
|
if (type !== 'remove_all') {
|
|
groupKey = message.group;
|
|
resolvedEntityGroup = this.zigbee.resolveEntity(message.group) as Group;
|
|
|
|
if (!resolvedEntityGroup || !(resolvedEntityGroup instanceof Group)) {
|
|
error = `Group '${message.group}' does not exist`;
|
|
}
|
|
}
|
|
|
|
const parsed = this.zigbee.resolveEntityAndEndpoint(message.device);
|
|
resolvedEntityDevice = parsed?.entity as Device;
|
|
|
|
if (!error && (!resolvedEntityDevice || !(resolvedEntityDevice instanceof Device))) {
|
|
error = `Device '${message.device}' does not exist`;
|
|
}
|
|
|
|
if (!error) {
|
|
resolvedEntityEndpoint = parsed.endpoint;
|
|
|
|
if (parsed.endpointID && !resolvedEntityEndpoint) {
|
|
error = `Device '${parsed.ID}' does not have endpoint '${parsed.endpointID}'`;
|
|
}
|
|
}
|
|
}
|
|
|
|
return {
|
|
resolvedEntityGroup,
|
|
resolvedEntityDevice,
|
|
type,
|
|
error,
|
|
groupKey,
|
|
deviceKey,
|
|
triggeredViaLegacyApi,
|
|
skipDisableReporting,
|
|
resolvedEntityEndpoint,
|
|
};
|
|
}
|
|
|
|
@bind private async onMQTTMessage(data: eventdata.MQTTMessage): Promise<void> {
|
|
const parsed = await this.parseMQTTMessage(data);
|
|
|
|
if (!parsed || !parsed.type) {
|
|
return;
|
|
}
|
|
|
|
const {
|
|
resolvedEntityGroup,
|
|
resolvedEntityDevice,
|
|
type,
|
|
triggeredViaLegacyApi,
|
|
groupKey,
|
|
deviceKey,
|
|
skipDisableReporting,
|
|
resolvedEntityEndpoint,
|
|
} = parsed;
|
|
let error = parsed.error;
|
|
let changedGroups: Group[] = [];
|
|
|
|
if (!error) {
|
|
try {
|
|
const keys = [
|
|
`${resolvedEntityDevice.ieeeAddr}/${resolvedEntityEndpoint.ID}`,
|
|
`${resolvedEntityDevice.name}/${resolvedEntityEndpoint.ID}`,
|
|
];
|
|
const endpointNameLocal = resolvedEntityDevice.endpointName(resolvedEntityEndpoint);
|
|
|
|
if (endpointNameLocal) {
|
|
keys.push(`${resolvedEntityDevice.ieeeAddr}/${endpointNameLocal}`);
|
|
keys.push(`${resolvedEntityDevice.name}/${endpointNameLocal}`);
|
|
}
|
|
|
|
if (!endpointNameLocal) {
|
|
keys.push(resolvedEntityDevice.name);
|
|
keys.push(resolvedEntityDevice.ieeeAddr);
|
|
}
|
|
|
|
if (type === 'add') {
|
|
logger.info(`Adding '${resolvedEntityDevice.name}' to '${resolvedEntityGroup.name}'`);
|
|
await resolvedEntityEndpoint.addToGroup(resolvedEntityGroup.zh);
|
|
settings.addDeviceToGroup(resolvedEntityGroup.ID.toString(), keys);
|
|
changedGroups.push(resolvedEntityGroup);
|
|
|
|
/* istanbul ignore else */
|
|
if (settings.get().advanced.legacy_api) {
|
|
const message = {friendly_name: resolvedEntityDevice.name, group: resolvedEntityGroup.name};
|
|
|
|
await this.mqtt.publish('bridge/log', stringify({type: `device_group_add`, message}));
|
|
}
|
|
} else if (type === 'remove') {
|
|
logger.info(`Removing '${resolvedEntityDevice.name}' from '${resolvedEntityGroup.name}'`);
|
|
await resolvedEntityEndpoint.removeFromGroup(resolvedEntityGroup.zh);
|
|
settings.removeDeviceFromGroup(resolvedEntityGroup.ID.toString(), keys);
|
|
changedGroups.push(resolvedEntityGroup);
|
|
|
|
/* istanbul ignore else */
|
|
if (settings.get().advanced.legacy_api) {
|
|
const message = {friendly_name: resolvedEntityDevice.name, group: resolvedEntityGroup.name};
|
|
|
|
await this.mqtt.publish('bridge/log', stringify({type: `device_group_remove`, message}));
|
|
}
|
|
} else {
|
|
// remove_all
|
|
logger.info(`Removing '${resolvedEntityDevice.name}' from all groups`);
|
|
changedGroups = this.zigbee.groups().filter((g) => g.zh.members.includes(resolvedEntityEndpoint));
|
|
await resolvedEntityEndpoint.removeFromAllGroups();
|
|
|
|
for (const settingsGroup of settings.getGroups()) {
|
|
settings.removeDeviceFromGroup(settingsGroup.ID.toString(), keys);
|
|
|
|
/* istanbul ignore else */
|
|
if (settings.get().advanced.legacy_api) {
|
|
const message = {friendly_name: resolvedEntityDevice.name};
|
|
|
|
await this.mqtt.publish('bridge/log', stringify({type: `device_group_remove_all`, message}));
|
|
}
|
|
}
|
|
}
|
|
} catch (e) {
|
|
error = `Failed to ${type} from group (${e.message})`;
|
|
logger.debug(e.stack);
|
|
}
|
|
}
|
|
|
|
if (!triggeredViaLegacyApi) {
|
|
const message = utils.parseJSON(data.message, data.message);
|
|
const responseData: KeyValue = {device: deviceKey};
|
|
|
|
if (groupKey) {
|
|
responseData.group = groupKey;
|
|
}
|
|
|
|
await this.mqtt.publish(`bridge/response/group/members/${type}`, stringify(utils.getResponse(message, responseData, error)));
|
|
}
|
|
|
|
if (error) {
|
|
logger.error(error);
|
|
} else {
|
|
for (const group of changedGroups) {
|
|
this.eventBus.emitGroupMembersChanged({group, action: type, endpoint: resolvedEntityEndpoint, skipDisableReporting});
|
|
}
|
|
}
|
|
}
|
|
}
|