homebridge-logo-platform
Version:
This is a Siemens LOGO! Platform Plugin.
469 lines (332 loc) • 17.1 kB
text/typescript
import { AccessoryPlugin, API, Service, CharacteristicValue } from 'homebridge';
import { QueueSendItem, QueueReceiveItem } from "../queue";
import { ErrorNumber } from "../error";
import { LoggerType, InfluxDBLogItem, InfluxDBFild } from "../logger";
import { md5 } from "../md5";
import { Accessory, SubAccessory } from '../logo';
import { SwitchPlatformAccessory } from './switchPlatformAccessory';
import { LightbulbPlatformAccessory } from './lightbulbPlatformAccessory';
import { BlindPlatformAccessory } from './blindPlatformAccessory';
import { WindowPlatformAccessory } from './windowPlatformAccessory';
import { GaragedoorPlatformAccessory } from './garagedoorPlatformAccessory';
import { FanPlatformAccessory } from './fanPlatformAccessory';
import { FilterMaintenancePlatformAccessory } from './filterMaintenancePlatformAccessory';
import { OutletPlatformAccessory } from './outletPlatformAccessory';
import { LightSensorPlatformAccessory } from '../sensors/lightSensorPlatformAccessory';
import { MotionSensorPlatformAccessory } from '../sensors/motionSensorPlatformAccessory';
import { ContactSensorPlatformAccessory } from '../sensors/contactSensorPlatformAccessory';
import { SmokeSensorPlatformAccessory } from '../sensors/smokeSensorPlatformAccessory';
import { TemperatureSensorPlatformAccessory } from '../sensors/temperatureSensorPlatformAccessory';
import { HumiditySensorPlatformAccessory } from '../sensors/humiditySensorPlatformAccessory';
import { CarbonDioxideSensorPlatformAccessory } from '../sensors/carbonDioxideSensorPlatformAccessory';
import { AirQualitySensorPlatformAccessory } from '../sensors/airQualitySensorPlatformAccessory';
import { LeakSensorPlatformAccessory } from '../sensors/leakSensorPlatformAccessory';
import { WatchdogPlatformAccessory } from '../sensors/watchdogPlatformAccessory';
export class ThermostatPlatformAccessory implements AccessoryPlugin {
private model: string = "Thermostat";
private api: API;
private service: Service;
private information: Service;
private fakegatoService: any;
private subs: any[];
public services: Service[];
private platform: any;
private device: any;
private pushButton: number;
private logging: number;
private updateCurrentHeatingCoolingStateQueued: boolean;
private updateTargetHeatingCoolingStateQueued: boolean;
private updateCurrentTemperatureQueued: boolean;
private updateTargetTemperatureQueued: boolean;
private accStates = {
CurrentHeatingCoolingState: 0,
TargetHeatingCoolingState: 0,
CurrentTemperature: 0,
TargetTemperature: 10,
TemperatureDisplayUnits: 0,
MinTemperature: -270,
MaxTemperature: 100,
};
name: string;
isParentAccessory: boolean;
constructor( api: API, platform: any, device: any, parent?: any ) {
this.name = device.name;
this.api = api;
this.platform = platform;
this.device = device;
this.pushButton = this.device.pushButton || this.platform.pushButton;
this.logging = this.device.logging || 0;
this.fakegatoService = [];
this.subs = [];
this.services = [];
this.isParentAccessory = false;
this.errorCheck();
this.service = new this.api.hap.Service.Thermostat(this.device.name);
if (parent) {
this.service.subtype = 'sub-' + this.model + "-" + this.name.replace(" ", "-");
}
this.service.getCharacteristic(this.platform.Characteristic.CurrentHeatingCoolingState)
.onGet(this.getCurrentHeatingCoolingState.bind(this));
this.service.getCharacteristic(this.platform.Characteristic.TargetHeatingCoolingState)
.onSet(this.setTargetHeatingCoolingState.bind(this))
.onGet(this.getTargetHeatingCoolingState.bind(this));
this.service.getCharacteristic(this.platform.Characteristic.CurrentTemperature)
.onGet(this.getCurrentTemperature.bind(this));
this.service.getCharacteristic(this.platform.Characteristic.TargetTemperature)
.onSet(this.setTargetTemperature.bind(this))
.onGet(this.getTargetTemperature.bind(this));
this.service.getCharacteristic(this.platform.Characteristic.TemperatureDisplayUnits)
.onGet(this.getTemperatureDisplayUnits.bind(this));
this.information = new this.api.hap.Service.AccessoryInformation()
.setCharacteristic(this.api.hap.Characteristic.Manufacturer, this.platform.manufacturer)
.setCharacteristic(this.api.hap.Characteristic.Model, this.model + ' @ ' + this.platform.model)
.setCharacteristic(this.api.hap.Characteristic.SerialNumber, md5(this.device.name + this.model))
.setCharacteristic(this.api.hap.Characteristic.FirmwareRevision, this.platform.firmwareRevision);
const configDevices = this.platform.config.devices;
for (const dev of configDevices) {
if (dev.parentAccessory == this.name) {
this.isParentAccessory = true;
switch (dev.type) {
case Accessory.Switch:
this.subs.push( new SwitchPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Lightbulb:
this.subs.push( new LightbulbPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Blind:
this.subs.push( new BlindPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Window:
this.subs.push( new WindowPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Garagedoor:
this.subs.push( new GaragedoorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Thermostat:
this.subs.push( new ThermostatPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Fan:
this.subs.push( new FanPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.FilterMaintenance:
this.subs.push( new FilterMaintenancePlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Outlet:
this.subs.push( new OutletPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.LightSensor:
this.subs.push( new LightSensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.MotionSensor:
this.subs.push( new MotionSensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.ContactSensor:
this.subs.push( new ContactSensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.SmokeSensor:
this.subs.push( new SmokeSensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.TemperatureSensor:
this.subs.push( new TemperatureSensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.HumiditySensor:
this.subs.push( new HumiditySensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.CarbonDioxideSensor:
this.subs.push( new CarbonDioxideSensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.AirQualitySensor:
this.subs.push( new AirQualitySensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.LeakSensor:
this.subs.push( new LeakSensorPlatformAccessory(api, platform, dev, this) );
break;
case Accessory.Watchdog:
this.subs.push( new WatchdogPlatformAccessory(api, platform, dev, this) );
break;
}
}
}
if (this.isParentAccessory == true) {
this.service.subtype = 'main-' + this.model + "-" + this.name.replace(" ", "-");
}
this.services.push(this.service, this.information);
if (parent) {
parent.service.addLinkedService(this.service);
parent.services.push(this.service);
}
this.updateCurrentHeatingCoolingStateQueued = false;
this.updateTargetHeatingCoolingStateQueued = false;
this.updateCurrentTemperatureQueued = false;
this.updateTargetTemperatureQueued = false;
if (this.platform.config.updateInterval) {
setInterval(() => {
this.updateCurrentHeatingCoolingState();
this.updateTargetHeatingCoolingState();
this.updateCurrentTemperature();
this.updateTargetTemperature();
}, this.platform.config.updateInterval);
}
if (this.logging) {
if (this.platform.loggerType == LoggerType.Fakegato) {
this.fakegatoService = new this.platform.FakeGatoHistoryService("custom", this, {storage: 'fs'});
this.services.push(this.fakegatoService);
}
setInterval(() => {
this.logAccessory();
}, this.platform.loggerInterval);
}
}
errorCheck() {
if (!this.device.thermostatGetHCState || !this.device.thermostatGetTargetHCState || !this.device.thermostatSetTargetHCState ||
!this.device.thermostatGetTemp || !this.device.thermostatGetTargetTemp || !this.device.thermostatSetTargetTemp) {
this.platform.log.error('[%s] One or more LOGO! Addresses are not correct!', this.device.name);
}
}
getServices(): Service[] {
return this.services;
}
async setTargetHeatingCoolingState(value: CharacteristicValue) {
this.accStates.TargetHeatingCoolingState = value as number;
if (this.platform.config.debugMsgLog || this.device.debugMsgLog) {
this.platform.log.info('[%s] Set TargetHeatingCoolingState <- %i', this.device.name, value);
}
let qItem: QueueSendItem = new QueueSendItem(this.device.thermostatSetTargetHCState, this.accStates.TargetHeatingCoolingState, 0);
this.platform.queue.bequeue(qItem);
}
async setTargetTemperature(value: CharacteristicValue) {
this.accStates.TargetTemperature = value as number;
if (this.platform.config.debugMsgLog || this.device.debugMsgLog) {
this.platform.log.info('[%s] Set TargetTemperature <- %i', this.device.name, value);
}
let newValue: number;
if (this.device.thermostatConvertValue) {
newValue = (this.accStates.TargetTemperature * 10)
} else {
newValue = this.accStates.TargetTemperature
}
let qItem: QueueSendItem = new QueueSendItem(this.device.thermostatSetTargetTemp, newValue, 0);
this.platform.queue.bequeue(qItem);
}
async getCurrentHeatingCoolingState(): Promise<CharacteristicValue> {
const isCurrentHeatingCoolingState = this.accStates.CurrentHeatingCoolingState;
this.updateCurrentHeatingCoolingState();
return isCurrentHeatingCoolingState;
}
async getTargetHeatingCoolingState(): Promise<CharacteristicValue> {
const isTargetHeatingCoolingState = this.accStates.TargetHeatingCoolingState;
this.updateTargetHeatingCoolingState();
return isTargetHeatingCoolingState;
}
async getCurrentTemperature(): Promise<CharacteristicValue> {
const isCurrentTemperature = this.accStates.CurrentTemperature;
this.updateCurrentTemperature();
return isCurrentTemperature;
}
async getTargetTemperature(): Promise<CharacteristicValue> {
const isTargetTemperature = this.accStates.TargetTemperature;
this.updateTargetTemperature();
return isTargetTemperature;
}
async getTemperatureDisplayUnits(): Promise<CharacteristicValue> {
const isTemperatureDisplayUnits = this.accStates.TemperatureDisplayUnits;
return isTemperatureDisplayUnits;
}
updateCurrentHeatingCoolingState() {
if (this.updateCurrentHeatingCoolingStateQueued) {return;}
let qItem: QueueReceiveItem = new QueueReceiveItem(this.device.thermostatGetHCState, async (value: number) => {
if (value != ErrorNumber.noData) {
this.accStates.CurrentHeatingCoolingState = value as number;
if (this.platform.config.debugMsgLog || this.device.debugMsgLog) {
this.platform.log.info('[%s] Get CurrentHeatingCoolingState -> %i', this.device.name, this.accStates.CurrentHeatingCoolingState);
}
this.service.updateCharacteristic(this.api.hap.Characteristic.CurrentHeatingCoolingState, this.accStates.CurrentHeatingCoolingState);
}
this.updateCurrentHeatingCoolingStateQueued = false;
});
if (this.platform.queue.enqueue(qItem) === 1) {
this.updateCurrentHeatingCoolingStateQueued = true;
};
}
updateTargetHeatingCoolingState() {
if (this.updateTargetHeatingCoolingStateQueued) {return;}
let qItem: QueueReceiveItem = new QueueReceiveItem(this.device.thermostatGetTargetHCState, async (value: number) => {
if (value != ErrorNumber.noData) {
this.accStates.TargetHeatingCoolingState = value as number;
if (this.platform.config.debugMsgLog || this.device.debugMsgLog) {
this.platform.log.info('[%s] Get TargetHeatingCoolingState -> %i', this.device.name, this.accStates.TargetHeatingCoolingState);
}
this.service.updateCharacteristic(this.api.hap.Characteristic.TargetHeatingCoolingState, this.accStates.TargetHeatingCoolingState);
}
this.updateTargetHeatingCoolingStateQueued = false;
});
if (this.platform.queue.enqueue(qItem) === 1) {
this.updateTargetHeatingCoolingStateQueued = true;
};
}
updateCurrentTemperature() {
if (this.updateCurrentTemperatureQueued) {return;}
let qItem: QueueReceiveItem = new QueueReceiveItem(this.device.thermostatGetTemp, async (value: number) => {
if (value != ErrorNumber.noData) {
if (this.device.thermostatConvertValue) {
this.accStates.CurrentTemperature = (value as number / 10);
} else {
this.accStates.CurrentTemperature = value as number;
}
if (this.accStates.CurrentTemperature < this.accStates.MinTemperature) {
this.accStates.CurrentTemperature = this.accStates.MinTemperature;
}
if (this.accStates.CurrentTemperature > this.accStates.MaxTemperature) {
this.accStates.CurrentTemperature = this.accStates.MaxTemperature;
}
if (this.platform.config.debugMsgLog || this.device.debugMsgLog) {
this.platform.log.info('[%s] Get CurrentTemperature -> %i', this.device.name, this.accStates.CurrentTemperature);
}
this.service.updateCharacteristic(this.api.hap.Characteristic.CurrentTemperature, this.accStates.CurrentTemperature);
}
this.updateCurrentTemperatureQueued = false;
});
if (this.platform.queue.enqueue(qItem) === 1) {
this.updateCurrentTemperatureQueued = true;
};
}
updateTargetTemperature() {
if (this.updateTargetTemperatureQueued) {return;}
let qItem: QueueReceiveItem = new QueueReceiveItem(this.device.thermostatGetTargetTemp, async (value: number) => {
if (value != ErrorNumber.noData) {
if (this.device.thermostatConvertValue) {
this.accStates.TargetTemperature = (value as number / 10);
} else {
this.accStates.TargetTemperature = value as number;
}
if (this.accStates.TargetTemperature < 10) {
this.accStates.TargetTemperature = 10;
}
if (this.accStates.TargetTemperature > 38) {
this.accStates.TargetTemperature = 38;
}
if (this.platform.config.debugMsgLog || this.device.debugMsgLog) {
this.platform.log.info('[%s] Get TargetTemperature -> %i', this.device.name, this.accStates.TargetTemperature);
}
this.service.updateCharacteristic(this.api.hap.Characteristic.TargetTemperature, this.accStates.TargetTemperature);
}
this.updateTargetTemperatureQueued = false;
});
if (this.platform.queue.enqueue(qItem) === 1) {
this.updateTargetTemperatureQueued = true;
};
}
logAccessory() {
if ((this.platform.loggerType == LoggerType.InfluxDB) && this.platform.influxDB.isConfigured) {
let logItems: InfluxDBLogItem[] = [];
logItems.push(new InfluxDBLogItem("CurrentHeatingCoolingState", this.accStates.CurrentHeatingCoolingState, InfluxDBFild.Int));
logItems.push(new InfluxDBLogItem("TargetHeatingCoolingState", this.accStates.TargetHeatingCoolingState, InfluxDBFild.Int));
logItems.push(new InfluxDBLogItem("CurrentTemperature", this.accStates.CurrentTemperature, InfluxDBFild.Int));
logItems.push(new InfluxDBLogItem("TargetTemperature", this.accStates.TargetTemperature, InfluxDBFild.Int));
this.platform.influxDB.logMultipleValues(this.device.name, logItems);
}
if (this.platform.loggerType == LoggerType.Fakegato) {
this.fakegatoService.addEntry({time: Math.round(new Date().valueOf() / 1000), temp: this.accStates.CurrentTemperature, setTemp: this.accStates.TargetTemperature});
}
}
}