UNPKG

homebridge-luxor

Version:

Homebridge Plug-in for the FX Luminaire (Luxor) lighting controller

321 lines 45.2 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.IControllerType = exports.BaseController = void 0; const ZD_Light_1 = require("../lights/ZD_Light"); const Queue_1 = __importDefault(require("../Queue")); const axios = require('axios').default; class BaseController { constructor(data, log) { this.log = log; this.callbackList = []; this.GroupList = []; if (typeof data.ip === 'undefined') { this.log.debug(`Initializing base controller.`); return; } this.ip = data.ip; this.name = data.Controller; this.type = data.type; this.platform = data.platform; this.hideGroups = data.hideGroups; this.independentColors = data.independentColors; this.commandTimeout = data.commandTimeout || 750; log.info(`Assigning ${this.type} Controller to IP ${this.ip}`); // this.updateLights(); setTimeout(async () => { await this.pollController(); }, 30000); } getStatus(result) { switch (result) { case 0: return ('Ok'); //StatusOk case (1): return ('Unknown Method'); //StatusUnknownMethod case (101): return ('Unparseable Request'); //StatusUnparseableRequest case (102): return ('Invalid Request'); //StatusInvalidRequest case (151): return ('Color Value Out of Range'); case (201): return ('Precondition Failed'); //StatusPreconditionFailed case (202): return ('Group Name In Use'); //StatusGroupNameInUse case (205): return ('Group Number In Use'); //StatusGroupNumberInUse case (241): return ('Item Does Not Exist'); //StatusThemeIndexOutOfRange case (242): return ('Bad Group Number'); //StatusThemeIndexOutOfRange case (243): return ('Theme Index Out Of Range'); //StatusThemeIndexOutOfRange case (251): return ('Bad Theme Index'); //StatusThemeIndexOutOfRange case (252): return ('Theme Changes Restricted'); //StatusThemeIndexOutOfRange default: return ('Unknown status'); } } async doRequest(url, data) { return new Promise(async (resolve, reject) => { try { switch (url) { case 'GroupListGet': if (this.hideGroups) return; if (typeof this.cacheGroupList !== 'undefined' && Date.now() - this.cacheGroupList < 2000) { resolve({ Status: 1, StatusStr: 'Cached', GroupList: this.GroupList }); return; } break; case 'ThemeListGet': if (typeof this.cacheThemeList !== 'undefined' && Date.now() - this.cacheThemeList < 2000) { resolve({ Status: 1, StatusStr: 'Cached', ThemeList: this.ThemeList }); return; } break; case 'ColorListGet': if (typeof this.cacheColorList !== 'undefined' && Date.now() - this.cacheColorList < 2000) { resolve({ Status: 1, StatusStr: 'Cached', ColorList: this.ColorList }); return; } break; } const response = await axios({ method: 'post', url: 'http://' + this.ip + '/' + url + '.json', data, headers: { 'cache-control': 'no-cache' }, timeout: this.commandTimeout }); response.data.StatusStr = this.getStatus(response.data.Status); if (response.data.StatusStr !== 'Ok' || response.code === 'ETIMEOUT') { this.log.error(`Controller ${this.name} responded with error ${response.data.StatusStr} to '${url}'`); resolve({ Status: 1, StatusStr: 'Ok', GroupList: this.GroupList || [], ThemeList: this.ThemeList || [], ColorList: this.ColorList || [] }); } else resolve(response.data); } catch (err) { this.log.error(`Error communicating with controller ${this.name}. URL ${url}.json.\n\t${err}`); reject(err); } }); } async sleep(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } async IlluminateAllAsync() { return Queue_1.default.enqueue(async () => { let status = await this.doRequest('IlluminateAll'); return status; }); } async ExtinguishAllAsync() { return Queue_1.default.enqueue(async () => { let status = await this.doRequest('ExtinguishAll'); return status; }); } async GetGroupAsync(group) { if (this.hideGroups) return; return new Promise(async (resolve, reject) => { try { await this.GroupListGetAsync(); for (let i in this.GroupList) { if (this.GroupList[i].GroupNumber === group) return resolve(this.GroupList[i]); } reject(`No Group Found in GroupGetAsync for group ${group}.\n${JSON.stringify(this.GroupList)}`); } catch (err) { reject(`Error with GetGroupAsync: ${err}`); this.log.error(err); } }); } async GroupListGetAsync() { // Get the list of light groups from the controller if (typeof this.cacheGroupList !== 'undefined' && Date.now() - this.cacheGroupList < 2000) { return (this.GroupList); } return Queue_1.default.enqueue(async () => { let data = await this.doRequest('GroupListGet'); if (data.Status === 0) { this.cacheGroupList = Date.now(); this.processGroupListGet(data); } return this.GroupList; }); } processGroupListGet(data) { // override with ZDC/ZDTWO this.GroupList = data.GroupList; for (let i in this.GroupList) { this.GroupList[i].type = ZD_Light_1.ILightType.ZD; } } async GroupListEditAsync(name, groupNumber, color) { // Same in ZDC/ZDTWO var requestData = JSON.stringify({ 'Name': name, 'GroupNumber': groupNumber, 'Color': color }); return Queue_1.default.enqueue(async () => { let status = await this.doRequest('GroupListEdit', requestData); return status; }); /* let status = await this.queueRequest('GroupListEdit', requestData); return status; */ } async ThemeListGetAsync() { if (typeof this.cacheThemeList !== 'undefined' && Date.now() - this.cacheThemeList < 2000) { return (this.ThemeList); } return Queue_1.default.enqueue(async () => { let data = await this.doRequest('ThemeListGet'); if (data.Status === 0) { this.cacheThemeList = Date.now(); this.processThemeListGet(data); } return this.ThemeList; }); } processThemeListGet(data) { this.ThemeList = data.ThemeList; for (var i in this.ThemeList) { this.ThemeList[i].isOn = this.ThemeList[i].OnOff === 1; this.ThemeList[i].type = ZD_Light_1.ILightType.THEME; } } async GetThemeAsync(index) { return new Promise(async (resolve, reject) => { try { await this.ThemeListGetAsync(); for (let i in this.ThemeList) { if (this.ThemeList[i].ThemeIndex === index) return resolve(this.ThemeList[i]); } reject(`No Theme Found in ThemeGetAsync for theme ${index}.\n${JSON.stringify(this.ThemeList)}`); } catch (err) { this.log.error(err); reject(`Error with GetThemeAsync ${err}`); } }); } async IlluminateThemeAsync(themeIndex, onOff) { return Queue_1.default.enqueue(async () => { let status = await this.doRequest('IlluminateTheme', { 'ThemeIndex': themeIndex, 'OnOff': onOff }); this.cacheThemeList = undefined; setTimeout(async () => { await this.updateLights(); }, 250); return status; }); } async IlluminateGroupAsync(groupNumber, desiredIntensity) { return Queue_1.default.enqueue(async () => { let status = await this.doRequest('IlluminateGroup', { 'GroupNumber': groupNumber, 'Intensity': desiredIntensity }); return status; }); } async ColorListGetAsync() { return {}; } async ColorListSetAsync(color, hue, saturation) { return {}; } processColorListGet(data) { this.ColorList = data.ColorList; } ; async GetColorAsync(color) { return {}; } registerCallback(UUID, type, index, characteristic, fn) { // look for an existing UUID/characteristic pair and update that if the light attributes get updates for (let i = 0; i < this.callbackList.length; i++) { let callback = this.callbackList[i]; if (callback.UUID == UUID && callback.characteristic === characteristic) { callback = { UUID, type, index, characteristic, fn }; return; } } // not found, add a new callback. this.callbackList.push({ UUID, type, index, characteristic, fn }); } execCallbacks() { for (let i = 0; i < this.callbackList.length; i++) { let callback = this.callbackList[i]; try { switch (callback.type) { case ZD_Light_1.ILightType.ZD: case ZD_Light_1.ILightType.ZDC: if (callback.characteristic.name === this.platform.Characteristic.Brightness.name) { for (let i in this.GroupList) { // need to use loop as these are not 0 based indexes if (this.GroupList[i].GroupNumber === callback.index) { if (typeof this.GroupList[i].Intensity !== 'undefined') callback.fn(this.GroupList[i].Intensity); break; } } // let group = this.GroupList.find(g=>g.GroupNumber === callback.index, this.GroupList); } break; case ZD_Light_1.ILightType.THEME: if (callback.characteristic.name === this.platform.Characteristic.On.name) { for (let i in this.GroupList) { // need to use loop as these are not 0 based indexes if (this.ThemeList[i].ThemeIndex === callback.index) { if (typeof this.ThemeList[i].OnOff !== 'undefined') callback.fn(this.ThemeList[i].OnOff === 1); break; } } } break; } } catch (err) { this.log.error(`execCallbacks: ${err}`); } } } async updateLights(force = false) { if (force) { this.cacheGroupList = undefined; this.cacheThemeList = undefined; } await this.GroupListGetAsync(); await this.ThemeListGetAsync(); this.execCallbacks(); } async pollController() { try { await this.updateLights(); } catch (err) { this.log.error(`${this.name} error: ${err}`); } finally { setTimeout(async () => { await this.pollController(); }, 30 * 1000); } } } exports.BaseController = BaseController; var IControllerType; (function (IControllerType) { IControllerType["ZD"] = "ZD"; IControllerType["ZDC"] = "ZDC"; IControllerType["ZDTWO"] = "ZDTWO"; })(IControllerType = exports.IControllerType || (exports.IControllerType = {})); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BaseController.js","sourceRoot":"","sources":["../../src/controller/BaseController.ts"],"names":[],"mappings":";;;;;;AAEA,iDAAgD;AAEhD,qDAA6B;AAE7B,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC;AAEvC,MAAa,cAAc;IAgBzB,YAAY,IAAS,EAAE,GAAQ;QAC7B,IAAI,CAAC,GAAG,GAAG,GAAG,CAAC;QACf,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,IAAI,CAAC,SAAS,GAAG,EAAE,CAAA;QACnB,IAAI,OAAO,IAAI,CAAC,EAAE,KAAK,WAAW,EAAE;YAClC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,+BAA+B,CAAC,CAAC;YAChD,OAAO;SACR;QACD,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC;QAClB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC;QAC5B,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC;QACtB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;QAC9B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC;QAClC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,CAAC;QAChD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,IAAI,GAAG,CAAC;QAEjD,GAAG,CAAC,IAAI,CAAC,aAAa,IAAI,CAAC,IAAI,qBAAqB,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC;QAC/D,uBAAuB;QACvB,UAAU,CAAC,KAAK,IAAI,EAAE,GAAG,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;IAClE,CAAC;IAES,SAAS,CAAC,MAAM;QACxB,QAAQ,MAAM,EAAE;YACd,KAAK,CAAC;gBACJ,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,UAAU;YAC3B,KAAK,CAAC,CAAC,CAAC;gBACN,OAAO,CAAC,gBAAgB,CAAC,CAAC,CAAC,qBAAqB;YAClD,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,qBAAqB,CAAC,CAAC,CAAC,0BAA0B;YAC5D,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,iBAAiB,CAAC,CAAC,CAAC,sBAAsB;YACpD,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,0BAA0B,CAAC,CAAC;YACtC,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,qBAAqB,CAAC,CAAC,CAAC,0BAA0B;YAC5D,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,mBAAmB,CAAC,CAAC,CAAC,sBAAsB;YACtD,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,qBAAqB,CAAC,CAAC,CAAC,wBAAwB;YAC1D,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,qBAAqB,CAAC,CAAC,CAAC,4BAA4B;YAC9D,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,kBAAkB,CAAC,CAAC,CAAC,4BAA4B;YAC3D,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,0BAA0B,CAAC,CAAC,CAAC,4BAA4B;YACnE,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,iBAAiB,CAAC,CAAC,CAAC,4BAA4B;YAC1D,KAAK,CAAC,GAAG,CAAC;gBACR,OAAO,CAAC,0BAA0B,CAAC,CAAC,CAAC,4BAA4B;YACnE;gBACE,OAAO,CAAC,gBAAgB,CAAC,CAAC;SAC7B;IACH,CAAC;IAED,KAAK,CAAC,SAAS,CAAC,GAAW,EAAE,IAAU;QACrC,OAAO,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAAE,MAAM,EAAuE,EAAE;YAChH,IAAI;gBACF,QAAQ,GAAG,EAAE;oBACX,KAAK,cAAc;wBACjB,IAAI,IAAI,CAAC,UAAU;4BAAE,OAAO;wBAC5B,IAAI,OAAO,IAAI,CAAC,cAAc,KAAK,WAAW,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,cAAc,GAAG,IAAI,EAAE;4BACzF,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,SAAS,EAAE,QAAQ,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;4BACvE,OAAO;yBACR;wBACD,MAAM;oBACR,KAAK,cAAc;wBACjB,IAAI,OAAO,IAAI,CAAC,cAAc,KAAK,WAAW,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,cAAc,GAAG,IAAI,EAAE;4BACzF,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,SAAS,EAAE,QAAQ,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;4BACvE,OAAO;yBACR;wBACD,MAAM;oBACR,KAAK,cAAc;wBACjB,IAAI,OAAO,IAAI,CAAC,cAAc,KAAK,WAAW,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,cAAc,GAAG,IAAI,EAAE;4BACzF,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,SAAS,EAAE,QAAQ,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;4BACvE,OAAO;yBACR;wBACD,MAAM;iBACT;gBACD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC;oBAC3B,MAAM,EAAE,MAAM;oBACd,GAAG,EAAE,SAAS,GAAG,IAAI,CAAC,EAAE,GAAG,GAAG,GAAG,GAAG,GAAG,OAAO;oBAC9C,IAAI;oBACJ,OAAO,EAAE;wBACP,eAAe,EAAE,UAAU;qBAC5B;oBACD,OAAO,EAAE,IAAI,CAAC,cAAc;iBAC7B,CAAC,CAAA;gBACF,QAAQ,CAAC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC/D,IAAI,QAAQ,CAAC,IAAI,CAAC,SAAS,KAAK,IAAI,IAAI,QAAQ,CAAC,IAAI,KAAK,UAAU,EAAE;oBACpE,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,cAAc,IAAI,CAAC,IAAI,yBAAyB,QAAQ,CAAC,IAAI,CAAC,SAAS,QAAQ,GAAG,GAAG,CAAC,CAAC;oBACtG,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,SAAS,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,IAAI,EAAE,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,IAAI,EAAE,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,IAAI,EAAE,EAAE,CAAC,CAAC;iBAC5I;;oBAEC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;aAC1B;YACD,OAAO,GAAG,EAAE;gBACV,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,uCAAuC,IAAI,CAAC,IAAI,SAAS,GAAG,aAAa,GAAG,EAAE,CAAC,CAAC;gBAC/F,MAAM,CAAC,GAAG,CAAC,CAAC;aACb;QACH,CAAC,CAAC,CAAA;IACJ,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,EAAE;QACZ,OAAO,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;IACzD,CAAC;IACD,KAAK,CAAC,kBAAkB;QACtB,OAAO,eAAK,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE;YAC9B,IAAI,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;YACnD,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAA;IACJ,CAAC;IAED,KAAK,CAAC,kBAAkB;QACtB,OAAO,eAAK,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE;YAC9B,IAAI,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;YACnD,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAA;IACJ,CAAC;IACM,KAAK,CAAC,aAAa,CAAC,KAAa;QACtC,IAAI,IAAI,CAAC,UAAU;YAAE,OAAO;QAC5B,OAAO,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAAE,MAAM,EAAE,EAAE;YAC3C,IAAI;gBACF,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBAC/B,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,SAAS,EAAE;oBAC5B,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,WAAW,KAAK,KAAK;wBAAE,OAAO,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;iBAChF;gBACD,MAAM,CAAC,6CAA6C,KAAK,MAAM,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAA;aACjG;YACD,OAAO,GAAG,EAAE;gBACV,MAAM,CAAC,6BAA6B,GAAG,EAAE,CAAC,CAAA;gBAC1C,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;aACrB;QACH,CAAC,CAAC,CAAA;IACJ,CAAC;IACD,KAAK,CAAC,iBAAiB;QACrB,mDAAmD;QACnD,IAAI,OAAO,IAAI,CAAC,cAAc,KAAK,WAAW,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,cAAc,GAAG,IAAI,EAAE;YACzF,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;SACzB;QAED,OAAO,eAAK,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE;YAC9B,IAAI,IAAI,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;YAChD,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE;gBACrB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBACjC,IAAI,CAAC,mBAAmB,CAAC,IAAsB,CAAC,CAAC;aAClD;YACD,OAAO,IAAI,CAAC,SAAS,CAAC;QACxB,CAAC,CAAC,CAAA;IACJ,CAAC;IACS,mBAAmB,CAAC,IAAoB;QAChD,0BAA0B;QAC1B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;QAChC,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,SAAS,EAAE;YAC5B,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,GAAG,qBAAU,CAAC,EAAE,CAAC;SACxC;IACH,CAAC;IACD,KAAK,CAAC,kBAAkB,CAAC,IAAY,EAAE,WAAmB,EAAE,KAAc;QACxE,oBAAoB;QACpB,IAAI,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC;YAC/B,MAAM,EAAE,IAAI;YACZ,aAAa,EAAE,WAAW;YAC1B,OAAO,EAAE,KAAK;SACf,CAAC,CAAC;QACH,OAAO,eAAK,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE;YAC9B,IAAI,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,eAAe,EAAE,WAAW,CAAC,CAAC;YAChE,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAA;QACF;6BACqB;IACvB,CAAC;IACD,KAAK,CAAC,iBAAiB;QACrB,IAAI,OAAO,IAAI,CAAC,cAAc,KAAK,WAAW,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,cAAc,GAAG,IAAI,EAAE;YACzF,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;SACzB;QACD,OAAO,eAAK,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE;YAC9B,IAAI,IAAI,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;YAChD,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE;gBACrB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBACjC,IAAI,CAAC,mBAAmB,CAAC,IAAsB,CAAC,CAAC;aAClD;YACD,OAAO,IAAI,CAAC,SAAS,CAAC;QACxB,CAAC,CAAC,CAAA;IACJ,CAAC;IACS,mBAAmB,CAAC,IAAoB;QAChD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;QAChC,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,SAAS,EAAE;YAC5B,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC;YACvD,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,GAAG,qBAAU,CAAC,KAAK,CAAC;SAC3C;IACH,CAAC;IACM,KAAK,CAAC,aAAa,CAAC,KAAa;QACtC,OAAO,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAAE,MAAM,EAAE,EAAE;YAC3C,IAAI;gBACF,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBAC/B,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,SAAS,EAAE;oBAC5B,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,UAAU,KAAK,KAAK;wBAAE,OAAO,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;iBAC/E;gBACD,MAAM,CAAC,6CAA6C,KAAK,MAAM,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAA;aACjG;YACD,OAAO,GAAG,EAAE;gBACV,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACpB,MAAM,CAAC,4BAA4B,GAAG,EAAE,CAAC,CAAA;aAC1C;QACH,CAAC,CAAC,CAAA;IAEJ,CAAC;IACD,KAAK,CAAC,oBAAoB,CAAC,UAAkB,EAAE,KAAa;QAC1D,OAAO,eAAK,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE;YAC9B,IAAI,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,iBAAiB,EAAE;gBACnD,YAAY,EAAE,UAAU;gBACxB,OAAO,EAAE,KAAK;aACf,CAAC,CAAC;YACH,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;YAChC,UAAU,CAAC,KAAK,IAAI,EAAE,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAA,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;YAC3D,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAA;IACJ,CAAC;IACD,KAAK,CAAC,oBAAoB,CAAC,WAAmB,EAAE,gBAAwB;QACtE,OAAO,eAAK,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE;YAC9B,IAAI,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,iBAAiB,EAAE;gBACnD,aAAa,EAAE,WAAW;gBAC1B,WAAW,EAAE,gBAAgB;aAC9B,CAAC,CAAC;YACH,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAA;IACJ,CAAC;IACD,KAAK,CAAC,iBAAiB;QACrB,OAAO,EAAkB,CAAC;IAC5B,CAAC;IACD,KAAK,CAAC,iBAAiB,CAAC,KAAa,EAAE,GAAW,EAAE,UAAkB;QACpE,OAAO,EAAa,CAAC;IACvB,CAAC;IACD,mBAAmB,CAAC,IAAoB;QACtC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;IAClC,CAAC;IAAA,CAAC;IACF,KAAK,CAAC,aAAa,CAAC,KAAa,IAAyB,OAAO,EAAgB,CAAC,CAAC,CAAC;IAEpF,gBAAgB,CAAC,IAAY,EAAE,IAAgB,EAAE,KAAa,EAAE,cAAmB,EAAE,EAAY;QAC/F,oGAAoG;QACpG,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YACjD,IAAI,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;YACpC,IAAI,QAAQ,CAAC,IAAI,IAAI,IAAI,IAAI,QAAQ,CAAC,cAAc,KAAK,cAAc,EAAE;gBACvE,QAAQ,GAAG,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC;gBACrD,OAAO;aACR;SACF;QACD,iCAAiC;QACjC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE,EAAE,CAAC,CAAC;IACpE,CAAC;IACD,aAAa;QACX,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YACjD,IAAI,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;YACpC,IAAI;gBACF,QAAQ,QAAQ,CAAC,IAAI,EAAE;oBACrB,KAAK,qBAAU,CAAC,EAAE,CAAC;oBACnB,KAAK,qBAAU,CAAC,GAAG;wBACjB,IAAI,QAAQ,CAAC,cAAc,CAAC,IAAI,KAAK,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,UAAU,CAAC,IAAI,EAAE;4BACjF,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,SAAS,EAAC,EAAE,oDAAoD;gCACjF,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,WAAW,KAAK,QAAQ,CAAC,KAAK,EAAC;oCACnD,IAAI,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,KAAK,WAAW;wCAAE,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;oCACjG,MAAM;iCACP;6BACF;4BACD,wFAAwF;yBAEzF;wBACD,MAAM;oBACR,KAAK,qBAAU,CAAC,KAAK;wBACnB,IAAI,QAAQ,CAAC,cAAc,CAAC,IAAI,KAAK,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,EAAE,CAAC,IAAI,EAAE;4BACzE,KAAK,IAAI,CAAC,IAAI,IAAI,CAAC,SAAS,EAAC,EAAE,oDAAoD;gCACjF,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,UAAU,KAAK,QAAQ,CAAC,KAAK,EAAC;oCAClD,IAAI,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,KAAK,KAAK,WAAW;wCAAE,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC,CAAC;oCAC/F,MAAM;iCACP;6BACF;yBACF;wBACD,MAAM;iBACT;aACF;YACD,OAAO,GAAG,EAAE;gBACV,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,kBAAkB,GAAG,EAAE,CAAC,CAAA;aACxC;SACF;IACH,CAAC;IACD,KAAK,CAAC,YAAY,CAAC,QAAiB,KAAK;QACvC,IAAI,KAAK,EAAE;YACT,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;YAChC,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;SACjC;QACD,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IACD,KAAK,CAAC,cAAc;QAClB,IAAI;YACF,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;SAC3B;QACD,OAAO,GAAG,EAAE;YAAE,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,IAAI,WAAW,GAAG,EAAE,CAAC,CAAA;SAAE;gBACpD;YAAE,UAAU,CAAC,KAAK,IAAI,EAAE,GAAG,MAAM,IAAI,CAAC,cAAc,EAAE,CAAA,CAAC,CAAC,EAAE,EAAE,GAAG,IAAI,CAAC,CAAC;SAAE;IACjF,CAAC;CAEF;AA7TD,wCA6TC;AA+CD,IAAY,eAEX;AAFD,WAAY,eAAe;IACzB,4BAAS,CAAA;IAAE,8BAAW,CAAA;IAAE,kCAAe,CAAA;AACzC,CAAC,EAFW,eAAe,GAAf,uBAAe,KAAf,uBAAe,QAE1B","sourcesContent":["import { rejects } from \"assert\";\nimport { Characteristic, Logger } from \"homebridge\";\nimport { ILightType } from \"../lights/ZD_Light\";\nimport { LuxorPlatform } from \"../LuxorPlatform\";\nimport Queue from \"../Queue\";\n\nconst axios = require('axios').default;\n\nexport class BaseController {\n  protected ip: string;\n  public name: string;\n  protected hideGroups: boolean;\n  protected independentColors: boolean;\n  protected log: Logger;\n  public type: IControllerType;\n  protected platform: LuxorPlatform;\n  protected GroupList: IGroupList[];\n  protected cacheGroupList: number;\n  protected ColorList: IColorList[];\n  protected cacheColorList: number;\n  protected ThemeList: IThemeList[];\n  protected cacheThemeList: number;\n  protected commandTimeout: number;\n  protected callbackList: { UUID: string, type: ILightType, index: number, characteristic: any, fn: (a: number | boolean) => {} }[];\n  constructor(data: any, log: any) {\n    this.log = log;\n    this.callbackList = [];\n    this.GroupList = []\n    if (typeof data.ip === 'undefined') {\n      this.log.debug(`Initializing base controller.`);\n      return;\n    }\n    this.ip = data.ip;\n    this.name = data.Controller;\n    this.type = data.type;\n    this.platform = data.platform;\n    this.hideGroups = data.hideGroups;\n    this.independentColors = data.independentColors;\n    this.commandTimeout = data.commandTimeout || 750;\n\n    log.info(`Assigning ${this.type} Controller to IP ${this.ip}`);\n    // this.updateLights();\n    setTimeout(async () => { await this.pollController(); }, 30000);\n  }\n\n  protected getStatus(result): string {\n    switch (result) {\n      case 0:\n        return ('Ok'); //StatusOk\n      case (1):\n        return ('Unknown Method'); //StatusUnknownMethod\n      case (101):\n        return ('Unparseable Request'); //StatusUnparseableRequest\n      case (102):\n        return ('Invalid Request'); //StatusInvalidRequest\n      case (151):\n        return ('Color Value Out of Range');\n      case (201):\n        return ('Precondition Failed'); //StatusPreconditionFailed\n      case (202):\n        return ('Group Name In Use'); //StatusGroupNameInUse\n      case (205):\n        return ('Group Number In Use'); //StatusGroupNumberInUse\n      case (241):\n        return ('Item Does Not Exist'); //StatusThemeIndexOutOfRange\n      case (242):\n        return ('Bad Group Number'); //StatusThemeIndexOutOfRange\n      case (243):\n        return ('Theme Index Out Of Range'); //StatusThemeIndexOutOfRange\n      case (251):\n        return ('Bad Theme Index'); //StatusThemeIndexOutOfRange\n      case (252):\n        return ('Theme Changes Restricted'); //StatusThemeIndexOutOfRange\n      default:\n        return ('Unknown status');\n    }\n  }\n\n  async doRequest(url: string, data?: any): Promise<IThemeListResp | IGroupListResp | IColorListResp | IStatus> {\n    return new Promise(async (resolve, reject): Promise<IThemeListResp | IGroupListResp | IColorListResp | IStatus> => {\n      try {\n        switch (url) {\n          case 'GroupListGet':\n            if (this.hideGroups) return;\n            if (typeof this.cacheGroupList !== 'undefined' && Date.now() - this.cacheGroupList < 2000) {\n              resolve({ Status: 1, StatusStr: 'Cached', GroupList: this.GroupList });\n              return;\n            }\n            break;\n          case 'ThemeListGet':\n            if (typeof this.cacheThemeList !== 'undefined' && Date.now() - this.cacheThemeList < 2000) {\n              resolve({ Status: 1, StatusStr: 'Cached', ThemeList: this.ThemeList });\n              return;\n            }\n            break;\n          case 'ColorListGet':\n            if (typeof this.cacheColorList !== 'undefined' && Date.now() - this.cacheColorList < 2000) {\n              resolve({ Status: 1, StatusStr: 'Cached', ColorList: this.ColorList });\n              return;\n            }\n            break;\n        }\n        const response = await axios({\n          method: 'post',\n          url: 'http://' + this.ip + '/' + url + '.json',\n          data,\n          headers: {\n            'cache-control': 'no-cache'\n          },\n          timeout: this.commandTimeout\n        })\n        response.data.StatusStr = this.getStatus(response.data.Status);\n        if (response.data.StatusStr !== 'Ok' || response.code === 'ETIMEOUT') {\n          this.log.error(`Controller ${this.name} responded with error ${response.data.StatusStr} to '${url}'`);\n          resolve({ Status: 1, StatusStr: 'Ok', GroupList: this.GroupList || [], ThemeList: this.ThemeList || [], ColorList: this.ColorList || [] });\n        }\n        else\n          resolve(response.data);\n      }\n      catch (err) {\n        this.log.error(`Error communicating with controller ${this.name}. URL ${url}.json.\\n\\t${err}`);\n        reject(err);\n      }\n    })\n  }\n\n  async sleep(ms) {\n    return new Promise(resolve => setTimeout(resolve, ms));\n  }\n  async IlluminateAllAsync(): Promise<IStatus> {\n    return Queue.enqueue(async () => {\n      let status = await this.doRequest('IlluminateAll');\n      return status;\n    })\n  }\n\n  async ExtinguishAllAsync(): Promise<IStatus> {\n    return Queue.enqueue(async () => {\n      let status = await this.doRequest('ExtinguishAll');\n      return status;\n    })\n  }\n  public async GetGroupAsync(group: number): Promise<IGroupList> {\n    if (this.hideGroups) return;\n    return new Promise(async (resolve, reject) => {\n      try {\n        await this.GroupListGetAsync();\n        for (let i in this.GroupList) {\n          if (this.GroupList[i].GroupNumber === group) return resolve(this.GroupList[i]);\n        }\n        reject(`No Group Found in GroupGetAsync for group ${group}.\\n${JSON.stringify(this.GroupList)}`)\n      }\n      catch (err) {\n        reject(`Error with GetGroupAsync: ${err}`)\n        this.log.error(err);\n      }\n    })\n  }\n  async GroupListGetAsync(): Promise<IGroupList[]> {\n    // Get the list of light groups from the controller\n    if (typeof this.cacheGroupList !== 'undefined' && Date.now() - this.cacheGroupList < 2000) {\n      return (this.GroupList);\n    }\n\n    return Queue.enqueue(async () => {\n      let data = await this.doRequest('GroupListGet');\n      if (data.Status === 0) {\n        this.cacheGroupList = Date.now();\n        this.processGroupListGet(data as IGroupListResp);\n      }\n      return this.GroupList;\n    })\n  }\n  protected processGroupListGet(data: IGroupListResp): void {\n    // override with ZDC/ZDTWO\n    this.GroupList = data.GroupList;\n    for (let i in this.GroupList) {\n      this.GroupList[i].type = ILightType.ZD;\n    }\n  }\n  async GroupListEditAsync(name: string, groupNumber: number, color?: number): Promise<any> {\n    // Same in ZDC/ZDTWO\n    var requestData = JSON.stringify({\n      'Name': name,\n      'GroupNumber': groupNumber,\n      'Color': color\n    });\n    return Queue.enqueue(async () => {\n      let status = await this.doRequest('GroupListEdit', requestData);\n      return status;\n    })\n    /*     let status = await this.queueRequest('GroupListEdit', requestData);\n        return status; */\n  }\n  async ThemeListGetAsync(): Promise<IThemeList[]> {\n    if (typeof this.cacheThemeList !== 'undefined' && Date.now() - this.cacheThemeList < 2000) {\n      return (this.ThemeList);\n    }\n    return Queue.enqueue(async () => {\n      let data = await this.doRequest('ThemeListGet');\n      if (data.Status === 0) {\n        this.cacheThemeList = Date.now();\n        this.processThemeListGet(data as IThemeListResp);\n      }\n      return this.ThemeList;\n    })\n  }\n  protected processThemeListGet(data: IThemeListResp): void {\n    this.ThemeList = data.ThemeList;\n    for (var i in this.ThemeList) {\n      this.ThemeList[i].isOn = this.ThemeList[i].OnOff === 1;\n      this.ThemeList[i].type = ILightType.THEME;\n    }\n  }\n  public async GetThemeAsync(index: number): Promise<IThemeList> {\n    return new Promise(async (resolve, reject) => {\n      try {\n        await this.ThemeListGetAsync();\n        for (let i in this.ThemeList) {\n          if (this.ThemeList[i].ThemeIndex === index) return resolve(this.ThemeList[i]);\n        }\n        reject(`No Theme Found in ThemeGetAsync for theme ${index}.\\n${JSON.stringify(this.ThemeList)}`)\n      }\n      catch (err) {\n        this.log.error(err);\n        reject(`Error with GetThemeAsync ${err}`)\n      }\n    })\n\n  }\n  async IlluminateThemeAsync(themeIndex: number, onOff: number): Promise<IStatus> {\n    return Queue.enqueue(async () => {\n      let status = await this.doRequest('IlluminateTheme', {\n        'ThemeIndex': themeIndex,\n        'OnOff': onOff\n      });\n      this.cacheThemeList = undefined;\n      setTimeout(async () => { await this.updateLights() }, 250);\n      return status;\n    })\n  }\n  async IlluminateGroupAsync(groupNumber: number, desiredIntensity: number): Promise<IStatus> {\n    return Queue.enqueue(async () => {\n      let status = await this.doRequest('IlluminateGroup', {\n        'GroupNumber': groupNumber,\n        'Intensity': desiredIntensity\n      });\n      return status;\n    })\n  }\n  async ColorListGetAsync(): Promise<IColorList[]> {\n    return {} as IColorList[];\n  }\n  async ColorListSetAsync(color: number, hue: number, saturation: number): Promise<IStatus> {\n    return {} as IStatus;\n  }\n  processColorListGet(data: IColorListResp) {\n    this.ColorList = data.ColorList;\n  };\n  async GetColorAsync(color: number): Promise<IColorList> { return {} as IColorList; }\n\n  registerCallback(UUID: string, type: ILightType, index: number, characteristic: any, fn: () => {}) {\n    // look for an existing UUID/characteristic pair and update that if the light attributes get updates\n    for (let i = 0; i < this.callbackList.length; i++) {\n      let callback = this.callbackList[i];\n      if (callback.UUID == UUID && callback.characteristic === characteristic) {\n        callback = { UUID, type, index, characteristic, fn };\n        return;\n      }\n    }\n    // not found, add a new callback.\n    this.callbackList.push({ UUID, type, index, characteristic, fn });\n  }\n  execCallbacks() {\n    for (let i = 0; i < this.callbackList.length; i++) {\n      let callback = this.callbackList[i];\n      try {\n        switch (callback.type) {\n          case ILightType.ZD:\n          case ILightType.ZDC:\n            if (callback.characteristic.name === this.platform.Characteristic.Brightness.name) {\n              for (let i in this.GroupList){ // need to use loop as these are not 0 based indexes\n                if (this.GroupList[i].GroupNumber === callback.index){\n                  if (typeof this.GroupList[i].Intensity !== 'undefined') callback.fn(this.GroupList[i].Intensity);\n                  break;\n                }\n              }\n              // let group = this.GroupList.find(g=>g.GroupNumber === callback.index, this.GroupList);\n              \n            }\n            break;\n          case ILightType.THEME:\n            if (callback.characteristic.name === this.platform.Characteristic.On.name) {\n              for (let i in this.GroupList){ // need to use loop as these are not 0 based indexes\n                if (this.ThemeList[i].ThemeIndex === callback.index){\n                  if (typeof this.ThemeList[i].OnOff !== 'undefined') callback.fn(this.ThemeList[i].OnOff === 1);\n                  break;\n                }\n              }\n            }\n            break;\n        }\n      }\n      catch (err) {\n        this.log.error(`execCallbacks: ${err}`)\n      }\n    }\n  }\n  async updateLights(force: boolean = false) {\n    if (force) {\n      this.cacheGroupList = undefined;\n      this.cacheThemeList = undefined;\n    }\n    await this.GroupListGetAsync();\n    await this.ThemeListGetAsync();\n    this.execCallbacks();\n  }\n  async pollController() {\n    try {\n      await this.updateLights();\n    }\n    catch (err) { this.log.error(`${this.name} error: ${err}`) }\n    finally { setTimeout(async () => { await this.pollController() }, 30 * 1000); }\n  }\n\n}\n\nexport interface IGroupListResp {\n  Status: number;\n  StatusStr: string;\n  GroupList: IGroupList[];\n}\nexport interface IGroupList {\n  Name: string;\n  Grp?: number;\n  GroupNumber?: number;\n  Colr?: number;\n  Color?: number;\n  Inten?: number;\n  Intensity?: number;\n  type: ILightType;\n  UUID?: string;\n}\nexport interface IStatus {\n  Status: number;\n  StatusStr: string;\n}\nexport interface IThemeListResp {\n  Status: number;\n  StatusStr: string;\n  Restricted: 0 | 1;\n  ThemeList: IThemeList[];\n}\nexport interface IThemeList {\n  Name: string;\n  ThemeIndex: number;\n  OnOff: 0 | 1;\n  isOn: boolean;\n  type?: ILightType;\n  UUID?: string;\n}\nexport interface IColorListResp {\n  Status: number;\n  StatusStr: string;\n  ListSize: number;\n  ColorList: IColorList[];\n}\nexport interface IColorList {\n  C: number;\n  Hue: number;\n  Sat: number;\n}\nexport enum IControllerType {\n  ZD = 'ZD', ZDC = 'ZDC', ZDTWO = 'ZDTWO'\n}"]}