"use strict"; var __assign = (this && this.__assign) || function () { __assign = Object.assign || function(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; return __assign.apply(this, arguments); }; var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; var __generator = (this && this.__generator) || function (thisArg, body) { var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; function verb(n) { return function (v) { return step([n, v]); }; } function step(op) { if (f) throw new TypeError("Generator is already executing."); while (g && (g = 0, op[0] && (_ = 0)), _) try { if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; if (y = 0, t) op = [op[0] & 2, t.value]; switch (op[0]) { case 0: case 1: t = op; break; case 4: _.label++; return { value: op[1], done: false }; case 5: _.label++; y = op[1]; op = [0]; continue; case 7: op = _.ops.pop(); _.trys.pop(); continue; default: if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } if (t[2]) _.ops.pop(); _.trys.pop(); continue; } op = body.call(thisArg, _); } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; } }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.WebSocket2Ha = exports.getEntityTypeById = void 0; var ws_1 = __importDefault(require("ws")); var lodash_1 = __importDefault(require("lodash")); var init_1 = require("./init"); var protocols_1 = require("./protocols"); var coolkit_api_device_1 = __importDefault(require("coolkit-api-device")); var coolkit_ws_device_1 = __importDefault(require("coolkit-ws-device")); var process_1 = __importDefault(require("process")); var logger_1 = require("../utils/logger"); var ENTITY_TYPE_ALLOW_LIST = ['switch', 'light']; function allowedEntityType(entityType) { return ENTITY_TYPE_ALLOW_LIST.includes(entityType); } function getEntityTypeById(entityId) { return entityId.split('.')[0]; } exports.getEntityTypeById = getEntityTypeById; function entityStateFilter(states) { var result = []; for (var i = 0; i < states.length; i++) { var entityType = getEntityTypeById(states[i]['entity_id']); if (allowedEntityType(entityType)) { result.push(states[i]); } } return result; } function entityFilter(list, states) { var result = []; for (var i = 0; i < list.length; i++) { var entityType = getEntityTypeById(list[i]['entity_id']); var deviceId = list[i]['device_id']; var entityState = lodash_1.default.find(states, { 'entity_id': list[i]['entity_id'] }); if (allowedEntityType(entityType) && deviceId) { result.push(list[i]); } } return result; } function getHaDeviceDataById(deviceList, id) { for (var i = 0; i < deviceList.length; i++) { for (var j = 0; j < deviceList[i].haDeviceData.entities.length; j++) { if (deviceList[i].haDeviceData.entities[j].entityId === id) { return deviceList[i]; } } } return null; } var WebSocket2Ha = (function () { function WebSocket2Ha() { this.wsUrl = ''; this.connected = false; this.cmdId = 1; this.wsUrl = process_1.default.env.DEBUG_MODE ? process_1.default.env.HASS_WS_URL : (process_1.default.env.HA_URL ? process_1.default.env.HA_URL + '/api/websocket' : 'http://supervisor/core/websocket'); this.connect(); } WebSocket2Ha.prototype.connect = function () { var ws = new ws_1.default(this.wsUrl); this.ws = ws; ws.on('open', this.handleOpen.bind(this)); ws.on('close', this.handleClose.bind(this)); ws.on('error', this.handleError.bind(this)); ws.on('message', this.handleMessage.bind(this)); }; WebSocket2Ha.prototype.handleOpen = function () { logger_1.logger.info('WebSocket2Ha connecting...'); }; WebSocket2Ha.prototype.handleClose = function () { logger_1.logger.info('WebSocket2Ha will close'); this.ws.removeEventListener('open'); this.ws.removeEventListener('close'); this.ws.removeEventListener('error'); this.ws.removeEventListener('message'); }; WebSocket2Ha.prototype.handleError = function (err) { logger_1.logger.error("WebSocket2Ha error: ".concat(err)); }; WebSocket2Ha.prototype.handleMessage = function (msg) { try { var data = JSON.parse(msg); switch (data.type) { case 'auth_required': var resMsg = { type: 'auth', access_token: process_1.default.env.DEBUG_MODE ? process_1.default.env.HASS_LLAT : process_1.default.env.SUPERVISOR_TOKEN, }; this.ws.send(JSON.stringify(resMsg)); break; case 'auth_ok': this.connected = true; logger_1.logger.info('WebSocket2Ha connect success'); this.subscribeEvents('state_changed'); this.heartBeat(); break; case 'result': logger_1.logger.verbose('HA WebSocket get result message'); break; case 'event': if (data.event.event_type === 'state_changed') { this.handleHaStateChangedEvent(data); } break; default: break; } } catch (err) { logger_1.logger.error("WebSocket2Ha error: ".concat(err)); } }; WebSocket2Ha.prototype.heartBeat = function () { var _this = this; setInterval(function () { _this.sendMessage({ id: _this.cmdId++, type: 'ping' }); }, 10000); }; WebSocket2Ha.prototype.handleHaStateChangedEvent = function (e) { return __awaiter(this, void 0, void 0, function () { var entityId, gwInList, syncDeviceData, deviceList, deviceData, subDeviceListRes, subDeviceList, oldState, i, newState, haOldState, params; return __generator(this, function (_a) { switch (_a.label) { case 0: entityId = e.event.data.entity_id; if (!init_1.curUserGwData) { return [2]; } gwInList = init_1.curUserGwData.gwInList, syncDeviceData = init_1.curUserGwData.syncDeviceData; if (!(gwInList && syncDeviceData && allowedEntityType(getEntityTypeById(entityId)))) return [3, 2]; deviceList = syncDeviceData.filter(function (item) { return item.syncState && item.ckDeviceData; }); deviceData = getHaDeviceDataById(deviceList, entityId); if (!deviceData) { return [2, -1]; } return [4, coolkit_api_device_1.default.getGateWayAndSubDevice()]; case 1: subDeviceListRes = _a.sent(); if (subDeviceListRes.error !== 0) { return [2, -1]; } subDeviceList = subDeviceListRes.data.subDevicesInfo; oldState = null; for (i = 0; i < subDeviceList.length; i++) { if (subDeviceList[i].itemData.deviceid === deviceData.ckDeviceData.deviceid) { oldState = subDeviceList[i].itemData.params; break; } } if (!oldState) { return [2, -1]; } newState = e.event.data.new_state; if (newState.state === 'unavailable') { logger_1.logger.info("HA device unavailable, new state: ".concat(JSON.stringify(newState))); (0, init_1.setCkDeviceOnlineState)({ uiid: deviceData.deviceUiid, subDevId: deviceData.haDeviceData.deviceId, online: false, deviceid: deviceData.ckDeviceData.deviceid }); return [2]; } haOldState = e.event.data.old_state; params = (0, protocols_1.getDeviceUpdateParams)(deviceData, oldState, newState, haOldState); logger_1.logger.verbose("Send message to CK, params: ".concat(JSON.stringify(params))); if (!params) { return [2, -1]; } if (lodash_1.default.get(init_1.ws2ckRes, 'error') === 0) { (0, init_1.setCkDeviceOnlineState)({ uiid: deviceData.deviceUiid, subDevId: deviceData.haDeviceData.deviceId, online: true, deviceid: deviceData.ckDeviceData.deviceid }); coolkit_ws_device_1.default.sendMessage(JSON.stringify({ action: 'update', apikey: init_1.curUserGwData.userApiKey, deviceid: deviceData.ckDeviceData.deviceid, userAgent: 'device', params: params })); } _a.label = 2; case 2: return [2]; } }); }); }; WebSocket2Ha.prototype.sendMessage = function (data) { logger_1.logger.verbose("Send message to HA Core, data: ".concat(JSON.stringify(data))); this.ws.send(JSON.stringify(__assign({ id: this.cmdId++ }, data))); }; WebSocket2Ha.prototype.subscribeEvents = function (eventType) { var msg = { type: 'subscribe_events', event_type: eventType, id: this.cmdId++ }; this.ws.send(JSON.stringify(msg)); }; WebSocket2Ha.prototype.getHaData = function (type) { var _this = this; return new Promise(function (resolve) { var id = _this.cmdId++; var handler = function (data) { _this.ws.removeEventListener('message', handler); var msg = JSON.parse(data); if (msg.success) { resolve(msg.result); } else { resolve(-1); } }; _this.ws.send(JSON.stringify({ id: id, type: type })); _this.ws.on('message', handler); setTimeout(function () { _this.ws.removeEventListener('message', handler); resolve(-1); }, 5000); }); }; WebSocket2Ha.prototype.getEntityStates = function () { return __awaiter(this, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4, this.getHaData('get_states')]; case 1: return [2, _a.sent()]; } }); }); }; WebSocket2Ha.prototype.getConfigDeviceRegistryList = function () { return __awaiter(this, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4, this.getHaData('config/device_registry/list')]; case 1: return [2, _a.sent()]; } }); }); }; WebSocket2Ha.prototype.getConfigEntityRegistryList = function () { return __awaiter(this, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4, this.getHaData('config/entity_registry/list')]; case 1: return [2, _a.sent()]; } }); }); }; WebSocket2Ha.prototype.getHaDeviceEntityMap = function () { return __awaiter(this, void 0, void 0, function () { var result, entityStatesRes, entityStateList, deviceRes, deviceList, entityRes, entityList, i, device, deviceId, entityId, entityState, index; return __generator(this, function (_a) { switch (_a.label) { case 0: result = []; return [4, this.getEntityStates()]; case 1: entityStatesRes = _a.sent(); logger_1.logger.verbose("getHaDeviceEntityMap: entityStatesRes: ".concat(JSON.stringify(entityStatesRes))); if (entityStatesRes === -1) { return [2, result]; } entityStateList = entityStateFilter(entityStatesRes); logger_1.logger.verbose("getHaDeviceEntityMap: entityStateList: ".concat(JSON.stringify(entityStateList))); return [4, this.getConfigDeviceRegistryList()]; case 2: deviceRes = _a.sent(); if (deviceRes === -1) { return [2, result]; } deviceList = deviceRes; logger_1.logger.verbose("getHaDeviceEntityMap: deviceList: ".concat(JSON.stringify(deviceList))); return [4, this.getConfigEntityRegistryList()]; case 3: entityRes = _a.sent(); if (entityRes === -1) { return [2, result]; } entityList = entityFilter(entityRes, entityStateList); logger_1.logger.verbose("getHaDeviceEntityMap: entityList: ".concat(JSON.stringify(entityList))); for (i = 0; i < entityList.length; i++) { device = lodash_1.default.find(deviceList, { id: entityList[i]['device_id'] }); deviceId = device['id']; entityId = entityList[i]['entity_id']; entityState = lodash_1.default.find(entityStateList, { entity_id: entityId }); index = lodash_1.default.findIndex(result, { deviceId: deviceId }); if (index === -1) { result.push({ deviceId: deviceId, deviceData: device, entities: [{ entityId: entityId, entityData: entityList[i], entityState: entityState }] }); } else { result[index].entities.push({ entityId: entityId, entityData: entityList[i], entityState: entityState }); } } logger_1.logger.verbose("getHaDeviceEntityMap: result: ".concat(JSON.stringify(result))); return [2, result]; } }); }); }; return WebSocket2Ha; }()); exports.WebSocket2Ha = WebSocket2Ha;