UNPKG

homebridge-melcloud-control

Version:

Homebridge plugin to control Mitsubishi Air Conditioner, Heat Pump and Energy Recovery Ventilation.

817 lines (778 loc) • 79 kB
import EventEmitter from 'events'; import MelCloudErv from './melclouderv.js'; import RestFul from './restful.js'; import Mqtt from './mqtt.js'; import { TemperatureDisplayUnits, Ventilation } from './constants.js'; let Accessory, Characteristic, Service, Categories, AccessoryUUID; class DeviceErv extends EventEmitter { constructor(api, account, device, contextKey, accountName, deviceId, deviceName, deviceTypeText, devicesFile, refreshInterval, useFahrenheit, restFul, mqtt) { super(); Accessory = api.platformAccessory; Characteristic = api.hap.Characteristic; Service = api.hap.Service; Categories = api.hap.Categories; AccessoryUUID = api.hap.uuid; //account config this.displayMode = device.displayMode; this.temperatureSensor = device.temperatureSensor || false; this.temperatureSensorOutdoor = device.temperatureSensorOutdoor || false; this.temperatureSensorSupply = device.temperatureSensorSupply || false; this.presets = device.presets || []; this.buttons = device.buttonsSensors || []; this.disableLogInfo = account.disableLogInfo || false; this.disableLogDeviceInfo = account.disableLogDeviceInfo || false; this.enableDebugMode = account.enableDebugMode || false; this.contextKey = contextKey; this.accountName = accountName; this.deviceId = deviceId; this.deviceName = deviceName; this.deviceTypeText = deviceTypeText; this.devicesFile = devicesFile; this.refreshInterval = refreshInterval; this.startPrepareAccessory = true; this.displayDeviceInfo = true; //external integrations this.restFul = restFul; this.restFulConnected = false; this.mqtt = mqtt; this.mqttConnected = false; //presets configured this.presetsConfigured = []; for (const preset of this.presets) { const displayType = preset.displayType ?? 0; if (displayType === 0) { continue; }; const presetyServiceType = ['', Service.Outlet, Service.Switch, Service.MotionSensor, Service.OccupancySensor, Service.ContactSensor][displayType]; const presetCharacteristicType = ['', Characteristic.On, Characteristic.On, Characteristic.MotionDetected, Characteristic.OccupancyDetected, Characteristic.ContactSensorState][displayType]; preset.name = preset.name || 'Preset' preset.serviceType = presetyServiceType; preset.characteristicType = presetCharacteristicType; preset.state = false; preset.previousSettings = {}; this.presetsConfigured.push(preset); } this.presetsConfiguredCount = this.presetsConfigured.length || 0; //buttons configured this.buttonsConfigured = []; for (const button of this.buttons) { const displayType = button.displayType ?? 0; if (displayType === 0) { continue; }; const buttonServiceType = ['', Service.Outlet, Service.Switch, Service.MotionSensor, Service.OccupancySensor, Service.ContactSensor][displayType]; const buttonCharacteristicType = ['', Characteristic.On, Characteristic.On, Characteristic.MotionDetected, Characteristic.OccupancyDetected, Characteristic.ContactSensorState][displayType]; button.name = button.name || 'Button' button.serviceType = buttonServiceType; button.characteristicType = buttonCharacteristicType; button.state = false; button.previousValue = null; this.buttonsConfigured.push(button);; } this.buttonsConfiguredCount = this.buttonsConfigured.length || 0; //device data this.deviceData = {}; //accessory this.accessory = {}; this.useFahrenheit = useFahrenheit ? 1 : 0; }; async externalIntegrations() { try { //RESTFul server const restFulEnabled = this.restFul.enable || false; if (restFulEnabled) { if (!this.restFulConnected) { this.restFul1 = new RestFul({ port: this.deviceId.toString().slice(-4).replace(/^0/, '9'), debug: this.restFul.debug || false }); this.restFul1.on('connected', (message) => { this.restFulConnected = true; this.emit('success', message); }) .on('set', async (key, value) => { try { await this.setOverExternalIntegration('RESTFul', this.deviceData, key, value); } catch (error) { this.emit('warn', error); }; }) .on('debug', (debug) => { this.emit('debug', debug); }) .on('warn', (warn) => { this.emit('warn', warn); }) .on('error', (error) => { this.emit('error', error); }); } } //MQTT client const mqttEnabled = this.mqtt.enable || false; if (mqttEnabled) { if (!this.mqttConnected) { this.mqtt1 = new Mqtt({ host: this.mqtt.host, port: this.mqtt.port || 1883, clientId: this.mqtt.clientId ? `melcloud_${this.mqtt.clientId}_${Math.random().toString(16).slice(3)}` : `melcloud_${Math.random().toString(16).slice(3)}`, prefix: this.mqtt.prefix ? `melcloud/${this.mqtt.prefix}/${this.deviceTypeText}/${this.deviceName}` : `melcloud/${this.deviceTypeText}/${this.deviceName}`, user: this.mqtt.user, passwd: this.mqtt.passwd, debug: this.mqtt.debug || false }); this.mqtt1.on('connected', (message) => { this.mqttConnected = true; this.emit('success', message); }) .on('subscribed', (message) => { this.emit('success', message); }) .on('set', async (key, value) => { try { await this.setOverExternalIntegration('MQTT', this.deviceData, key, value); } catch (error) { this.emit('warn', error); }; }) .on('debug', (debug) => { this.emit('debug', debug); }) .on('warn', (warn) => { this.emit('warn', warn); }) .on('error', (error) => { this.emit('error', error); }); } } } catch (error) { this.emit('warn', `External integration start error: ${error}`); }; } async setOverExternalIntegration(integration, deviceData, key, value) { try { let set = false switch (key) { case 'Power': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'OperationMode': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.OperationMode; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'VentilationMode': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.VentilationMode; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'SetTemperature': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetTemperature; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'DefaultCoolingSetTemperature': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetTemperature; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'DefaultHeatingSetTemperature': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetTemperature; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'NightPurgeMode': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.NightPurgeMode; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'SetFanSpeed': deviceData.Device[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetFanSpeed; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'HideRoomTemperature': deviceData[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Prohibit; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'HideSupplyTemperature': deviceData[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Prohibit; set = await this.melCloudErv.send(deviceData, this.displayMode); break; case 'HideOutdoorTemperature': deviceData[key] = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Prohibit; set = await this.melCloudErv.send(deviceData, this.displayMode); break; default: this.emit('warn', `${integration}, received key: ${key}, value: ${value}`); break; }; return set; } catch (error) { throw new Error(`${integration} set key: ${key}, value: ${value}, error: ${error.message ?? error}`); }; } async startImpulseGenerator() { try { //start impule generator await new Promise(resolve => setTimeout(resolve, 1000)); await this.melCloudErv.impulseGenerator.start([{ name: 'checkState', sampling: this.refreshInterval }]); return true; } catch (error) { throw new Error(`Impulse generator start error: ${error}`); }; } //prepare accessory async prepareAccessory(deviceData) { try { const deviceId = this.deviceId; const deviceTypeText = this.deviceTypeText; const deviceName = this.deviceName; const accountName = this.accountName; const presetsOnServer = this.accessory.presets; const hasRoomTemperature = this.accessory.hasRoomTemperature; const hasSupplyTemperature = this.accessory.hasSupplyTemperature; const hasOutdoorTemperature = this.accessory.hasOutdoorTemperature; const hasCoolOperationMode = this.accessory.hasCoolOperationMode; const hasHeatOperationMode = this.accessory.hasHeatOperationMode; const hasAutoVentilationMode = this.accessory.hasAutoVentilationMode; const hasBypassVentilationMode = this.accessory.hasBypassVentilationMode; const hasAutomaticFanSpeed = this.accessory.hasAutomaticFanSpeed; const hasCO2Sensor = this.accessory.hasCO2Sensor; const hasPM25Sensor = this.accessory.hasPM25Sensor; const numberOfFanSpeeds = this.accessory.numberOfFanSpeeds; //accessory const debug = this.enableDebugMode ? this.emit('debug', `Prepare accessory`) : false; const accessoryName = deviceName; const accessoryUUID = AccessoryUUID.generate(accountName + deviceId.toString()); const accessoryCategory = [Categories.OTHER, Categories.AIR_PURIFIER, Categories.THERMOSTAT][this.displayType]; const accessory = new Accessory(accessoryName, accessoryUUID, accessoryCategory); //information service const debug1 = this.enableDebugMode ? this.emit('debug', `Prepare information service`) : false; accessory.getService(Service.AccessoryInformation) .setCharacteristic(Characteristic.Manufacturer, this.manufacturer) .setCharacteristic(Characteristic.Model, this.model) .setCharacteristic(Characteristic.SerialNumber, this.serialNumber) .setCharacteristic(Characteristic.FirmwareRevision, this.firmwareRevision) .setCharacteristic(Characteristic.ConfiguredName, accessoryName); //services const serviceName = `${deviceTypeText} ${accessoryName}`; switch (this.displayMode) { case 1: //Heater Cooler const debug = this.enableDebugMode ? this.emit('debug', `Prepare heather/cooler service`) : false; this.melCloudService = new Service.HeaterCooler(serviceName, `HeaterCooler ${deviceId}`); this.melCloudService.setPrimaryService(true); this.melCloudService.getCharacteristic(Characteristic.Active) .onGet(async () => { const state = this.accessory.power; return state; }) .onSet(async (state) => { try { deviceData.Device.Power = [false, true][state]; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power; await this.melCloudErv.send(deviceData, this.displayMode); const info = this.disableLogInfo ? false : this.emit('info', `Set power: ${state ? 'ON' : 'OFF'}`); } catch (error) { this.emit('warn', `Set power error: ${error}`); }; }); this.melCloudService.getCharacteristic(Characteristic.CurrentHeaterCoolerState) .onGet(async () => { const value = this.accessory.currentOperationMode; return value; }); this.melCloudService.getCharacteristic(Characteristic.TargetHeaterCoolerState) .setProps({ minValue: this.accessory.operationModeSetPropsMinValue, maxValue: this.accessory.operationModeSetPropsMaxValue, validValues: this.accessory.operationModeSetPropsValidValues }) .onGet(async () => { const value = this.accessory.targetOperationMode ?? 0; //LOSSNAY, BYPASS, AUTO return value; }) .onSet(async (value) => { try { switch (value) { case 0: //AUTO - AUTO deviceData.Device.VentilationMode = hasAutoVentilationMode ? 2 : 0; break; case 1: //HEAT - LOSSNAY deviceData.Device.VentilationMode = 0; break; case 2: //COOL - BYPASS deviceData.Device.VentilationMode = hasBypassVentilationMode ? 1 : 0; break; }; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.VentilationMode; await this.melCloudErv.send(deviceData, this.displayMode); const operationModeText = Ventilation.VentilationMode[deviceData.Device.VentilationMode]; const info = this.disableLogInfo ? false : this.emit('info', `Set operation mode: ${operationModeText}`); } catch (error) { this.emit('warn', `Set operation mode error: ${error}`); }; }); this.melCloudService.getCharacteristic(Characteristic.CurrentTemperature) .onGet(async () => { const value = this.accessory.roomTemperature; return value; }); this.melCloudService.getCharacteristic(Characteristic.RotationSpeed) .setProps({ minValue: 0, maxValue: this.accessory.fanSpeedSetPropsMaxValue, minStep: 1 }) .onGet(async () => { const value = this.accessory.fanSpeed; //STOP, 1, 2, 3, 4, OFF return value; }) .onSet(async (value) => { try { let fanSpeedModeText = ''; switch (numberOfFanSpeeds) { case 2: //Fan speed mode 2 fanSpeedModeText = hasAutomaticFanSpeed ? [5, 1, 2, 0][value] : [5, 1, 2][value]; deviceData.Device.SetFanSpeed = hasAutomaticFanSpeed ? [0, 1, 2, 0][value] : [1, 1, 2][value]; break; case 3: //Fan speed mode 3 fanSpeedModeText = hasAutomaticFanSpeed ? [5, 1, 2, 3, 0][value] : [5, 1, 2, 3][value]; deviceData.Device.SetFanSpeed = hasAutomaticFanSpeed ? [0, 1, 2, 3, 0][value] : [1, 1, 2, 3][value]; break; case 4: //Fan speed mode 4 fanSpeedModeText = hasAutomaticFanSpeed ? [5, 1, 2, 3, 4, 0][value] : [5, 1, 2, 3, 4][value]; deviceData.Device.SetFanSpeed = hasAutomaticFanSpeed ? [0, 1, 2, 3, 4, 0][value] : [1, 1, 2, 3, 4][value]; break; case 5: //Fan speed mode 5 5; fanSpeedModeText = hasAutomaticFanSpeed ? [5, 1, 2, 3, 4, 5, 0][value] : [5, 1, 2, 3, 4, 5][value]; deviceData.Device.SetFanSpeed = hasAutomaticFanSpeed ? [0, 1, 2, 3, 4, 5, 0][value] : [1, 1, 2, 3, 4, 5][value]; break;; }; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetFanSpeed; await this.melCloudErv.send(deviceData, this.displayMode); const info = this.disableLogInfo ? false : this.emit('info', `Set fan speed mode: ${Ventilation.FanSpeed[fanSpeedModeText]}`); } catch (error) { this.emit('warn', `Set fan speed mode error: ${error}`); }; }); //device can cool if (hasAutoVentilationMode && hasCoolOperationMode) { this.melCloudService.getCharacteristic(Characteristic.CoolingThresholdTemperature) .setProps({ minValue: this.accessory.minTempCoolDry, maxValue: this.accessory.maxTempCoolDry, minStep: this.accessory.temperatureIncrement }) .onGet(async () => { const value = this.accessory.ventilationMode === 2 ? this.accessory.defaultHeatingSetTemperature : this.accessory.setTemperature; return value; }) .onSet(async (value) => { try { deviceData.Device.DefaultCoolingSetTemperature = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetTemperature; await this.melCloudErv.send(deviceData, this.displayMode); const info = this.disableLogInfo ? false : this.emit('info', `Set cooling threshold temperature: ${value}${this.accessory.temperatureUnit}`); } catch (error) { this.emit('warn', `Set cooling threshold temperature error: ${error}`); }; }); }; //device can heat if (hasAutoVentilationMode && hasHeatOperationMode) { this.melCloudService.getCharacteristic(Characteristic.HeatingThresholdTemperature) .setProps({ minValue: this.accessory.minTempHeat, maxValue: this.accessory.maxTempHeat, minStep: this.accessory.temperatureIncrement }) .onGet(async () => { const value = this.accessory.ventilationMode === 2 ? this.accessory.defaultHeatingSetTemperature : this.accessory.setTemperature; return value; }) .onSet(async (value) => { try { deviceData.Device.DefaultHeatingSetTemperature = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetTemperature; await this.melCloudErv.send(deviceData, this.displayMode); const info = this.disableLogInfo ? false : this.emit('info', `Set heating threshold temperature: ${value}${this.accessory.temperatureUnit}`); } catch (error) { this.emit('warn', `Set heating threshold temperature error: ${error}`); }; }); }; //this.melCloudService.getCharacteristic(Characteristic.LockPhysicalControls) // .onGet(async () => { // const value = this.accessory.lockPhysicalControl; // const info = this.disableLogInfo ? false : this.emit('info', `Lock physical controls: ${value ? 'LOCKED' : 'UNLOCKED'}`); // return value; // }) // .onSet(async (value) => { // try { // value = value ? true : false; // deviceData.Device = deviceData.Device; // deviceData.Device.EffectiveFlags = CONSTANTS.Ventilation.EffectiveFlags.Prohibit; // await this.melCloudErv.send(deviceData, this.displayMode); // const info = this.disableLogInfo ? false : this.emit('info', `Set local physical controls: ${value ? 'LOCK' : 'UNLOCK'}`); // } catch (error) { // this.emit('warn', `Set lock physical controls error: ${error}`); // }; // }); this.melCloudService.getCharacteristic(Characteristic.TemperatureDisplayUnits) .onGet(async () => { const value = this.accessory.useFahrenheit; return value; }) .onSet(async (value) => { try { value = [false, true][value]; this.accessory.useFahrenheit = value; this.emit('melCloud', 'UseFahrenheit', value); const info = this.disableLogInfo ? false : this.emit('info', `Set temperature display unit: ${TemperatureDisplayUnits[value]}`); } catch (error) { this.emit('warn', `Set temperature display unit error: ${error}`); }; }); accessory.addService(this.melCloudService); break; case 2: //Thermostat const debug1 = this.enableDebugMode ? this.emit('debug', `Prepare thermostat service`) : false; this.melCloudService = new Service.Thermostat(serviceName, `Thermostat ${deviceId}`); this.melCloudService.setPrimaryService(true); this.melCloudService.getCharacteristic(Characteristic.CurrentHeatingCoolingState) .onGet(async () => { const value = this.accessory.currentOperationMode; return value; }); this.melCloudService.getCharacteristic(Characteristic.TargetHeatingCoolingState) .setProps({ minValue: this.accessory.operationModeSetPropsMinValue, maxValue: this.accessory.operationModeSetPropsMaxValue, validValues: this.accessory.operationModeSetPropsValidValues }) .onGet(async () => { const value = this.accessory.targetOperationMode ?? 0; //LOSSNAY, BYPASS, AUTO return value; }) .onSet(async (value) => { try { switch (value) { case 0: //OFF - POWER OFF deviceData.Device.Power = false; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power; break; case 1: //HEAT - LOSSNAY deviceData.Device.Power = true; deviceData.Device.VentilationMode = 0; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.VentilationMode; break; case 2: //COOL - BYPASS deviceData.Device.Power = true; deviceData.Device.VentilationMode = hasBypassVentilationMode ? 1 : 0; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.VentilationMode; break; case 3: //AUTO - AUTO deviceData.Device.Power = true; deviceData.Device.VentilationMode = hasAutoVentilationMode ? 2 : 0; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.VentilationMode; break; }; await this.melCloudErv.send(deviceData, this.displayMode); const operationModeText = Ventilation.VentilationMode[deviceData.Device.VentilationMode]; const info = this.disableLogInfo ? false : this.emit('info', `Set operation mode: ${operationModeText}`); } catch (error) { this.emit('warn', `Set operation mode error: ${error}`); }; }); this.melCloudService.getCharacteristic(Characteristic.CurrentTemperature) .onGet(async () => { const value = this.accessory.roomTemperature; return value; }); this.melCloudService.getCharacteristic(Characteristic.TargetTemperature) .setProps({ minValue: this.accessory.minTempHeat, maxValue: this.accessory.maxTempHeat, minStep: this.accessory.temperatureIncrement }) .onGet(async () => { const value = this.accessory.setTemperature; return value; }) .onSet(async (value) => { try { deviceData.Device.SetTemperature = value; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.SetTemperature; await this.melCloudErv.send(deviceData, this.displayMode); const info = this.disableLogInfo ? false : this.emit('info', `Set temperature: ${value}${this.accessory.temperatureUnit}`); } catch (error) { this.emit('warn', `Set temperature error: ${error}`); }; }); this.melCloudService.getCharacteristic(Characteristic.TemperatureDisplayUnits) .onGet(async () => { const value = this.accessory.useFahrenheit; return value; }) .onSet(async (value) => { try { value = [false, true][value]; this.accessory.useFahrenheit = value; this.emit('melCloud', 'UseFahrenheit', value); const info = this.disableLogInfo ? false : this.emit('info', `Set temperature display unit: ${TemperatureDisplayUnits[value]}`); } catch (error) { this.emit('warn', `Set temperature display unit error: ${error}`); }; }); accessory.addService(this.melCloudService); break; }; //temperature sensor service room if (this.temperatureSensor && hasRoomTemperature && this.accessory.roomTemperature !== null) { const debug = this.enableDebugMode ? this.emit('debug', `Prepare room temperature sensor service`) : false; this.roomTemperatureSensorService = new Service.TemperatureSensor(`${serviceName} Room`, `Room Temperature Sensor ${deviceId}`); this.roomTemperatureSensorService.addOptionalCharacteristic(Characteristic.ConfiguredName); this.roomTemperatureSensorService.setCharacteristic(Characteristic.ConfiguredName, `${serviceName} Room`); this.roomTemperatureSensorService.getCharacteristic(Characteristic.CurrentTemperature) .setProps({ minValue: -35, maxValue: 150, minStep: 0.5 }) .onGet(async () => { const state = this.accessory.roomTemperature; return state; }) accessory.addService(this.roomTemperatureSensorService); }; //temperature sensor service supply if (this.temperatureSensorSupply && hasSupplyTemperature && this.accessory.supplyTemperature !== null) { const debug = this.enableDebugMode ? this.emit('debug', `Prepare supply temperature sensor service`) : false; this.supplyTemperatureSensorService = new Service.TemperatureSensor(`${serviceName} Supply`, `Supply Temperature Sensor ${deviceId}`); this.supplyTemperatureSensorService.addOptionalCharacteristic(Characteristic.ConfiguredName); this.supplyTemperatureSensorService.setCharacteristic(Characteristic.ConfiguredName, `${serviceName} Supply`); this.supplyTemperatureSensorService.getCharacteristic(Characteristic.CurrentTemperature) .setProps({ minValue: -35, maxValue: 150, minStep: 0.5 }) .onGet(async () => { const state = this.accessory.supplyTemperature; return state; }) accessory.addService(this.supplyTemperatureSensorService); }; //temperature sensor service outdoor if (this.temperatureSensorOutdoor && hasOutdoorTemperature && this.accessory.outdoorTemperature !== null) { const debug = this.enableDebugMode ? this.emit('debug', `Prepare outdoor temperature sensor service`) : false; this.outdoorTemperatureSensorService = new Service.TemperatureSensor(`${serviceName} Outdoor`, `Outdoor Temperature Sensor ${deviceId}`); this.outdoorTemperatureSensorService.addOptionalCharacteristic(Characteristic.ConfiguredName); this.outdoorTemperatureSensorService.setCharacteristic(Characteristic.ConfiguredName, `${serviceName} Outdoor`); this.outdoorTemperatureSensorService.getCharacteristic(Characteristic.CurrentTemperature) .setProps({ minValue: -35, maxValue: 150, minStep: 0.5 }) .onGet(async () => { const state = this.accessory.outdoorTemperature; return state; }) accessory.addService(this.outdoorTemperatureSensorService); }; //core maintenance this.coreMaintenanceService = new Service.FilterMaintenance(`${serviceName} Core Maintenance`, `CoreMaintenance ${deviceId}`); this.coreMaintenanceService.addOptionalCharacteristic(Characteristic.ConfiguredName); this.coreMaintenanceService.setCharacteristic(Characteristic.ConfiguredName, `${serviceName} Core Maintenance`); this.coreMaintenanceService.getCharacteristic(Characteristic.FilterChangeIndication) .onGet(async () => { const value = this.accessory.coreMaintenanceRequired; return value; }); this.coreMaintenanceService.getCharacteristic(Characteristic.ResetFilterIndication) .onSet(async (state) => { }); accessory.addService(this.coreMaintenanceService); //filter maintenance this.filterMaintenanceService = new Service.FilterMaintenance(`${serviceName} Filter Maintenance`, `FilterMaintenance ${deviceId}`); this.filterMaintenanceService.addOptionalCharacteristic(Characteristic.ConfiguredName); this.filterMaintenanceService.setCharacteristic(Characteristic.ConfiguredName, `${serviceName} Filter Maintenance`); this.filterMaintenanceService.getCharacteristic(Characteristic.FilterChangeIndication) .onGet(async () => { const value = this.accessory.filterMaintenanceRequired; return value; }); this.filterMaintenanceService.getCharacteristic(Characteristic.ResetFilterIndication) .onSet(async (state) => { }); accessory.addService(this.filterMaintenanceService); //room CO2 sensor if (hasCO2Sensor) { this.carbonDioxideSensorService = new Service.CarbonDioxideSensor(`${serviceName} CO2 Sensor`, `CO2Sensor ${deviceId}`); this.carbonDioxideSensorService.addOptionalCharacteristic(Characteristic.ConfiguredName); this.carbonDioxideSensorService.setCharacteristic(Characteristic.ConfiguredName, `${serviceName} CO2 Sensor`); this.carbonDioxideSensorService.getCharacteristic(Characteristic.CarbonDioxideDetected) .onGet(async () => { const value = this.accessory.roomCO2Detected; return value; }); this.carbonDioxideSensorService.getCharacteristic(Characteristic.CarbonDioxideLevel) .onGet(async () => { const value = this.accessory.roomCO2Level; return value; }); accessory.addService(this.carbonDioxideSensorService); } //room PM2.5 sensor if (hasPM25Sensor) { this.airQualitySensorService = new Service.AirQualitySensor(`${serviceName} PM2.5 Sensor`, `PM25Sensor ${deviceId}`); this.airQualitySensorService.addOptionalCharacteristic(Characteristic.ConfiguredName); this.airQualitySensorService.setCharacteristic(Characteristic.ConfiguredName, `${serviceName} PM2.5 Sensor`); this.airQualitySensorService.getCharacteristic(Characteristic.AirQuality) .onGet(async () => { const value = this.accessory.pM25AirQuality; return value; }); this.airQualitySensorService.getCharacteristic(Characteristic.PM2_5Density) .onGet(async () => { const value = this.accessory.pM25Level; return value; }); accessory.addService(this.airQualitySensorService); } //presets services if (this.presetsConfiguredCount > 0) { const debug = this.enableDebugMode ? this.emit('debug', `Prepare presets services`) : false; this.presetsServices = []; this.presetsConfigured.forEach((preset, i) => { const presetData = presetsOnServer.find(p => p.ID === preset.Id); //get preset name const name = preset.name; //get preset name prefix const namePrefix = preset.namePrefix; const serviceName = namePrefix ? `${accessoryName} ${name}` : name; const serviceType = preset.serviceType; const characteristicType = preset.characteristicType; const presetService = new serviceType(serviceName, `Preset ${deviceId} ${i}`); presetService.addOptionalCharacteristic(Characteristic.ConfiguredName); presetService.setCharacteristic(Characteristic.ConfiguredName, serviceName); presetService.getCharacteristic(characteristicType) .onGet(async () => { const state = preset.state; return state; }) .onSet(async (state) => { try { switch (state) { case true: preset.previousSettings = deviceData.Device; deviceData.Device.SetTemperature = presetData.SetTemperature; deviceData.Device.Power = presetData.Power; deviceData.Device.OperationMode = presetData.OperationMode; deviceData.Device.VentilationMode = presetData.VentilationMode; deviceData.Device.SetFanSpeed = presetData.FanSpeed; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power; break; case false: deviceData.Device.SetTemperature = preset.previousSettings.SetTemperature; deviceData.Device.Power = preset.previousSettings.Power; deviceData.Device.OperationMode = preset.previousSettings.OperationMode; deviceData.Device.VentilationMode = preset.previousSettings.VentilationMode; deviceData.Device.SetFanSpeed = preset.previousSettings.FanSpeed; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power; break; }; await this.melCloudErv.send(deviceData, this.displayMode); const info = this.disableLogInfo ? false : this.emit('info', `${state ? 'Set:' : 'Unset:'} ${name}`); } catch (error) { this.emit('warn', `Set preset error: ${error}`); }; }); }); this.presetsServices.push(presetService); accessory.addService(presetService); }; //buttons services if (this.buttonsConfiguredCount > 0) { const debug = this.enableDebugMode ? this.emit('debug', `Prepare buttons services`) : false; this.buttonsServices = []; this.buttonsConfigured.forEach((button, i) => { //get button mode const mode = button.mode; //get button name const buttonName = button.name; //get button name prefix const namePrefix = button.namePrefix; const serviceName = namePrefix ? `${accessoryName} ${buttonName}` : buttonName; const serviceType = button.serviceType; const characteristicType = button.characteristicType; const buttonService = new serviceType(serviceName, `Button ${deviceId} ${i}`); buttonService.addOptionalCharacteristic(Characteristic.ConfiguredName); buttonService.setCharacteristic(Characteristic.ConfiguredName, serviceName); buttonService.getCharacteristic(characteristicType) .onGet(async () => { const state = button.state; return state; }) .onSet(async (state) => { try { switch (mode) { case 0: //POWER ON,OFF deviceData.Device.Power = state; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power; break; case 1: //OPERATING MODE RECOVERY button.previousValue = state ? deviceData.Device.VentilationMode : button.previousValue ?? deviceData.Device.VentilationMode; deviceData.Device.Power = true; deviceData.Device.VentilationMode = state ? 0 : button.previousValue; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.VentilationMode; break; case 2: //OPERATING MODE BYPASS button.previousValue = state ? deviceData.Device.VentilationMode : button.previousValue ?? deviceData.Device.VentilationMode; deviceData.Device.Power = true; deviceData.Device.VentilationMode = state ? 1 : button.previousValue; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.VentilationMode; break case 3: //OPERATING MODE AUTO button.previousValue = state ? deviceData.Device.VentilationMode : button.previousValue ?? deviceData.Device.VentilationMode; deviceData.Device.Power = true; deviceData.Device.VentilationMode = state ? 2 : button.previousValue; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.VentilationMode; break; case 4: //NIGHT PURGE MODE deviceData.Device.Power = true; deviceData.Device.NightPurgeMode = state; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power break; case 10: //FAN SPEED MODE AUTO button.previousValue = state ? deviceData.Device.SetFanSpeed : button.previousValue ?? deviceData.Device.SetFanSpeed; deviceData.Device.Power = true; deviceData.Device.SetFanSpeed = state ? 0 : button.previousValue; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.SetFanSpeed; break; case 11: //FAN SPEED MODE 1 button.previousValue = state ? deviceData.Device.SetFanSpeed : button.previousValue ?? deviceData.Device.SetFanSpeed; deviceData.Device.Power = true; deviceData.Device.SetFanSpeed = state ? 1 : button.previousValue; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.SetFanSpeed; break; case 12: //FAN SPEED MODE 2 button.previousValue = state ? deviceData.Device.SetFanSpeed : button.previousValue ?? deviceData.Device.SetFanSpeed; deviceData.Device.Power = true; deviceData.Device.SetFanSpeed = state ? 2 : button.previousValue; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.SetFanSpeed; break; case 13: //FAN SPEED MODE 3 button.previousValue = state ? deviceData.Device.SetFanSpeed : button.previousValue ?? deviceData.Device.SetFanSpeed; deviceData.Device.Power = true; deviceData.Device.SetFanSpeed = state ? 3 : button.previousValue; deviceData.Device.EffectiveFlags = Ventilation.EffectiveFlags.Power + Ventilation.EffectiveFlags.SetFanSpeed; break;