UNPKG

homebridge-meraki-control

Version:
201 lines (181 loc) • 9.18 kB
import MerakiMs from './merakims.js'; import EventEmitter from 'events'; let Accessory, Characteristic, Service, Categories, AccessoryUUID; class MerakiDevice extends EventEmitter { constructor(api, config, deviceName, deviceUuid, deviceData) { super(); Accessory = api.platformAccessory; Characteristic = api.hap.Characteristic; Service = api.hap.Service; Categories = api.hap.Categories; AccessoryUUID = api.hap.uuid; //meraki configuration this.networkName = config.name; this.organizationId = config.organizationId; this.networkId = config.networkId; //system configuration this.host = config.host; this.apiKey = config.apiKey; this.deviceName = deviceName; this.deviceUuid = deviceUuid; this.deviceData = deviceData; this.refreshInterval = config.refreshInterval * 1000 || 5000; this.enableDebugMode = config.enableDebugMode || false; this.disableLogInfo = config.disableLogInfo || false; this.disableLogDeviceInfo = config.disableLogDeviceInfo || false; //variables this.startPrepareAccessory = true; this.prefixForPortName = deviceData.enablePrefixForPortName || false; this.portsSensor = deviceData.enableSensorPorts || false; this.poePortsControl = deviceData.enablePoePortsControl || false; }; async startImpulseGenerator() { try { //start impulse generator await this.merakiMs.impulseGenerator.start([{ name: 'checkDeviceInfo', sampling: this.refreshInterval }]); return true; } catch (error) { throw new Error(`Impulse generator start error: ${error}`); }; } //prepare accessory async prepareAccessory() { try { //prepare accessory const debug = !this.enableDebugMode ? false : this.emit('debug', `prepare accessory`); const accessoryName = this.deviceName; const accessoryUUID = AccessoryUUID.generate(this.deviceUuid); const accessoryCategory = Categories.AIRPORT; const accessory = new Accessory(accessoryName, accessoryUUID, accessoryCategory); //prepare information service const debug1 = !this.enableDebugMode ? false : this.emit('debug', `prepare information service`); accessory.getService(Service.AccessoryInformation) .setCharacteristic(Characteristic.Manufacturer, 'Cisco Meraki') .setCharacteristic(Characteristic.Model, accessoryName) .setCharacteristic(Characteristic.SerialNumber, this.networkId) .setCharacteristic(Characteristic.FirmwareRevision, this.organizationId) .setCharacteristic(Characteristic.ConfiguredName, accessoryName); const debug2 = !this.enableDebugMode ? false : this.emit('debug', `prepare meraki service`); const exposedPorts = this.exposedPorts; //device this.services = []; this.sensorServices = []; for (const port of exposedPorts) { const portName = port.name; const portId = port.id; const serviceName = this.prefixForPortName ? `${portId}.${portName}` : portName; const service = accessory.addService(Service.Outlet, serviceName, `Port Service ${portName}`); service.addOptionalCharacteristic(Characteristic.ConfiguredName); service.setCharacteristic(Characteristic.ConfiguredName, serviceName); service.getCharacteristic(Characteristic.On) .onGet(async () => { const state = port.state ?? false; const logInfo = this.disableLogInfo ? false : this.emit('message', `Port: ${portId}, Name: ${portName}, State: ${state ? 'Enabled' : 'Disabled'}`); return state; }) .onSet(async (state) => { try { state = state ? true : false; const switchPortUrl = `/devices/${this.deviceUuid}/switch/ports/${portId}`; const switchPortData = this.poePortControl ? { 'enabled': state, 'poeEnabled': state } : { 'enabled': state }; await this.merakiMs.send(switchPortUrl, switchPortData); const logInfo = this.disableLogInfo ? false : this.emit('message', `Port: ${portId}, Name: ${portName}, set State: ${state ? 'Enabled' : 'Disabled'}`); } catch (error) { this.emit('warn', `Port: ${portId}, Name: ${portName}, set state error: %${error}`); } }); this.services.push(service); if (this.portsSensor) { const debug = !this.enableDebugMode && k > 0 ? false : this.emit('debug', `prepare meraki sensor service`); const sensorServiceName = this.prefixForPortName ? `Sensor ${portId}.${portName}` : `Sensor ${portName}`; const sensorService = accessory.addService(Service.ContactSensor, sensorServiceName, `Port Service Sensor ${portName}`); sensorService.addOptionalCharacteristic(Characteristic.ConfiguredName); sensorService.setCharacteristic(Characteristic.ConfiguredName, sensorServiceName); sensorService.getCharacteristic(Characteristic.ContactSensorState) .onGet(async () => { const state = port.state; return state; }); this.sensorServices.push(sensorService); }; }; return accessory; } catch (error) { throw new Error(error); }; }; //start async start() { try { this.merakiMs = new MerakiMs({ host: this.host, apiKey: this.apiKey, deviceData: this.deviceData, enableDebugMode: this.enableDebugMode }); this.merakiMs.on('deviceInfo', (portsCount) => { //meraki info if (this.startPrepareAccessory) { //connect to deice success this.emit('success', `Connect Success.`) if (!this.disableLogDeviceInfo) { this.emit('devInfo', `---- ${this.deviceName}: ${this.deviceUuid} ----`); this.emit('devInfo', `Manufacturer: Cisco/Meraki`); this.emit('devInfo', `Network: ${this.networkName}`); this.emit('devInfo', `Network Id: ${this.networkId}`); this.emit('devInfo', `Organization Id: ${this.organizationId}`); this.emit('devInfo', `Exposed Ports: ${portsCount}`); this.emit('devInfo', `----------------------------------`) }; }; }).on('deviceState', async (exposedPorts, portsCount) => { this.exposedPorts = exposedPorts; //update characteristics of exposed ports for (let i = 0; i < portsCount; i++) { const state = exposedPorts[i].state; if (this.services) { this.services[i].updateCharacteristic(Characteristic.On, state); }; if (this.sensorServices && this.portsSensor) { this.sensorServices[i].updateCharacteristic(Characteristic.ContactSensorState, state ? 0 : 1) }; }; }) .on('success', (success) => { this.emit('success', success); }) .on('info', (info) => { this.emit('message', info); }) .on('debug', (debug) => { this.emit('debug', debug); }) .on('warn', (warn) => { this.emit('warn', warn); }) .on('error', (error) => { this.emit('error', error); }) //connect const connect = await this.merakiMs.connect(); if (!connect) { return false; }; //prepare accessory if (this.startPrepareAccessory) { const accessory = await this.prepareAccessory(); this.emit('publishAccessory', accessory); this.startPrepareAccessory = false; } return true; } catch (error) { throw new Error(`Start error: ${error.message || error}}.`); }; }; }; export default MerakiDevice;