Created
April 5, 2020 01:03
-
-
Save imnotbob/310d9dde3a25cb17a243451445eb3ae2 to your computer and use it in GitHub Desktop.
IntesisHome for Hubitat
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* IntesisHome Connect | |
* | |
* Author: ERS | |
* based off device work by Martin Blomgren | |
* Last update: 2019-12-14 | |
* | |
* Licensed under the Apache License, Version 2.0 (the 'License'); you may not | |
* use this file except in compliance with the License. You may obtain a copy | |
* of the License at: | |
* | |
* http://www.apache.org/licenses/LICENSE-2.0 | |
* | |
* Unless required by applicable law or agreed to in writing, software | |
* distributed under the License is distributed on an 'AS IS' BASIS, WITHOUT | |
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | |
* License for the specific language governing permissions and limitations | |
* under the License. | |
*/ | |
definition( | |
name: 'IntesisHome Connect', | |
namespace: 'imnotbob', | |
author: 'ERS', | |
description: 'Allows you to integrate your Intesis with Hubitat.', | |
singleInstance: true, | |
iconUrl: "", | |
iconX2Url: "", | |
iconX3Url: "", | |
// importUrl: "https://raw.githubusercontent.com/tonesto7/nst-manager-he/master/apps/nstManager.groovy", | |
) | |
preferences { | |
page(name: 'mainPage') | |
} | |
def mainPage() { | |
dynamicPage( | |
name: 'mainPage', | |
install: true, | |
uninstall: true, | |
refreshInterval: 30 | |
) { | |
section("Authentication") { | |
input "username", "text", title: "Username" | |
input "password", "password", title: "Password" | |
} | |
if(username && password) { | |
section("Disable updating here") { | |
input "enabled", "bool", defaultValue: "true", title: "Enabled?" | |
} | |
} | |
section("Logging") { | |
input name: "logEnable", type: "bool", title: "Enable debug logging", defaultValue: true | |
input name: "txtEnable", type: "bool", title: "Enable descriptionText logging", defaultValue: true | |
} | |
} | |
} | |
void installed() { | |
//log.debug('Installed') | |
initialize() | |
} | |
void updated() { | |
//log.debug('Updated') | |
initialize() | |
} | |
void initialize() { | |
//log.debug 'Initializing' | |
unschedule() | |
if (logEnable) runIn(1800,logsOff) | |
if(enabled) pollStatus() | |
else { | |
def tdev = getTelnetDev() | |
if(tdev) { | |
tdev.stop() | |
} | |
} | |
} | |
void logsOff() { | |
debug "logsOff", "debug logging disabled..." | |
app.updateSetting("logEnable",[value:"false",type:"bool"]) | |
app.updateSetting("txtEnable",[value:"false",type:"bool"]) | |
} | |
void pollStatus() { | |
if (logEnable) debug("pollStatus()", "") | |
if(username && password && enabled) { | |
def params = [ | |
uri : INTESIS_URL, | |
contentType: "application/x-www-form-urlencoded", | |
body : 'username=' + username + '&password=' + password + '&cmd={"status":{"hash":"x"},"config":{"hash":"x"}}&version=1.8.5' | |
] | |
try { | |
asynchttpPost('handlePollResponse', params, data) | |
} catch (e) { | |
queuePollStatus(600) | |
error("pollStatus", "Error polling", e) | |
} | |
} else { log.warn "missing settings $username $enabled" } | |
} | |
void handlePollResponse(response, data) { | |
if (logEnable) debug("handlePollResponse()", "") | |
def responseError = response.hasError() | |
def responseJson | |
if (!responseError) responseJson = parseJson(response.data) | |
if (responseError || responseJson?.errorMessage) { | |
queuePollStatus(900) | |
debug("hasError", "$responseError") | |
def responseErrorStatus = response.getStatus() | |
debug("errorStatus", "$responseErrorStatus") | |
def responseErrorData = response.getErrorData() | |
debug("errorData", "$responseErrorData") | |
def responseErrorMessage= response.getErrorMessage() | |
debug("errorMessage", "$responseErrorMessage") | |
debug("errorjsonData", "$responseJson") | |
return | |
} | |
//if (logEnable) debug("responseJson", "$responseJson") | |
def devMap = [:] | |
def instMap = [:] | |
atomicState.server = responseJson['config']['serverIP'] | |
atomicState.serverPort = responseJson['config']['serverPort'] | |
atomicState.token = responseJson['config']['token'] | |
responseJson['config']['inst'].each { installation -> | |
instMap["${installation.name}"] = installation | |
if (logEnable) debug("Found installation", "${installation.name} total installations: ${instMap.size()}") | |
state.installationMap = instMap | |
//log.warn "Installation: $installation" | |
installation['devices'].each { device -> | |
devMap["${device.id}"] = [:] + device | |
devMap["${device.id}"].valMap = [:] | |
if (logEnable) debug("Found device", "${device.name}, total devices ${devMap.size()}") | |
//state.deviceId = device.id | |
state.deviceMap = devMap | |
} | |
} | |
// Update state attributes | |
responseJson['status']['status'].each { status -> | |
devMap["${status.deviceId}"].valMap."${status.uid}" = status.value | |
state.deviceMap = devMap | |
} | |
def child | |
devMap.each { dev -> | |
def t0 = "${dev.value.id}" | |
def tdev = getChildDevice(t0) | |
if(tdev) { | |
//log.warn "found device ${tdev}" | |
} else { | |
child = addChildDevice( 'imnotbob', 'IntesisHome HVAC', t0, null, [label: "${dev.value.name}"]) | |
debug( | |
"handlePollResponse", "Created ${child.displayName} with id: " + | |
"${child.id}, MAC: ${child.deviceNetworkId}" | |
) | |
pause(2000) | |
} | |
} | |
def child1 = getTelnetDev() | |
if(!child1) { | |
child1 = addChildDevice( 'imnotbob', 'IntesisHome Telnet', 'IntTelnet', null, [label: "IntesisHome Telnet Driver"]) | |
debug( | |
"halePollResponse", "Created ${child1.displayName} with id: " + | |
"${child1.id}, MAC: ${child1.deviceNetworkId}" | |
) | |
pause(2000) | |
} | |
devMap.each { dev -> | |
def t0 = "${dev.value.id}" | |
def tdev = getChildDevice(t0) | |
if(tdev) { | |
tdev.generateEvent(dev.value) | |
} else { debug "handlePollResponse", "child not found $t0" } | |
} | |
if(child1) child1.connect() | |
else debug "handlePollResponse", "Telnet device not found" | |
} | |
void telnetUp() { | |
if (logEnable) debug("telnetUp", "") | |
atomicState.telnet = true | |
} | |
void telnetDown(boolean runPoll=false) { | |
if (logEnable) debug("telnetDown(${runPoll})", "") | |
atomicState.telnet = false | |
atomicState.server = null | |
atomicState.serverPort = null | |
atomicState.token = null | |
if(runPoll) runIn(24, pollStatus) | |
else if(enabled) runIn(900, pollStatus) | |
} | |
void queuePollStatus(int delay=3) { | |
if (logEnable) debug("queuePollStatus()", "") | |
runIn(delay, pollStatus) | |
} | |
def getTelnetDev() { | |
def tdev = app.getChildDevices() | |
def myDev | |
tdev.each { dev -> | |
if(dev?.typeName in ['IntesisHome Telnet']) { | |
myDev = dev | |
} | |
} | |
myDev | |
} | |
void updateDeviceState(long deviceId, int uid, short value) { | |
def t0 = "${deviceId}" | |
def tdev = app.getChildDevice(t0) | |
if(tdev) tdev.updateDeviceState(deviceId, uid, value) | |
else log.warn "no tdev $deviceId $uid $value" | |
} | |
def getParams() { | |
return [server: atomicState.server, port: atomicState.serverPort, token: atomicState.token] | |
} | |
// --- "Constants" & Global variables | |
def getINTESIS_URL() { return "https://user.intesishome.com/api.php/get/control" } | |
def getINTESIS_CMD_STATUS() { return '{"status":{"hash":"x"},"config":{"hash":"x"}}' } | |
def getINTESIS_API_VER() { return "2.1" } | |
def sendMsg(String msg) { | |
def tdev = getTelnetDev() | |
if(tdev) tdev.sendMsg(msg) | |
else log.warn "did not find telnet device" | |
} | |
void refresh() { | |
if (logEnable) log.debug "refresh" | |
} | |
private String createLogString(String context, String message) { | |
return "[IntesisHome Connect." + context + "] " + message | |
} | |
private void error(String context, String text, Exception e) { | |
error(context, text, e, true) | |
} | |
private void error(String context, String text, Exception e, Boolean remote) { | |
log.error(createLogString(context, text) + e?.message) | |
} | |
private void debug(String context, String text) { | |
debug(context, text, true) | |
} | |
private void debug(String context, String text, Boolean remote) { | |
log.debug(createLogString(context, text)) | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* Intesis Telnet 0.1 | |
* | |
* Author: ERS | |
* based off device work by Martin Blomgren | |
* Last update: 2019-12-14 | |
* | |
* Thanks to James Nimmo for the massive work with the Python IntesisHome module | |
* (https://github.com/jnimmo/pyIntesisHome) | |
* | |
* MIT License | |
* | |
* Copyright (c) 2019 | |
* | |
* Permission is hereby granted, free of charge, to any person obtaining a copy | |
* of this software and associated documentation files (the "Software"), to deal | |
* in the Software without restriction, including without limitation the rights | |
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |
* copies of the Software, and to permit persons to whom the Software is | |
* furnished to do so, subject to the following conditions: | |
* | |
* The above copyright notice and this permission notice shall be included in all | |
* copies or substantial portions of the Software. | |
* | |
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |
* SOFTWARE. | |
* | |
*/ | |
def version() {"v0.1"} | |
import groovy.transform.Field | |
import hubitat.helper.InterfaceUtils | |
import groovy.json.JsonSlurper | |
metadata { | |
definition (name: "IntesisHome Telnet", namespace: 'imnotbob', author: "ERS") { | |
capability "Configuration" | |
capability "Initialize" | |
capability "Refresh" | |
capability "Telnet" | |
capability "Actuator" | |
attribute "Telnet", "string" | |
command "connect" | |
command "stop" | |
} | |
preferences { | |
// if(username && password) { | |
// section("Disable updating here") { | |
// input "enabled", "bool", defaultValue: "true", title: "Enabled?" | |
// } | |
// } | |
section("Logging") { | |
input name: "logEnable", type: "bool", title: "Enable debug logging", defaultValue: true | |
input name: "txtEnable", type: "bool", title: "Enable descriptionText logging", defaultValue: true | |
} | |
} | |
} | |
void initialize() { | |
debug "initialize", "" | |
state.enabled = true | |
if(state.connected) checkLastReceived() | |
else connect() | |
} | |
void installed() { | |
initialize() | |
} | |
void logsOff() { | |
debug "logsOff", "debug logging disabled..." | |
device.updateSetting("logEnable",[value:"false",type:"bool"]) | |
} | |
void updated() { | |
unschedule() | |
debug "updated", "debug logging is: ${logEnable == true}" | |
debug "updated", "description logging is: ${txtEnable == true}" | |
if (logEnable) runIn(1800,logsOff) | |
runEvery10Minutes(checkLastReceived) | |
initialize() | |
} | |
void connect() { | |
state.enabled = true | |
if(!state.connected) { | |
def myVars = parent.getParams() | |
state.server = myVars.server | |
state.serverPort = myVars.port | |
state.token = myVars.token | |
if (state.enabled && state.server && state.serverPort && state.token) { | |
debug "connect:", " session to IntesisHome at ${state.server}:${state.serverPort}" | |
//Connect to the IntesisHome | |
try { | |
state.connected = true | |
//open telnet connection | |
telnetConnect([termChars: [125,125], terminalType: 'VT100'], state.server, state.serverPort, null, null) | |
} | |
catch(e) { | |
state.connected = false | |
if (logEnable) debug "connect:", "initialize error ${e.message}" | |
error "connect:", "Telnet connect failed in connect()", e | |
parent.telnetDown() | |
return | |
} | |
connectionMade() | |
} else { | |
//Get connection details | |
debug("connect:", "something missing ${state.enabled} ${state.server} ${state.serverPort} ${state.token}") | |
if(state.enabled) { | |
parent.telnetDown(state.enabled) | |
} | |
} | |
} else { | |
debug "connect:", "Connect called while connected, skipping" | |
} | |
} | |
void connectionMade() { | |
// Authenticate | |
String authMsg = '{"command":"connect_req","data":{"token":' + state.token + '}}' | |
if (logEnable) debug "connectionMade:", "authMsg ${authMsg}" | |
sendHubCommand(new hubitat.device.HubAction(authMsg, hubitat.device.Protocol.TELNET)) | |
} | |
void sendMsg(String msg) { | |
if(!state.connected) { log.warn "sendMsg when not connected" } | |
sendHubCommand(new hubitat.device.HubAction(msg, hubitat.device.Protocol.TELNET)) | |
} | |
void stop() { | |
debug "stop", "" | |
state.enabled = false | |
// device.updateSetting("enabled",[value:"false",type:"bool"]) | |
if(state.connected) { | |
state.connected = false | |
telnetClose() | |
} | |
debug "stop", "Telnet connection dropped..." | |
sendEvent(name: "Telnet", value: "Disconnected") | |
parent.telnetDown() | |
} | |
// Parse incoming device messages to generate events | |
void parse(String message) { | |
if(!state.enabled) { stop(); return } | |
if (message.contains('"uid":60002')) return // rssi | |
// As we don't have any termination character we nee to put back the curly braces again | |
//def msg = message + '}}' | |
//log.debug "[IntesisHome] parse message: ${msg}" | |
def jsonSlurper = new JsonSlurper() | |
def messageJson = jsonSlurper.parseText(message + '}}') | |
if (logEnable) debug "parse", "messageJson: ${messageJson}" | |
switch (messageJson.command) { | |
case "connect_rsp": | |
if (messageJson.data.status == "ok") { | |
sendEvent(name: "Telnet", value: "Connected"); | |
parent.telnetUp() | |
} | |
break | |
case "status": | |
//updateDeviceState(Int deviceId, Int uid, short value) | |
parent.updateDeviceState(messageJson.data.deviceId, messageJson.data.uid, (Short)messageJson.data.value) | |
break | |
case "rssi": | |
// [command:rssi, data:[deviceId:0123456789, value:196]] | |
break | |
default: | |
break | |
} | |
//[command:connect_rsp, data:[status:ok]] | |
state.lastReceived = now() | |
} | |
void checkLastReceived() { | |
long t0 = now() | |
def t1 = state.lastReceived | |
if (state.connected && t0 > (t1 + 300000)) { // 5 mins no data | |
debug "checkLastReceived", "Telnet connection dropped...lastReceived: ${t1}" | |
endConnection() | |
telnetClose() | |
parent.telnetDown(state.enabled) | |
} else if(!state.connected) parent.telnetDown(state.enabled) | |
} | |
void endConnection() { | |
state.connected = false | |
state.server = null | |
state.serverPort = null | |
state.token = null | |
sendEvent(name: "Telnet", value: "Disconnected") | |
} | |
void telnetStatus(String status) { | |
if (logEnable) debug "telnetStatus", "${status}" | |
if (status == "receive error: Stream is closed") { | |
debug "telnetStatus", "Telnet connection dropped..." | |
endConnection() | |
parent.telnetDown(state.enabled) | |
if(!state.enabled) debug "telnetStatus", "Telnet driver disabled" | |
} else { | |
if (txtEnable) debug "telnetStatus", "OK, ${status}" | |
state.connected = true | |
sendEvent(name: "Telnet", value: "Connected") | |
parent.telnetUp() | |
} | |
} | |
def setValue() {} | |
void refresh() { | |
if (logEnable) debug "refresh", "" | |
} | |
void configure() { | |
if (txtEnable) debug "Configure", "Reporting and Bindings." | |
initialize() | |
} | |
private String createLogString(String context, String message) { | |
return "[IntesisHome Telnet." + context + "] " + message | |
} | |
private void error(String context, String text, Exception e) { | |
error(context, text, e, true) | |
} | |
private void error(String context, String text, Exception e, Boolean remote) { | |
log.error(createLogString(context, text) + e?.message) | |
} | |
private void debug(String context, String text) { | |
debug(context, text, true) | |
} | |
private void debug(String context, String text, Boolean remote) { | |
log.debug(createLogString(context, text)) | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* Intesis HVAC 0.1 | |
* | |
* Author: ERS | |
* based off device work by Martin Blomgren | |
* Last update: 2020-3-25 | |
* | |
* Thanks to James Nimmo for the massive work with the Python IntesisHome module | |
* (https://github.com/jnimmo/pyIntesisHome) | |
* | |
* MIT License | |
* | |
* Copyright (c) 2019 | |
* | |
* Permission is hereby granted, free of charge, to any person obtaining a copy | |
* of this software and associated documentation files (the "Software"), to deal | |
* in the Software without restriction, including without limitation the rights | |
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |
* copies of the Software, and to permit persons to whom the Software is | |
* furnished to do so, subject to the following conditions: | |
* | |
* The above copyright notice and this permission notice shall be included in all | |
* copies or substantial portions of the Software. | |
* | |
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |
* SOFTWARE. | |
* | |
*/ | |
def version() {"v0.1"} | |
import groovy.transform.Field | |
import hubitat.helper.InterfaceUtils | |
import groovy.json.JsonSlurper | |
metadata { | |
definition (name: "IntesisHome HVAC", namespace: 'imnotbob', author: "ERS") { | |
capability "Configuration" | |
capability "Refresh" | |
capability "Actuator" | |
capability "FanControl" | |
// capability "Relative Humidity Measurement" | |
capability "Temperature Measurement" | |
capability "Sensor" | |
capability "Energy Meter" | |
capability "Power Meter" | |
capability "Thermostat" | |
// capability "Switch" | |
//attribute "swing", "string" | |
//attribute "temperatureUnit","string" | |
attribute "outdoorTemperature", "number" | |
// attribute "latestMode", "string" | |
attribute "iFanSpeed", "string" | |
attribute "ivvane", "string" | |
attribute "ihvvane", "string" | |
command "dry" | |
command "on" | |
} | |
preferences { | |
// if(username && password) { | |
// section("Disable updating here") { | |
// input "enabled", "bool", defaultValue: "true", title: "Enabled?" | |
// } | |
// } | |
section("Logging") { | |
input name: "logEnable", type: "bool", title: "Enable debug logging", defaultValue: true | |
input name: "txtEnable", type: "bool", title: "Enable descriptionText logging", defaultValue: true | |
} | |
} | |
} | |
// --- "Constants" & Global variables | |
def getINTESIS_URL() { return "https://user.intesishome.com/api.php/get/control" } | |
def getINTESIS_CMD_STATUS() { return '{"status":{"hash":"x"},"config":{"hash":"x"}}' } | |
def getINTESIS_API_VER() { return "2.1" } | |
def getAPI_DISCONNECTED() { return "Disconnected" } | |
def getAPI_CONNECTING() { return "Connecting" } | |
def getAPI_AUTHENTICATED() { return "Connected" } | |
def getAPI_AUTH_FAILED() { return "Wrong username/password" } | |
Map getINTESIS_MAP() { | |
String map = """ | |
{ | |
"1": {"name": "power", "values": {"0": "off", "1": "on"}}, | |
"2": {"name": "mode", "values": {"0": "auto", "1": "heat", "2": "dry", "3": "fan", "4": "cool"}}, | |
"4": {"name": "fan_speed", "values": {"0": "auto", "1": "quiet", "2": "low", "3": "medium", "4": "high"}}, | |
"5": {"name": "vvane", "values": {"0": "auto/stop", "10": "swing", "1": "manual1", "2": "manual2", "3": "manual3", "4": "manual4", "5": "manual5"}}, | |
"6": {"name": "hvane", "values": {"0": "auto/stop", "10": "swing", "1": "manual1", "2": "manual2", "3": "manual3", "4": "manual4", "5": "manual5"}}, | |
"9": {"name": "setpoint", "null": 32768}, | |
"10": {"name": "temperature"}, | |
"13": {"name": "working_hours"}, | |
"35": {"name": "setpoint_min"}, | |
"36": {"name": "setpoint_max"}, | |
"37": {"name": "outdoor_temperature"}, | |
"68": {"name": "current_power_consumption"}, | |
"69": {"name": "total_power_consumption"}, | |
"70": {"name": "weekly_power_consumption"} | |
} | |
""" | |
/* """ */ | |
return new JsonSlurper().parseText(map) | |
} | |
Map getCOMMAND_MAP() { | |
String cmd = """ | |
{ | |
"power": {"uid": 1, "values": {"off": 0, "on": 1}}, | |
"mode": {"uid": 2, "values": {"auto": 0, "heat": 1, "dry": 2, "fan": 3, "cool": 4}}, | |
"fan_speed": {"uid": 4, "values": {"auto": 0, "quiet": 1, "low": 2, "medium": 3, "high": 4}}, | |
"vvane": {"uid": 5, "values": {"auto/stop": 0, "swing": 10, "manual1": 1, "manual2": 2, "manual3": 3, "manual4": 4, "manual5": 5}}, | |
"hvane": {"uid": 6, "values": {"auto/stop": 0, "swing": 10, "manual1": 1, "manual2": 2, "manual3": 3, "manual4": 4, "manual5": 5}}, | |
"setpoint": {"uid": 9} | |
} | |
""" | |
return new JsonSlurper().parseText(cmd) | |
} | |
void initialize() { | |
debug "initialize", "" | |
setModes() | |
} | |
def installed() { | |
String tempscale = getTemperatureScale() | |
def tz = location.timeZone | |
if(!tz || !(tempscale == "F" || tempscale == "C")) { | |
log.warn "Timezone (${tz}) or Temperature Scale (${tempscale}) not set" | |
} | |
// set some dummy values, for google integration | |
if(tempscale=='F') { | |
sendEvent(name:"coolingSetpoint", value:80) | |
}else{ | |
sendEvent(name:"coolingSetpoint", value:28) | |
} | |
initialize() | |
return [:] | |
} | |
void logsOff() { | |
debug "logsOff", "text logging disabled..." | |
debug "logsOff", "debug logging disabled..." | |
device.updateSetting("logEnable",[value:"false",type:"bool"]) | |
device.updateSetting("txtEnable",[value:"false",type:"bool"]) | |
} | |
def updated() { | |
debug "updated", "debug logging is: ${logEnable == true}" | |
debug "updated", "description logging is: ${txtEnable == true}" | |
if (logEnable) runIn(1800,logsOff) | |
initialize() | |
return [:] | |
} | |
void setModes() { | |
// supported in device "auto", "heat", "dry", "fan", "cool" | |
def supportedThermostatModes = ["off", "auto", "heat", "cool"] // HE capabilities (no "emerency heat") | |
// supported in device "auto", "quiet", "low", "medium", "high" | |
def supportedFanModes = ["auto", "on", "circulate"] // HE capabilities | |
sendEvent(name: "supportedThermostatModes", value: supportedThermostatModes, displayed: false ) | |
sendEvent(name: "supportedThermostatFanModes", value: supportedFanModes, displayed: false) | |
// not allowed | |
//def supportedThermostatModes = ["off", "auto", "heat", "dry", "fan", "cool"] | |
//def supportedFanModes = ["auto", "quiet", "low", "medium", "high"] | |
} | |
void generateEvent(tData) { | |
Long myId = "${tData.id}".toLong() | |
state.deviceId = myId | |
tData.valMap.each { val -> | |
updateDeviceState(myId, val.key.toInteger(), (Short)val.value) | |
} | |
} | |
void updateDeviceState(Long deviceId, Integer uid, short value) { | |
if (uid == 60002) return | |
// if (logEnable) log.debug "[IntesisHome.thermostat] updateDeviceState: deviceId=${deviceId}, uid=${uid}, value=${value}" | |
String sUid = uid.toString() | |
Map myINTESIS_MAP = INTESIS_MAP | |
if (myINTESIS_MAP.containsKey(sUid)) { | |
if (myINTESIS_MAP[sUid].containsKey('values')) { // power, mode, fan_speed, vvane, hvane | |
String valuesValue = myINTESIS_MAP[sUid].values[value.toString()] | |
switch ((String)myINTESIS_MAP[sUid].name) { | |
case "power": // off, on | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState power: $valuesValue" | |
if (valuesValue == "off") { | |
state.mpower = false | |
sendEvent(name: "thermostatMode", value: valuesValue) | |
//sendEvent(name: "switch", value: "off") | |
sendEvent(name: "thermostatOperatingState", value: "idle") | |
} else if (valuesValue == "on") { | |
if((Boolean)state.mpower == false && (Boolean)state.mpower != null) { | |
state.mpower = true // if we transition off -> on, force re-update of variables | |
//sendEvent(name: "switch", value: "on") | |
parent.queuePollStatus() | |
return | |
} | |
// state.mpower = true | |
// sendEvent(name: "thermostatMode", value: device.currentValue("latestMode", true)) | |
// updateOperatingState() | |
} | |
break | |
case "mode": // auto, heat, dry, fan, cool | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState mode: $valuesValue" | |
// thermostatMode - auto, heat, cool, 'off', 'emergency heat' | |
String myVal = valuesValue | |
//state.lastMode = valuesValue //sendEvent(name: "latestMode", value: valuesValue) | |
if(myVal!='off')state.lastMode = myVal | |
if((Boolean)state.mpower) { | |
state.curMode=myVal | |
if(myVal == 'dry') myVal = 'cool' | |
else if(myVal == 'fan') { | |
myVal = 'off' | |
sendEvent(name: "thermostatFanMode", value: 'on') | |
} | |
sendEvent(name: "thermostatMode", value: myVal) | |
//if(myVal!='off')state.lastMode = myVal | |
updateOperatingState() | |
} else { | |
myVal = 'off' | |
state.curMode=myVal | |
sendEvent(name: "thermostatMode", value: myVal) | |
sendEvent(name: "thermostatOperatingState", value: "idle") | |
} | |
break | |
case "fan_speed": // auto, quiet, low, medium, high | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState fan_speed: $valuesValue" | |
//if (!state.mpower) sendEvent(name: "thermostatFanMode", value: 'auto') | |
//else | |
sendEvent(name: "thermostatFanMode", value: valuesValue != 'auto' ? 'on' : 'auto') | |
sendEvent(name: "speed", value: valuesValue) | |
sendEvent(name: "iFanSpeed", value: valuesValue) | |
break | |
case "vvane": // auto/stop, swing, manual1, manual2, manual3, manual4, manual5 | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState vvane: $valuesValue" | |
sendEvent(name: "ivvane", value: valuesValue) | |
break | |
case "hvane": // auto/stop, swing, manual1, manual2, manual3, manual4, manual5 | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState hvane: $valuesValue" | |
sendEvent(name: "ihvvane", value: valuesValue) | |
break | |
default: | |
if (logEnable) log.info "[IntesisHome.thermostat] updateDeviceState values uid NOT FOUND" | |
break | |
} | |
} else if (myINTESIS_MAP[sUid].containsKey('null') && value == myINTESIS_MAP[sUid].null) { | |
//setPointTemperature should be set to none... | |
} else { | |
def tempVal = getTemperatureScale() == 'C' ? value/10 : Math.round(( (value/10.0) * (9.0/5.0) + 32.0) ) | |
String myUnit = "\u00b0${getTemperatureScale()}" | |
switch ((String)myINTESIS_MAP[sUid].name) { | |
case "setpoint": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState setpoint: ${value/10}" | |
sendEvent(name: "thermostatSetpoint", value: tempVal, unit: myUnit) | |
def cVal = tempVal | |
def hVal = tempVal //t1 | |
String t0=(String)state.curMode | |
//String t0 = device.currentValue("latestMode", true) | |
//if (t0 == "heat") { cVal = 0 } | |
//if (t0 == "cool") { hVal = 0 } | |
if (t0 == "heat") sendEvent(name: "heatingSetpoint", value: hVal, unit: myUnit) | |
if (t0 == "cool") sendEvent(name: "coolingSetpoint", value: cVal, unit: myUnit) | |
//sendEvent(name: "coolingSetpoint", value: cVal, unit: myUnit) | |
//sendEvent(name: "heatingSetpoint", value: hVal, unit: myUnit) | |
break | |
case "temperature": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState temperature: ${value/10}" | |
sendEvent(name: "temperature", value: tempVal, unit: myUnit) | |
break | |
case "working_hours": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState working_hours: $value" | |
//sendEvent(name: "ThermostatSetpoint", value: value/) | |
break | |
case "setpoint_min": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState setpoint_min: ${value/10}" | |
//sendEvent(name: "ThermostatSetpoint", value: value/10) | |
break | |
case "setpoint_max": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState setpoint_max: ${value/10}" | |
//sendEvent(name: "ThermostatSetpoint", value: value/10) | |
break | |
case "outdoor_temperature": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState outdoor_temperature: ${value/10}" | |
sendEvent(name: "outdoorTemperature", value: tempVal, unit: myUnit) | |
break | |
case "current_power_consumption": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState current_power_consumption: $value" | |
sendEvent(name: "power", value: value) | |
// thermostatMode - auto, heat, dry, fan, cool | |
// thermostatOperatingState - ENUM ["vent economizer", "pending cool", "cooling", "heating", "pending heat", "fan only", "idle"] | |
if (value < 20 || !(Boolean)state.mpower) { | |
sendEvent(name: "thermostatOperatingState", value: "idle") | |
} else if ((Boolean)state.mpower) updateOperatingState() | |
break | |
case "total_power_consumption": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState total_power_consumption: $value" | |
sendEvent(name: "energy", value: value) | |
break | |
case "weekly_power_consumption": | |
if (txtEnable) log.info "[IntesisHome.thermostat] updateDeviceState weekly_power_consumption: $value" | |
//sendEvent(name: "ThermostatSetpoint", value: value/10) | |
break | |
default: | |
if (logEnable) log.debug "[IntesisHome.thermostat] updateDeviceState non-values uid NOT FOUND" | |
break | |
} | |
} | |
} | |
} | |
void updateOperatingState() { | |
if(!(Boolean)state.mpower) return | |
String t0=(String)state.curMode | |
//String t0 = device.currentValue("latestMode", true) | |
//off is handled elsewhere | |
if (t0 == "auto") { | |
sendEvent(name: "thermostatOperatingState", value: "heating") | |
//sendEvent(name: "thermostatOperatingState", value: "") | |
} else if (t0 == "heat") { | |
sendEvent(name: "thermostatOperatingState", value: "heating") | |
} else if (t0 == "dry") { | |
sendEvent(name: "thermostatOperatingState", value: "vent economizer") | |
} else if (t0 == "fan") { | |
sendEvent(name: "thermostatOperatingState", value: "fan only") | |
} else if (t0 == "cool") { | |
sendEvent(name: "thermostatOperatingState", value: "cooling") | |
} | |
} | |
void setPointAdjust(double value) { | |
Integer intVal = getTemperatureScale() == 'C' ? Math.round(value*10) : Math.round( ((value - 32.0) * (5.0/9.0)) * 10.0 ) | |
String myUnit = "\u00b0${getTemperatureScale()}" | |
if (txtEnable) log.info "[IntesisHome.thermostat] setPointAdjust to: $intVal from $value $myUnit" | |
//def uid = 9 | |
Integer uid = COMMAND_MAP['setpoint']['uid'] | |
String message = '{"command":"set","data":{"deviceId":' + (Long)state.deviceId + ',"uid":' + uid + ',"value":' + intVal + ',"seqNo":0}}' | |
parent.sendMsg(message) | |
} | |
void setHeatingSetpoint(double value) { | |
if (txtEnable) log.info "[IntesisHome.thermostat] setHeatingSetpoint to: $value" | |
setPointAdjust(value) | |
} | |
void setCoolingSetpoint(double value) { | |
if (txtEnable) log.info "[IntesisHome.thermostat] setCoolingSetpoint to: $value" | |
setPointAdjust(value) | |
} | |
void setThermostatMode(String mode) { | |
if (txtEnable) log.info "[IntesisHome.thermostat] setThermostatMode to: $mode" | |
//supportedThermostatModes : [off, auto, heat, dry, fan, cool] | |
if(mode == 'off') { | |
setPower('off') | |
} | |
if(mode == 'emergency heat') mode = 'heat' | |
Integer uid = COMMAND_MAP['mode']['uid'] | |
Integer value = COMMAND_MAP['mode'].values[mode] | |
String message = '{"command":"set","data":{"deviceId":' + (Long)state.deviceId + ',"uid":' + uid + ',"value":' + value + ',"seqNo":0}}' | |
if (logEnable) log.debug "[IntesisHome.thermostat] send message: $message" | |
parent.sendMsg(message) | |
} | |
void setThermostatFanMode(String mode) { | |
if (txtEnable) log.info "[IntesisHome.thermostat] setThermostatFanMode to: $mode" | |
//supportedThermostatFanModes : [auto, quiet, low, medium, high] | |
if(mode=='on' || mode=='circulate') { fanOn(); return } | |
Integer uid = COMMAND_MAP['fan_speed']['uid'] | |
Integer value = COMMAND_MAP['fan_speed'].values[mode] | |
String message = '{"command":"set","data":{"deviceId":' + (Long)state.deviceId + ',"uid":' + uid + ',"value":' + value + ',"seqNo":0}}' | |
if (logEnable) log.debug "[IntesisHome.thermostat] send message: $message" | |
parent.sendMsg(message) | |
} | |
void setSpeed(String fanspeed) { | |
if (txtEnable) log.info "[IntesisHome.thermostat] setSpeed to: $fanspeed" | |
if(!(Boolean)state.mpower) { | |
setPower('on') | |
setThermostatMode('fan') | |
} | |
switch(fanspeed) { | |
case ['low','low-medium']: | |
setThermostatFanMode("low") | |
break | |
case ['medium','medium-high']: | |
setThermostatFanMode("medium") | |
break | |
case 'high': | |
setThermostatFanMode("high") | |
break | |
case 'on': | |
setThermostatFanMode("on") | |
break | |
case ['auto','off']: | |
setThermostatFanMode("auto") | |
break | |
default: | |
log.warn "setSpeed: unknown speed" | |
} | |
} | |
void setPower(String mode) { | |
if (txtEnable) log.info "[IntesisHome.thermostat] setPower to: $mode" | |
//supports : [off, on] | |
Integer uid = COMMAND_MAP['power']['uid'] | |
Integer value = COMMAND_MAP['power'].values[mode] | |
String message = '{"command":"set","data":{"deviceId":' + (Long)state.deviceId + ',"uid":' + uid + ',"value":' + value + ',"seqNo":0}}' | |
if (logEnable) log.debug "[IntesisHome.thermostat] send message: $message" | |
parent.sendMsg(message) | |
} | |
/* thermostat mode commands */ | |
void cool() { | |
if(!(Boolean)state.mpower) setPower('on') | |
setThermostatMode('cool') | |
} | |
void heat() { | |
if(!(Boolean)state.mpower) setPower('on') | |
setThermostatMode('heat') | |
} | |
void auto() { | |
if(!(Boolean)state.mpower) setPower('on') | |
setThermostatMode('auto') | |
} | |
void emergencyHeat() { | |
if(!(Boolean)state.mpower) setPower('on') | |
setThermostatMode('heat') | |
} | |
void dry() { // custom command | |
if(!(Boolean)state.mpower) setPower('on') | |
setThermostatMode('dry') | |
} | |
void on() { // custom command | |
setPower('on') | |
} | |
void off() { | |
setPower('off') | |
} | |
/* Fan commands */ | |
void fanOn() { | |
setThermostatFanMode('low') | |
} | |
void fanAuto() { | |
setThermostatFanMode('auto') | |
} | |
void fanCirculate() { | |
setThermostatFanMode('low') | |
} | |
def setValue() {} | |
void refresh() { | |
if (logEnable) log.debug "refresh" | |
parent.queuePollStatus() | |
} | |
void configure() { | |
if (txtEnable) log.debug "Configuring Reporting and Bindings." | |
initialize() | |
} | |
private String createLogString(String context, String message) { | |
return "[IntesisHome.thermostat." + context + "] " + message | |
} | |
private void error(String context, String text, Exception e) { | |
error(context, text, e, true) | |
} | |
private void error(String context, String text, Exception e, Boolean remote) { | |
log.error(createLogString(context, text), e) | |
} | |
private void debug(String context, String text) { | |
debug(context, text, true) | |
} | |
private void debug(String context, String text, Boolean remote) { | |
log.debug(createLogString(context, text)) | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment