mirror of
https://github.com/Koenkk/zigbee2mqtt.git
synced 2024-11-17 02:48:31 -07:00
1b32a46649
* Initial * Updates * Some tests * tests * Updates * Move poll to bind * Move report to legacy * Updates * Remove toPoll key. https://github.com/Koenkk/zigbee2mqtt/pull/5529#pullrequestreview-570285310 * Updates * Updates * Update bridge.test.js * Fix tests
194 lines
7.7 KiB
JavaScript
194 lines
7.7 KiB
JavaScript
const zigbeeHerdsmanConverters = require('zigbee-herdsman-converters');
|
|
const logger = require('../../util/logger');
|
|
const settings = require('../../util/settings');
|
|
const ZNLDP12LM = zigbeeHerdsmanConverters.devices.find((d) => d.model === 'ZNLDP12LM');
|
|
const utils = require('../../util/utils');
|
|
const Extension = require('../extension');
|
|
|
|
const defaultConfiguration = {
|
|
minimumReportInterval: 3, maximumReportInterval: 300, reportableChange: 1,
|
|
};
|
|
|
|
const devicesNotSupportingReporting = [
|
|
zigbeeHerdsmanConverters.devices.find((d) => d.model === 'CC2530.ROUTER'),
|
|
zigbeeHerdsmanConverters.devices.find((d) => d.model === 'BASICZBR3'),
|
|
zigbeeHerdsmanConverters.devices.find((d) => d.model === 'ZM-CSW032-D'),
|
|
zigbeeHerdsmanConverters.devices.find((d) => d.model === 'TS0001'),
|
|
zigbeeHerdsmanConverters.devices.find((d) => d.model === 'TS0115'),
|
|
];
|
|
|
|
const reportKey = 1;
|
|
|
|
const getColorCapabilities = async (endpoint) => {
|
|
if (endpoint.getClusterAttributeValue('lightingColorCtrl', 'colorCapabilities') === undefined) {
|
|
await endpoint.read('lightingColorCtrl', ['colorCapabilities']);
|
|
}
|
|
|
|
const value = endpoint.getClusterAttributeValue('lightingColorCtrl', 'colorCapabilities');
|
|
return {
|
|
colorTemperature: (value & 1<<4) > 0,
|
|
colorXY: (value & 1<<3) > 0,
|
|
};
|
|
};
|
|
|
|
const clusters = {
|
|
'genOnOff': [
|
|
{attribute: 'onOff', ...defaultConfiguration, minimumReportInterval: 0, reportableChange: 0},
|
|
],
|
|
'genLevelCtrl': [
|
|
{attribute: 'currentLevel', ...defaultConfiguration},
|
|
],
|
|
'lightingColorCtrl': [
|
|
{
|
|
attribute: 'colorTemperature', ...defaultConfiguration,
|
|
condition: async (endpoint) => (await getColorCapabilities(endpoint)).colorTemperature,
|
|
},
|
|
{
|
|
attribute: 'currentX', ...defaultConfiguration,
|
|
condition: async (endpoint) => (await getColorCapabilities(endpoint)).colorXY,
|
|
},
|
|
{
|
|
attribute: 'currentY', ...defaultConfiguration,
|
|
condition: async (endpoint) => (await getColorCapabilities(endpoint)).colorXY,
|
|
},
|
|
],
|
|
'closuresWindowCovering': [
|
|
{attribute: 'currentPositionLiftPercentage', ...defaultConfiguration},
|
|
{attribute: 'currentPositionTiltPercentage', ...defaultConfiguration},
|
|
],
|
|
};
|
|
|
|
class Report extends Extension {
|
|
constructor(zigbee, mqtt, state, publishEntityState, eventBus) {
|
|
super(zigbee, mqtt, state, publishEntityState, eventBus);
|
|
this.queue = new Set();
|
|
this.failed = new Set();
|
|
this.enabled = settings.get().advanced.report;
|
|
}
|
|
|
|
shouldIgnoreClusterForDevice(cluster, definition) {
|
|
if (definition === ZNLDP12LM && cluster === 'closuresWindowCovering') {
|
|
// Device announces it but doesn't support it
|
|
// https://github.com/Koenkk/zigbee2mqtt/issues/2611
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
async setupReporting(resolvedEntity) {
|
|
const {device, definition} = resolvedEntity;
|
|
|
|
if (this.queue.has(device.ieeeAddr) || this.failed.has(device.ieeeAddr)) return;
|
|
this.queue.add(device.ieeeAddr);
|
|
|
|
const term1 = this.enabled ? 'Setup' : 'Disable';
|
|
const term2 = this.enabled ? 'setup' : 'disabled';
|
|
|
|
try {
|
|
for (const ep of device.endpoints) {
|
|
for (const [cluster, configuration] of Object.entries(clusters)) {
|
|
if (ep.supportsInputCluster(cluster) && !this.shouldIgnoreClusterForDevice(cluster, definition)) {
|
|
logger.debug(`${term1} reporting for '${device.ieeeAddr}' - ${ep.ID} - ${cluster}`);
|
|
|
|
const items = [];
|
|
for (const entry of configuration) {
|
|
if (!entry.hasOwnProperty('condition') || (await entry.condition(ep))) {
|
|
const toAdd = {...entry};
|
|
if (!this.enabled) toAdd.maximumReportInterval = 0xFFFF;
|
|
items.push(toAdd);
|
|
delete items[items.length - 1].condition;
|
|
}
|
|
}
|
|
|
|
this.enabled ?
|
|
await ep.bind(cluster, this.coordinatorEndpoint) :
|
|
await ep.unbind(cluster, this.coordinatorEndpoint);
|
|
|
|
await ep.configureReporting(cluster, items);
|
|
logger.info(
|
|
`Successfully ${term2} reporting for '${device.ieeeAddr}' - ${ep.ID} - ${cluster}`,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this.enabled) {
|
|
device.meta.reporting = reportKey;
|
|
} else {
|
|
delete device.meta.reporting;
|
|
this.eventBus.emit('reportingDisabled', {device});
|
|
}
|
|
|
|
this.eventBus.emit(`devicesChanged`);
|
|
} catch (error) {
|
|
logger.error(
|
|
`Failed to ${term1.toLowerCase()} reporting for '${device.ieeeAddr}' - ${error.stack}`,
|
|
);
|
|
|
|
this.failed.add(device.ieeeAddr);
|
|
}
|
|
|
|
device.save();
|
|
this.queue.delete(device.ieeeAddr);
|
|
}
|
|
|
|
shouldSetupReporting(resolvedEntity, messageType) {
|
|
if (!resolvedEntity || !resolvedEntity.device || !resolvedEntity.definition ||
|
|
messageType === 'deviceLeave') return false;
|
|
|
|
const {device, definition} = resolvedEntity;
|
|
// Handle messages of type endDeviceAnnce and devIncoming.
|
|
// This message is typically send when a device comes online after being powered off
|
|
// Ikea TRADFRI tend to forget their reporting after powered off.
|
|
// Re-setup reporting.
|
|
// Only resetup reporting if configuredReportings was not populated yet,
|
|
// else reconfigure is done in zigbee-herdsman-converters ikea.js/bulbOnEvent
|
|
// configuredReportings are saved since Zigbee2MQTT 1.17.0
|
|
// https://github.com/Koenkk/zigbee2mqtt/issues/966
|
|
if (this.enabled && messageType === 'deviceAnnounce' && utils.isIkeaTradfriDevice(device) &&
|
|
device.endpoints.filter((e) => e.configuredReportings.length === 0).length === device.endpoints.length) {
|
|
return true;
|
|
}
|
|
|
|
// These do not support reproting.
|
|
// https://github.com/Koenkk/zigbee-herdsman/issues/110
|
|
const philipsIgnoreSw = ['5.127.1.26581', '5.130.1.30000'];
|
|
if (device.manufacturerName === 'Philips' && philipsIgnoreSw.includes(device.softwareBuildID)) return false;
|
|
|
|
if (resolvedEntity.device.interviewing === true) return false;
|
|
if (device.type !== 'Router' || device.powerSource === 'Battery') return false;
|
|
// Gledopto devices don't support reporting.
|
|
if (devicesNotSupportingReporting.includes(definition) || definition.vendor === 'Gledopto') return false;
|
|
|
|
if (this.enabled && device.meta.hasOwnProperty('reporting') && device.meta.reporting === reportKey) {
|
|
return false;
|
|
}
|
|
|
|
if (!this.enabled && !device.meta.hasOwnProperty('reporting')) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
async onZigbeeStarted() {
|
|
this.coordinatorEndpoint = this.zigbee.getDevicesByType('Coordinator')[0].getEndpoint(1);
|
|
|
|
for (const device of this.zigbee.getClients()) {
|
|
const resolvedEntity = this.zigbee.resolveEntity(device);
|
|
if (this.shouldSetupReporting(resolvedEntity, null)) {
|
|
await this.setupReporting(resolvedEntity);
|
|
}
|
|
}
|
|
}
|
|
|
|
async onZigbeeEvent(type, data, resolvedEntity) {
|
|
if (this.shouldSetupReporting(resolvedEntity, type)) {
|
|
await this.setupReporting(resolvedEntity);
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = Report;
|