mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-03-15 05:10:50 +00:00
Compare commits
42 Commits
PROD_2017.
...
test-10
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4a14421ef0 | ||
|
|
de1894bfbf | ||
|
|
d0f8ec87bd | ||
|
|
1383ab1e07 | ||
|
|
d4f21b95d7 | ||
|
|
be2e19e406 | ||
|
|
54da556c17 | ||
|
|
8611d2e2d2 | ||
|
|
c650047f31 | ||
|
|
41adc9777a | ||
|
|
f334f6505a | ||
|
|
ded7501b84 | ||
|
|
eb4d5dcfb8 | ||
|
|
6385443f20 | ||
|
|
fd1ad51880 | ||
|
|
550214ceb5 | ||
|
|
a36500a216 | ||
|
|
445c115c14 | ||
|
|
9900e532a4 | ||
|
|
7648fd4a17 | ||
|
|
9686f3770b | ||
|
|
de37d0c813 | ||
|
|
bd3367fe0e | ||
|
|
30511d74af | ||
|
|
f0f02a2c00 | ||
|
|
57d20e2fca | ||
|
|
3216f63cc0 | ||
|
|
7cbc2d1780 | ||
|
|
5ad20fbd2a | ||
|
|
52357e4c50 | ||
|
|
03a7991279 | ||
|
|
f969027191 | ||
|
|
5607a3e346 | ||
|
|
bd44027038 | ||
|
|
c028515fcd | ||
|
|
751c98d123 | ||
|
|
5b874e8f3a | ||
|
|
9e10405527 | ||
|
|
32ceaff54d | ||
|
|
5b1da30a47 | ||
|
|
0a82077b24 | ||
|
|
bbad6dfa7a |
@@ -0,0 +1,46 @@
|
|||||||
|
/**
|
||||||
|
* Stateless On/Off Button Tile
|
||||||
|
*
|
||||||
|
* Author: Ronald Gouldner
|
||||||
|
*
|
||||||
|
* Date: 2015-05-14
|
||||||
|
*/
|
||||||
|
metadata {
|
||||||
|
// Automatically generated. Make future change here.
|
||||||
|
definition (name: "Stateless On-Off Button Tile", namespace: "gouldner", author: "Ronald Gouldner") {
|
||||||
|
capability "Actuator"
|
||||||
|
capability "Switch"
|
||||||
|
capability "Sensor"
|
||||||
|
}
|
||||||
|
|
||||||
|
// simulator metadata
|
||||||
|
simulator {
|
||||||
|
}
|
||||||
|
|
||||||
|
// UI tile definitions
|
||||||
|
tiles {
|
||||||
|
standardTile("button", "device.switch", width: 2, height: 2, canChangeIcon: true) {
|
||||||
|
state "offReady", label: 'Off', action: "switch.on", icon: "st.switches.switch.off", backgroundColor: "#ffffff", nextState: "onReady"
|
||||||
|
state "onReady", label: 'On', action: "switch.off", icon: "st.switches.switch.on", backgroundColor: "#79b821", nextState: "offReady"
|
||||||
|
state "off", label: 'Off', action: "switch.on", icon: "st.switches.switch.off", backgroundColor: "#ffffff"
|
||||||
|
state "on", label: 'On', action: "switch.off", icon: "st.switches.switch.on", backgroundColor: "#79b821"
|
||||||
|
}
|
||||||
|
main "button"
|
||||||
|
details "button"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def parse(String description) {
|
||||||
|
}
|
||||||
|
|
||||||
|
def on() {
|
||||||
|
log.debug "Stateless On/Off Button Tile Virtual Switch ${device.name} turned on"
|
||||||
|
sendEvent(name: "switch", value: "on")
|
||||||
|
sendEvent(name: "switch", value: "onReady")
|
||||||
|
}
|
||||||
|
|
||||||
|
def off() {
|
||||||
|
log.debug "Stateless On/Off Button Tile Virtual Switch ${device.name} turned off"
|
||||||
|
sendEvent(name: "switch", value: "off")
|
||||||
|
sendEvent(name: "switch", value: "offReady")
|
||||||
|
}
|
||||||
@@ -81,51 +81,47 @@ metadata {
|
|||||||
// parse events into attributes
|
// parse events into attributes
|
||||||
def parse(String description) {
|
def parse(String description) {
|
||||||
log.debug "Parse description $description"
|
log.debug "Parse description $description"
|
||||||
def map = [:]
|
List result = []
|
||||||
if (description?.startsWith("read attr -")) {
|
def descMap = zigbee.parseDescriptionAsMap(description)
|
||||||
def descMap = parseDescriptionAsMap(description)
|
log.debug "Desc Map: $descMap"
|
||||||
log.debug "Desc Map: $descMap"
|
List attrData = [[cluster: descMap.cluster ,attrId: descMap.attrId, value: descMap.value]]
|
||||||
if (descMap.cluster == "0201" && descMap.attrId == "0000") {
|
descMap.additionalAttrs.each {
|
||||||
|
attrData << [cluster: descMap.cluster, attrId: it.attrId, value: it.value]
|
||||||
|
}
|
||||||
|
attrData.each {
|
||||||
|
def map = [:]
|
||||||
|
if (it.cluster == "0201" && it.attrId == "0000") {
|
||||||
log.debug "TEMP"
|
log.debug "TEMP"
|
||||||
map.name = "temperature"
|
map.name = "temperature"
|
||||||
map.value = getTemperature(descMap.value)
|
map.value = getTemperature(it.value)
|
||||||
map.unit = temperatureScale
|
map.unit = temperatureScale
|
||||||
} else if (descMap.cluster == "0201" && descMap.attrId == "0011") {
|
} else if (it.cluster == "0201" && it.attrId == "0011") {
|
||||||
log.debug "COOLING SETPOINT"
|
log.debug "COOLING SETPOINT"
|
||||||
map.name = "coolingSetpoint"
|
map.name = "coolingSetpoint"
|
||||||
map.value = getTemperature(descMap.value)
|
map.value = getTemperature(it.value)
|
||||||
map.unit = temperatureScale
|
map.unit = temperatureScale
|
||||||
} else if (descMap.cluster == "0201" && descMap.attrId == "0012") {
|
} else if (it.cluster == "0201" && it.attrId == "0012") {
|
||||||
log.debug "HEATING SETPOINT"
|
log.debug "HEATING SETPOINT"
|
||||||
map.name = "heatingSetpoint"
|
map.name = "heatingSetpoint"
|
||||||
map.value = getTemperature(descMap.value)
|
map.value = getTemperature(it.value)
|
||||||
map.unit = temperatureScale
|
map.unit = temperatureScale
|
||||||
} else if (descMap.cluster == "0201" && descMap.attrId == "001c") {
|
} else if (it.cluster == "0201" && it.attrId == "001c") {
|
||||||
log.debug "MODE"
|
log.debug "MODE"
|
||||||
map.name = "thermostatMode"
|
map.name = "thermostatMode"
|
||||||
map.value = getModeMap()[descMap.value]
|
map.value = getModeMap()[it.value]
|
||||||
} else if (descMap.cluster == "0202" && descMap.attrId == "0000") {
|
} else if (it.cluster == "0202" && it.attrId == "0000") {
|
||||||
log.debug "FAN MODE"
|
log.debug "FAN MODE"
|
||||||
map.name = "thermostatFanMode"
|
map.name = "thermostatFanMode"
|
||||||
map.value = getFanModeMap()[descMap.value]
|
map.value = getFanModeMap()[it.value]
|
||||||
}
|
}
|
||||||
|
if (map) {
|
||||||
|
result << createEvent(map)
|
||||||
|
}
|
||||||
|
log.debug "Parse returned $map"
|
||||||
}
|
}
|
||||||
|
|
||||||
def result = null
|
|
||||||
if (map) {
|
|
||||||
result = createEvent(map)
|
|
||||||
}
|
|
||||||
log.debug "Parse returned $map"
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
def parseDescriptionAsMap(description) {
|
|
||||||
(description - "read attr - ").split(",").inject([:]) { map, param ->
|
|
||||||
def nameAndValue = param.split(":")
|
|
||||||
map += [(nameAndValue[0].trim()):nameAndValue[1].trim()]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
def getModeMap() { [
|
def getModeMap() { [
|
||||||
"00":"off",
|
"00":"off",
|
||||||
"03":"cool",
|
"03":"cool",
|
||||||
|
|||||||
@@ -82,7 +82,7 @@ def on() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def setLevel(value) {
|
def setLevel(value) {
|
||||||
zigbee.setLevel(value) + ["delay 500"] + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
zigbee.setLevel(value) + zigbee.onOffRefresh() + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -22,9 +22,10 @@ metadata {
|
|||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
capability "Health Check"
|
capability "Health Check"
|
||||||
|
|
||||||
fingerprint mfr:"0063", prod:"4457", deviceJoinName: "Z-Wave Wall Dimmer"
|
fingerprint mfr:"0063", prod:"4457", deviceJoinName: "GE In-Wall Smart Dimmer "
|
||||||
fingerprint mfr:"0063", prod:"4944", deviceJoinName: "Z-Wave Wall Dimmer"
|
fingerprint mfr:"0063", prod:"4944", deviceJoinName: "GE In-Wall Smart Dimmer "
|
||||||
fingerprint mfr:"0063", prod:"5044", deviceJoinName: "Z-Wave Plug-In Dimmer"
|
fingerprint mfr:"0063", prod:"5044", deviceJoinName: "GE Plug-In Smart Dimmer "
|
||||||
|
fingerprint mfr:"0063", prod:"4944", model:"3034", deviceJoinName: "GE In-Wall Smart Fan Control"
|
||||||
}
|
}
|
||||||
|
|
||||||
simulator {
|
simulator {
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ metadata {
|
|||||||
capability "Refresh"
|
capability "Refresh"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
capability "Health Check"
|
capability "Health Check"
|
||||||
|
capability "Light"
|
||||||
|
|
||||||
command "setAdjustedColor"
|
command "setAdjustedColor"
|
||||||
command "reset"
|
command "reset"
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ metadata {
|
|||||||
capability "Refresh"
|
capability "Refresh"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
capability "Health Check"
|
capability "Health Check"
|
||||||
|
capability "Light"
|
||||||
|
|
||||||
command "setAdjustedColor"
|
command "setAdjustedColor"
|
||||||
command "reset"
|
command "reset"
|
||||||
|
|||||||
@@ -14,7 +14,8 @@ metadata {
|
|||||||
capability "Switch"
|
capability "Switch"
|
||||||
capability "Refresh"
|
capability "Refresh"
|
||||||
capability "Sensor"
|
capability "Sensor"
|
||||||
capability "Health Check"
|
capability "Health Check"
|
||||||
|
capability "Light"
|
||||||
|
|
||||||
command "refresh"
|
command "refresh"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ metadata {
|
|||||||
capability "Switch"
|
capability "Switch"
|
||||||
capability "Refresh"
|
capability "Refresh"
|
||||||
capability "Health Check"
|
capability "Health Check"
|
||||||
|
capability "Light"
|
||||||
|
|
||||||
command "refresh"
|
command "refresh"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
metadata {
|
metadata {
|
||||||
|
|
||||||
definition (name: "Simulated Switch", namespace: "smartthings/testing", author: "bob") {
|
definition (name: "Simulated Switch", namespace: "smartthings/testing", author: "bob") {
|
||||||
capability "Switch"
|
capability "Switch"
|
||||||
capability "Relay Switch"
|
capability "Relay Switch"
|
||||||
@@ -39,9 +39,7 @@ metadata {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def parse(String description) {
|
def parse(description) {
|
||||||
def pair = description.split(":")
|
|
||||||
createEvent(name: pair[0].trim(), value: pair[1].trim())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
def on() {
|
def on() {
|
||||||
|
|||||||
@@ -86,7 +86,7 @@ def parse(String description) {
|
|||||||
def bodyString = msg.body
|
def bodyString = msg.body
|
||||||
if (bodyString) {
|
if (bodyString) {
|
||||||
unschedule("setOffline")
|
unschedule("setOffline")
|
||||||
def body = new XmlSlurper().parseText(bodyString)
|
def body = new XmlSlurper().parseText(bodyString.replaceAll("[^\\x20-\\x7e]", ""))
|
||||||
|
|
||||||
if (body?.property?.TimeSyncRequest?.text()) {
|
if (body?.property?.TimeSyncRequest?.text()) {
|
||||||
log.trace "Got TimeSyncRequest"
|
log.trace "Got TimeSyncRequest"
|
||||||
|
|||||||
@@ -78,7 +78,7 @@ def parse(String description) {
|
|||||||
def bodyString = msg.body
|
def bodyString = msg.body
|
||||||
if (bodyString) {
|
if (bodyString) {
|
||||||
unschedule("setOffline")
|
unschedule("setOffline")
|
||||||
def body = new XmlSlurper().parseText(bodyString)
|
def body = new XmlSlurper().parseText(bodyString.replaceAll("[^\\x20-\\x7e]", ""))
|
||||||
if (body?.property?.TimeSyncRequest?.text()) {
|
if (body?.property?.TimeSyncRequest?.text()) {
|
||||||
log.trace "Got TimeSyncRequest"
|
log.trace "Got TimeSyncRequest"
|
||||||
result << timeSyncResponse()
|
result << timeSyncResponse()
|
||||||
|
|||||||
@@ -84,7 +84,7 @@ def parse(String description) {
|
|||||||
def bodyString = msg.body
|
def bodyString = msg.body
|
||||||
if (bodyString) {
|
if (bodyString) {
|
||||||
unschedule("setOffline")
|
unschedule("setOffline")
|
||||||
def body = new XmlSlurper().parseText(bodyString)
|
def body = new XmlSlurper().parseText(bodyString.replaceAll("[^\\x20-\\x7e]", ""))
|
||||||
if (body?.property?.TimeSyncRequest?.text()) {
|
if (body?.property?.TimeSyncRequest?.text()) {
|
||||||
log.trace "Got TimeSyncRequest"
|
log.trace "Got TimeSyncRequest"
|
||||||
result << timeSyncResponse()
|
result << timeSyncResponse()
|
||||||
@@ -208,7 +208,7 @@ def subscribe(ip, port) {
|
|||||||
def existingIp = getDataValue("ip")
|
def existingIp = getDataValue("ip")
|
||||||
def existingPort = getDataValue("port")
|
def existingPort = getDataValue("port")
|
||||||
if (ip && ip != existingIp) {
|
if (ip && ip != existingIp) {
|
||||||
log.debug "Updating ip from $existingIp to $ip"
|
log.debug "Updating ip from $existingIp to $ip"
|
||||||
updateDataValue("ip", ip)
|
updateDataValue("ip", ip)
|
||||||
def ipvalue = convertHexToIP(getDataValue("ip"))
|
def ipvalue = convertHexToIP(getDataValue("ip"))
|
||||||
sendEvent(name: "currentIP", value: ipvalue, descriptionText: "IP changed to ${ipvalue}")
|
sendEvent(name: "currentIP", value: ipvalue, descriptionText: "IP changed to ${ipvalue}")
|
||||||
@@ -291,4 +291,4 @@ User-Agent: CyberGarage-HTTP/1.0
|
|||||||
</u:GetBinaryState>
|
</u:GetBinaryState>
|
||||||
</s:Body>
|
</s:Body>
|
||||||
</s:Envelope>""", physicalgraph.device.Protocol.LAN)
|
</s:Envelope>""", physicalgraph.device.Protocol.LAN)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,6 +28,10 @@ metadata {
|
|||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B05", outClusters: "0019", manufacturer: "OSRAM SYLVANIA", model: "iQBR30", deviceJoinName: "Sylvania Ultra iQ"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B05", outClusters: "0019", manufacturer: "OSRAM SYLVANIA", model: "iQBR30", deviceJoinName: "Sylvania Ultra iQ"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY PAR38 ON/OFF/DIM", deviceJoinName: "SYLVANIA Smart PAR38 Soft White"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY PAR38 ON/OFF/DIM", deviceJoinName: "SYLVANIA Smart PAR38 Soft White"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY BR ON/OFF/DIM", deviceJoinName: "SYLVANIA Smart BR30 Soft White"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0B04, FC0F", outClusters: "0019", manufacturer: "OSRAM", model: "LIGHTIFY BR ON/OFF/DIM", deviceJoinName: "SYLVANIA Smart BR30 Soft White"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008, 0702, 0B05", outClusters: "0019", manufacturer: "sengled", model: "E11-G13", deviceJoinName: "Sengled Element Classic"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008", outClusters: "0003, 0006, 0008, 0019, 0406", manufacturer: "Leviton", model: "DL6HD", deviceJoinName: "Leviton Dimmer Switch"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008", outClusters: "0003, 0006, 0008, 0019, 0406", manufacturer: "Leviton", model: "DL3HL", deviceJoinName: "Leviton Lumina RF Plug-In Dimmer"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006, 0008", outClusters: "0003, 0006, 0008, 0019, 0406", manufacturer: "Leviton", model: "DL1KD", deviceJoinName: "Leviton Lumina RF Dimmer Switch"
|
||||||
}
|
}
|
||||||
|
|
||||||
tiles(scale: 2) {
|
tiles(scale: 2) {
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ metadata {
|
|||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0006", outClusters: "0003, 0006, 0019, 0406", manufacturer: "Leviton", model: "ZSS-10", deviceJoinName: "Leviton Switch"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0006", outClusters: "0003, 0006, 0019, 0406", manufacturer: "Leviton", model: "ZSS-10", deviceJoinName: "Leviton Switch"
|
||||||
fingerprint profileId: "0104", inClusters: "0000, 0003, 0006", outClusters: "000A", manufacturer: "HAI", model: "65A21-1", deviceJoinName: "Leviton Wireless Load Control Module-30amp"
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0006", outClusters: "000A", manufacturer: "HAI", model: "65A21-1", deviceJoinName: "Leviton Wireless Load Control Module-30amp"
|
||||||
|
fingerprint profileId: "0104", inClusters: "0000, 0003, 0004, 0005, 0006", outClusters: "0003, 0006, 0008, 0019, 0406", manufacturer: "Leviton", model: "DL15A", deviceJoinName: "Leviton Lumina RF Plug-In Appliance Module"
|
||||||
}
|
}
|
||||||
|
|
||||||
// simulator metadata
|
// simulator metadata
|
||||||
|
|||||||
@@ -89,7 +89,7 @@ def on() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def setLevel(value) {
|
def setLevel(value) {
|
||||||
zigbee.setLevel(value) + ["delay 1500"] + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
zigbee.setLevel(value) + zigbee.onOffRefresh() + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
||||||
}
|
}
|
||||||
|
|
||||||
def refresh() {
|
def refresh() {
|
||||||
|
|||||||
@@ -115,7 +115,7 @@ def refreshAttributes() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def setLevel(value) {
|
def setLevel(value) {
|
||||||
zigbee.setLevel(value) + ["delay 1500"] + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
zigbee.setLevel(value) + zigbee.onOffRefresh() + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
||||||
}
|
}
|
||||||
|
|
||||||
def setColor(value){
|
def setColor(value){
|
||||||
|
|||||||
@@ -135,7 +135,7 @@ def setColorTemperature(value) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def setLevel(value) {
|
def setLevel(value) {
|
||||||
zigbee.setLevel(value) + ["delay 1500"] + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
zigbee.setLevel(value) + zigbee.onOffRefresh() + zigbee.levelRefresh() //adding refresh because of ZLL bulb not conforming to send-me-a-report
|
||||||
}
|
}
|
||||||
|
|
||||||
def setColor(value){
|
def setColor(value){
|
||||||
|
|||||||
@@ -90,7 +90,7 @@ def on() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def setLevel(value) {
|
def setLevel(value) {
|
||||||
zigbee.setLevel(value) + ["delay 1500"] + zigbee.levelRefresh()
|
zigbee.setLevel(value) + zigbee.onOffRefresh() + zigbee.levelRefresh()
|
||||||
}
|
}
|
||||||
|
|
||||||
def refresh() {
|
def refresh() {
|
||||||
|
|||||||
@@ -24,6 +24,10 @@ metadata {
|
|||||||
fingerprint inClusters: "0x26", deviceJoinName: "Z-Wave Dimmer"
|
fingerprint inClusters: "0x26", deviceJoinName: "Z-Wave Dimmer"
|
||||||
fingerprint mfr:"001D", prod:"1902", deviceJoinName: "Z-Wave Dimmer"
|
fingerprint mfr:"001D", prod:"1902", deviceJoinName: "Z-Wave Dimmer"
|
||||||
fingerprint mfr:"001D", prod:"1B03", model:"0334", deviceJoinName: "Leviton Universal Dimmer"
|
fingerprint mfr:"001D", prod:"1B03", model:"0334", deviceJoinName: "Leviton Universal Dimmer"
|
||||||
|
fingerprint mfr:"011A", prod:"0102", model:"0201", deviceJoinName: "Enerwave In-Wall Dimmer"
|
||||||
|
fingerprint mfr:"001D", prod:"1001", model:"0334", deviceJoinName: "Leviton 3-Speed Fan Controller"
|
||||||
|
fingerprint mfr:"001D", prod:"0602", model:"0334", deviceJoinName: "Leviton Magnetic Low Voltage Dimmer"
|
||||||
|
fingerprint mfr:"001D", prod:"0401", model:"0334", deviceJoinName: "Leviton 600W Incandescent Dimmer"
|
||||||
}
|
}
|
||||||
|
|
||||||
simulator {
|
simulator {
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ metadata {
|
|||||||
fingerprint mfr: "0060", prod: "0001", model: "0002", deviceJoinName: "Everspring Motion Sensor" // Everspring SP814
|
fingerprint mfr: "0060", prod: "0001", model: "0002", deviceJoinName: "Everspring Motion Sensor" // Everspring SP814
|
||||||
fingerprint mfr: "0060", prod: "0001", model: "0003", deviceJoinName: "Everspring Motion Sensor" // Everspring HSP02
|
fingerprint mfr: "0060", prod: "0001", model: "0003", deviceJoinName: "Everspring Motion Sensor" // Everspring HSP02
|
||||||
fingerprint mfr: "011A", prod: "0601", model: "0901", deviceJoinName: "Enerwave Motion Sensor" // Enerwave ZWN-BPC
|
fingerprint mfr: "011A", prod: "0601", model: "0901", deviceJoinName: "Enerwave Motion Sensor" // Enerwave ZWN-BPC
|
||||||
|
fingerprint mfr: "0063", prod: "4953", model: "3133", deviceJoinName: "GE Smart Motion Sensor"
|
||||||
}
|
}
|
||||||
|
|
||||||
simulator {
|
simulator {
|
||||||
|
|||||||
@@ -25,6 +25,9 @@ metadata {
|
|||||||
fingerprint mfr:"0063", prod:"4F50", model:"3031", deviceJoinName: "GE Plug-in Outdoor Switch"
|
fingerprint mfr:"0063", prod:"4F50", model:"3031", deviceJoinName: "GE Plug-in Outdoor Switch"
|
||||||
fingerprint mfr:"001D", prod:"1D04", model:"0334", deviceJoinName: "Leviton Outlet"
|
fingerprint mfr:"001D", prod:"1D04", model:"0334", deviceJoinName: "Leviton Outlet"
|
||||||
fingerprint mfr:"001D", prod:"1C02", model:"0334", deviceJoinName: "Leviton Switch"
|
fingerprint mfr:"001D", prod:"1C02", model:"0334", deviceJoinName: "Leviton Switch"
|
||||||
|
fingerprint mfr:"001D", prod:"0301", model:"0334", deviceJoinName: "Leviton 15A Switch"
|
||||||
|
fingerprint mfr:"011A", prod:"0101", model:"0102", deviceJoinName: "Enerwave On/Off Switch"
|
||||||
|
fingerprint mfr:"011A", prod:"0101", model:"0603", deviceJoinName: "Enerwave Duplex Receptacle"
|
||||||
}
|
}
|
||||||
|
|
||||||
// simulator metadata
|
// simulator metadata
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ def authPage() {
|
|||||||
return dynamicPage(name: "Credentials", title: "Authorize Connection", nextPage:"listDevices", uninstall: uninstallAllowed, install:false) {
|
return dynamicPage(name: "Credentials", title: "Authorize Connection", nextPage:"listDevices", uninstall: uninstallAllowed, install:false) {
|
||||||
section() {
|
section() {
|
||||||
paragraph "Tap below to log in to the netatmo and authorize SmartThings access."
|
paragraph "Tap below to log in to the netatmo and authorize SmartThings access."
|
||||||
href url:redirectUrl, style:"embedded", required:false, title:"Connect to ${getVendorName()}:", description:description
|
href url:redirectUrl, style:"embedded", required:false, title:"Connect to ${getVendorName()}", description:description
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -146,19 +146,24 @@ def callback() {
|
|||||||
|
|
||||||
// log.debug "PARAMS: ${params}"
|
// log.debug "PARAMS: ${params}"
|
||||||
|
|
||||||
httpPost(params) { resp ->
|
try {
|
||||||
|
httpPost(params) { resp ->
|
||||||
|
|
||||||
def slurper = new JsonSlurper()
|
def slurper = new JsonSlurper()
|
||||||
|
|
||||||
resp.data.each { key, value ->
|
resp.data.each { key, value ->
|
||||||
def data = slurper.parseText(key)
|
def data = slurper.parseText(key)
|
||||||
|
log.debug "Data: $data"
|
||||||
state.refreshToken = data.refresh_token
|
state.refreshToken = data.refresh_token
|
||||||
state.authToken = data.access_token
|
state.authToken = data.access_token
|
||||||
state.tokenExpires = now() + (data.expires_in * 1000)
|
//state.accessToken = data.access_token
|
||||||
// log.debug "swapped token: $resp.data"
|
state.tokenExpires = now() + (data.expires_in * 1000)
|
||||||
}
|
// log.debug "swapped token: $resp.data"
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.debug "callback: Call failed $e"
|
||||||
|
}
|
||||||
|
|
||||||
// Handle success and failure here, and render stuff accordingly
|
// Handle success and failure here, and render stuff accordingly
|
||||||
if (state.authToken) {
|
if (state.authToken) {
|
||||||
@@ -387,18 +392,18 @@ def getDeviceList() {
|
|||||||
state.deviceDetail = [:]
|
state.deviceDetail = [:]
|
||||||
state.deviceState = [:]
|
state.deviceState = [:]
|
||||||
|
|
||||||
apiGet("/api/devicelist") { response ->
|
apiGet("/api/getstationsdata") { response ->
|
||||||
response.data.body.devices.each { value ->
|
response.data.body.devices.each { value ->
|
||||||
def key = value._id
|
def key = value._id
|
||||||
deviceList[key] = "${value.station_name}: ${value.module_name}"
|
deviceList[key] = "${value.station_name}: ${value.module_name}"
|
||||||
state.deviceDetail[key] = value
|
state.deviceDetail[key] = value
|
||||||
state.deviceState[key] = value.dashboard_data
|
state.deviceState[key] = value.dashboard_data
|
||||||
}
|
value.modules.each { value2 ->
|
||||||
response.data.body.modules.each { value ->
|
def key2 = value2._id
|
||||||
def key = value._id
|
deviceList[key2] = "${value.station_name}: ${value2.module_name}"
|
||||||
deviceList[key] = "${state.deviceDetail[value.main_device].station_name}: ${value.module_name}"
|
state.deviceDetail[key2] = value2
|
||||||
state.deviceDetail[key] = value
|
state.deviceState[key2] = value2.dashboard_data
|
||||||
state.deviceState[key] = value.dashboard_data
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -448,6 +453,7 @@ def listDevices() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def apiGet(String path, Map query, Closure callback) {
|
def apiGet(String path, Map query, Closure callback) {
|
||||||
|
|
||||||
if(now() >= state.tokenExpires) {
|
if(now() >= state.tokenExpires) {
|
||||||
refreshToken();
|
refreshToken();
|
||||||
}
|
}
|
||||||
@@ -467,12 +473,16 @@ def apiGet(String path, Map query, Closure callback) {
|
|||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
// This is most likely due to an invalid token. Try to refresh it and try again.
|
// This is most likely due to an invalid token. Try to refresh it and try again.
|
||||||
log.debug "apiGet: Call failed $e"
|
log.debug "apiGet: Call failed $e"
|
||||||
if(refreshToken()) {
|
if(refreshToken()) {
|
||||||
log.debug "apiGet: Trying again after refreshing token"
|
log.debug "apiGet: Trying again after refreshing token"
|
||||||
httpGet(params) { response ->
|
try {
|
||||||
callback.call(response)
|
httpGet(params) { response ->
|
||||||
}
|
callback.call(response)
|
||||||
}
|
}
|
||||||
|
} catch (Exception f) {
|
||||||
|
log.debug "apiGet: Call failed $f"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -561,4 +571,4 @@ private Boolean hasAllHubsOver(String desiredFirmware) {
|
|||||||
|
|
||||||
private List getRealHubFirmwareVersions() {
|
private List getRealHubFirmwareVersions() {
|
||||||
return location.hubs*.firmwareVersionString.findAll { it }
|
return location.hubs*.firmwareVersionString.findAll { it }
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,461 @@
|
|||||||
|
/**
|
||||||
|
* OpenT2T SmartApp Test
|
||||||
|
*
|
||||||
|
* Copyright 2016 OpenT2T
|
||||||
|
*
|
||||||
|
* 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: "OpenT2T SmartApp Test",
|
||||||
|
namespace: "opent2t",
|
||||||
|
author: "OpenT2T",
|
||||||
|
description: "Test app to test end to end SmartThings scenarios via OpenT2T",
|
||||||
|
category: "SmartThings Labs",
|
||||||
|
iconUrl: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience.png",
|
||||||
|
iconX2Url: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience@2x.png",
|
||||||
|
iconX3Url: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience@2x.png")
|
||||||
|
|
||||||
|
/** --------------------+---------------+-----------------------+------------------------------------
|
||||||
|
* Device Type | Attribute Name| Commands | Attribute Values
|
||||||
|
* --------------------+---------------+-----------------------+------------------------------------
|
||||||
|
* switches | switch | on, off | on, off
|
||||||
|
* motionSensors | motion | | active, inactive
|
||||||
|
* contactSensors | contact | | open, closed
|
||||||
|
* presenceSensors | presence | | present, 'not present'
|
||||||
|
* temperatureSensors | temperature | | <numeric, F or C according to unit>
|
||||||
|
* accelerationSensors | acceleration | | active, inactive
|
||||||
|
* waterSensors | water | | wet, dry
|
||||||
|
* lightSensors | illuminance | | <numeric, lux>
|
||||||
|
* humiditySensors | humidity | | <numeric, percent>
|
||||||
|
* locks | lock | lock, unlock | locked, unlocked
|
||||||
|
* garageDoors | door | open, close | unknown, closed, open, closing, opening
|
||||||
|
* cameras | image | take | <String>
|
||||||
|
* thermostats | thermostat | setHeatingSetpoint, | temperature, heatingSetpoint, coolingSetpoint,
|
||||||
|
* | | setCoolingSetpoint, | thermostatSetpoint, thermostatMode,
|
||||||
|
* | | off, heat, cool, auto,| thermostatFanMode, thermostatOperatingState
|
||||||
|
* | | emergencyHeat, |
|
||||||
|
* | | setThermostatMode, |
|
||||||
|
* | | fanOn, fanAuto, |
|
||||||
|
* | | fanCirculate, |
|
||||||
|
* | | setThermostatFanMode |
|
||||||
|
* --------------------+---------------+-----------------------+------------------------------------
|
||||||
|
*/
|
||||||
|
|
||||||
|
//Device Inputs
|
||||||
|
preferences {
|
||||||
|
section("Allow <PLACEHOLDER: Your App Name> to control these things...") {
|
||||||
|
input "contactSensors", "capability.contactSensor", title: "Which Contact Sensors", multiple: true, required: false
|
||||||
|
input "garageDoors", "capability.garageDoorControl", title: "Which Garage Doors?", multiple: true, required: false
|
||||||
|
input "locks", "capability.lock", title: "Which Locks?", multiple: true, required: false
|
||||||
|
input "cameras", "capability.videoCapture", title: "Which Cameras?", multiple: true, required: false
|
||||||
|
input "motionSensors", "capability.motionSensor", title: "Which Motion Sensors?", multiple: true, required: false
|
||||||
|
input "presenceSensors", "capability.presenceSensor", title: "Which Presence Sensors", multiple: true, required: false
|
||||||
|
input "switches", "capability.switch", title: "Which Switches and Lights?", multiple: true, required: false
|
||||||
|
input "thermostats", "capability.thermostat", title: "Which Thermostat?", multiple: true, required: false
|
||||||
|
input "waterSensors", "capability.waterSensor", title: "Which Water Leak Sensors?", multiple: true, required: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def getInputs() {
|
||||||
|
def inputList = []
|
||||||
|
inputList += contactSensors ?: []
|
||||||
|
inputList += garageDoors ?: []
|
||||||
|
inputList += locks ?: []
|
||||||
|
inputList += cameras ?: []
|
||||||
|
inputList += motionSensors ?: []
|
||||||
|
inputList += presenceSensors ?: []
|
||||||
|
inputList += switches ?: []
|
||||||
|
inputList += thermostats ?: []
|
||||||
|
inputList += waterSensors ?: []
|
||||||
|
return inputList
|
||||||
|
}
|
||||||
|
|
||||||
|
//API external Endpoints
|
||||||
|
mappings {
|
||||||
|
path("/subscriptionURL/:url") {
|
||||||
|
action:
|
||||||
|
[
|
||||||
|
PUT: "updateEndpointURL"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
path("/connectionId/:connId") {
|
||||||
|
action:
|
||||||
|
[
|
||||||
|
PUT: "updateConnectionId"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
path("/devices") {
|
||||||
|
action:
|
||||||
|
[
|
||||||
|
GET: "getDevices"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
path("/devices/:id") {
|
||||||
|
action:
|
||||||
|
[
|
||||||
|
GET: "getDevice"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
path("/update/:id") {
|
||||||
|
action:
|
||||||
|
[
|
||||||
|
PUT: "updateDevice"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
path("/subscription/:id") {
|
||||||
|
action:
|
||||||
|
[
|
||||||
|
POST : "registerDeviceChange",
|
||||||
|
DELETE: "unregisterDeviceChange"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
log.debug "Installed with settings: ${settings}"
|
||||||
|
initialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
log.debug "Updated with settings: ${settings}"
|
||||||
|
unsubscribe()
|
||||||
|
registerSubscriptions()
|
||||||
|
}
|
||||||
|
|
||||||
|
def initialize() {
|
||||||
|
state.connectionId = ""
|
||||||
|
state.endpointURL = "https://ifs.windows-int.com/v1/cb/81C7E77B-EABC-488A-B2BF-FEC42F0DABD2/notify"
|
||||||
|
registerSubscriptions()
|
||||||
|
}
|
||||||
|
|
||||||
|
//Subscribe events for all devices
|
||||||
|
def registerSubscriptions() {
|
||||||
|
registerChangeHandler(inputs)
|
||||||
|
}
|
||||||
|
|
||||||
|
//Subscribe to events from a list of devices
|
||||||
|
def registerChangeHandler(myList) {
|
||||||
|
myList.each { myDevice ->
|
||||||
|
def theAtts = myDevice.supportedAttributes
|
||||||
|
theAtts.each { att ->
|
||||||
|
subscribe(myDevice, att.name, eventHandler)
|
||||||
|
log.info "Registering ${myDevice.displayName}.${att.name}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Endpoints function: Subscribe to events from a specific device
|
||||||
|
def registerDeviceChange() {
|
||||||
|
def myDevice = findDevice(params.id)
|
||||||
|
def theAtts = myDevice.supportedAttributes
|
||||||
|
try {
|
||||||
|
theAtts.each { att ->
|
||||||
|
subscribe(myDevice, att.name, eventHandler)
|
||||||
|
log.info "Registering ${myDevice.displayName}.${att.name}"
|
||||||
|
}
|
||||||
|
return ["succeed"]
|
||||||
|
} catch (e) {
|
||||||
|
httpError(500, "something went wrong: $e")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Endpoints function: Unsubscribe to events from a specific device
|
||||||
|
def unregisterDeviceChange() {
|
||||||
|
def myDevice = findDevice(params.id)
|
||||||
|
try {
|
||||||
|
unsubscribe(myDevice)
|
||||||
|
log.info "Unregistering ${myDevice.displayName}"
|
||||||
|
return ["succeed"]
|
||||||
|
} catch (e) {
|
||||||
|
httpError(500, "something went wrong: $e")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//When events are triggered, send HTTP post to web socket servers
|
||||||
|
def eventHandler(evt) {
|
||||||
|
def evt_device_id = evt.deviceId
|
||||||
|
def evt_device_value = evt.value
|
||||||
|
def evt_name = evt.name
|
||||||
|
def evt_device = evt.device
|
||||||
|
def evt_deviceType = getDeviceType(evt_device);
|
||||||
|
def params = [
|
||||||
|
uri : "${state.endpointURL}/${state.connectionId}",
|
||||||
|
body: [
|
||||||
|
name : evt_device.displayName,
|
||||||
|
id : evt_device.id,
|
||||||
|
deviceType : evt_deviceType,
|
||||||
|
manufacturer: evt_device.getManufacturerName(),
|
||||||
|
model : evt_device.getModelName(),
|
||||||
|
attributes : deviceAttributeList(evt_device)
|
||||||
|
]
|
||||||
|
]
|
||||||
|
try {
|
||||||
|
log.trace "POST URI: ${params.uri}"
|
||||||
|
log.trace "Payload: ${params.body}"
|
||||||
|
httpPostJson(params) { resp ->
|
||||||
|
resp.headers.each {
|
||||||
|
log.debug "${it.name} : ${it.value}"
|
||||||
|
}
|
||||||
|
log.trace "response status code: ${resp.status}"
|
||||||
|
log.trace "response data: ${resp.data}"
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
log.debug "something went wrong: $e"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Endpoints function: update subcription endpoint url [state.endpoint]
|
||||||
|
void updateEndpointURL() {
|
||||||
|
state.endpointURL = params.url
|
||||||
|
log.info "Updated EndpointURL to ${state.endpointURL}"
|
||||||
|
}
|
||||||
|
|
||||||
|
//Endpoints function: update global variable [state.connectionId]
|
||||||
|
void updateConnectionId() {
|
||||||
|
def connId = params.connId
|
||||||
|
state.connectionId = connId
|
||||||
|
log.info "Updated ConnectionID to ${state.connectionId}"
|
||||||
|
}
|
||||||
|
|
||||||
|
//Endpoints function: return all device data in json format
|
||||||
|
def getDevices() {
|
||||||
|
def deviceData = []
|
||||||
|
inputs?.each {
|
||||||
|
def deviceType = getDeviceType(it)
|
||||||
|
if (deviceType == "thermostat") {
|
||||||
|
deviceData << [name: it.displayName, id: it.id, deviceType: deviceType, manufacturer: it.getManufacturerName(), model: it.getModelName(), attributes: deviceAttributeList(it), locationMode: getLocationModeInfo()]
|
||||||
|
} else {
|
||||||
|
deviceData << [name: it.displayName, id: it.id, deviceType: deviceType, manufacturer: it.getManufacturerName(), model: it.getModelName(), attributes: deviceAttributeList(it)]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.debug "getDevices, return: ${deviceData}"
|
||||||
|
return deviceData
|
||||||
|
}
|
||||||
|
|
||||||
|
//Endpoints function: get device data
|
||||||
|
def getDevice() {
|
||||||
|
def it = findDevice(params.id)
|
||||||
|
def deviceType = getDeviceType(it)
|
||||||
|
def device
|
||||||
|
if (deviceType == "thermostat") {
|
||||||
|
device = [name: it.displayName, id: it.id, deviceType: deviceType, manufacturer: it.getManufacturerName(), model: it.getModelName(), attributes: deviceAttributeList(it), locationMode: getLocationModeInfo()]
|
||||||
|
} else {
|
||||||
|
device = [name: it.displayName, id: it.id, deviceType: deviceType, manufacturer: it.getManufacturerName(), model: it.getModelName(), attributes: deviceAttributeList(it)]
|
||||||
|
}
|
||||||
|
log.debug "getDevice, return: ${device}"
|
||||||
|
return device
|
||||||
|
}
|
||||||
|
|
||||||
|
//Endpoints function: update device data
|
||||||
|
void updateDevice() {
|
||||||
|
def device = findDevice(params.id)
|
||||||
|
request.JSON.each {
|
||||||
|
def command = it.key
|
||||||
|
def value = it.value
|
||||||
|
if (command) {
|
||||||
|
def commandList = mapDeviceCommands(command, value)
|
||||||
|
command = commandList[0]
|
||||||
|
value = commandList[1]
|
||||||
|
|
||||||
|
if (command == "setAwayMode") {
|
||||||
|
log.info "Setting away mode to ${value}"
|
||||||
|
if (location.modes?.find { it.name == value }) {
|
||||||
|
location.setMode(value)
|
||||||
|
}
|
||||||
|
} else if (command == "thermostatSetpoint") {
|
||||||
|
switch (device.currentThermostatMode) {
|
||||||
|
case "cool":
|
||||||
|
log.info "Update: ${device.displayName}, [${command}, ${value}]"
|
||||||
|
device.setCoolingSetpoint(value)
|
||||||
|
break
|
||||||
|
case "heat":
|
||||||
|
case "emergency heat":
|
||||||
|
log.info "Update: ${device.displayName}, [${command}, ${value}]"
|
||||||
|
device.setHeatingSetpoint(value)
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
httpError(501, "this mode: ${device.currentThermostatMode} does not allow changing thermostat setpoint.")
|
||||||
|
break
|
||||||
|
}
|
||||||
|
} else if (!device) {
|
||||||
|
log.error "updateDevice, Device not found"
|
||||||
|
httpError(404, "Device not found")
|
||||||
|
} else if (!device.hasCommand(command)) {
|
||||||
|
log.error "updateDevice, Device does not have the command"
|
||||||
|
httpError(404, "Device does not have such command")
|
||||||
|
} else {
|
||||||
|
if (command == "setColor") {
|
||||||
|
log.info "Update: ${device.displayName}, [${command}, ${value}]"
|
||||||
|
device."$command"(hex: value)
|
||||||
|
} else if (value.isNumber()) {
|
||||||
|
def intValue = value as Integer
|
||||||
|
log.info "Update: ${device.displayName}, [${command}, ${intValue}(int)]"
|
||||||
|
device."$command"(intValue)
|
||||||
|
} else if (value) {
|
||||||
|
log.info "Update: ${device.displayName}, [${command}, ${value}]"
|
||||||
|
device."$command"(value)
|
||||||
|
} else {
|
||||||
|
log.info "Update: ${device.displayName}, [${command}]"
|
||||||
|
device."$command"()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*** Private Functions ***/
|
||||||
|
|
||||||
|
//Return current location mode info
|
||||||
|
private getLocationModeInfo() {
|
||||||
|
return [mode: location.mode, supported: location.modes.name]
|
||||||
|
}
|
||||||
|
|
||||||
|
//Map each device to a type given it's capabilities
|
||||||
|
private getDeviceType(device) {
|
||||||
|
def deviceType
|
||||||
|
def caps = device.capabilities
|
||||||
|
log.debug "capabilities: [${device}, ${caps}]"
|
||||||
|
log.debug "supported commands: [${device}, ${device.supportedCommands}]"
|
||||||
|
caps.each {
|
||||||
|
switch (it.name.toLowerCase()) {
|
||||||
|
case "switch":
|
||||||
|
deviceType = "switch"
|
||||||
|
break
|
||||||
|
case "switch level":
|
||||||
|
deviceType = "light"
|
||||||
|
break
|
||||||
|
case "contact sensor":
|
||||||
|
deviceType = "contactSensor"
|
||||||
|
break
|
||||||
|
case "garageDoorControl":
|
||||||
|
deviceType = "garageDoor"
|
||||||
|
break
|
||||||
|
case "lock":
|
||||||
|
deviceType = "lock"
|
||||||
|
break
|
||||||
|
case "video camera":
|
||||||
|
deviceType = "camera"
|
||||||
|
break
|
||||||
|
case "motion sensor":
|
||||||
|
deviceType = "motionSensor"
|
||||||
|
break
|
||||||
|
case "presence sensor":
|
||||||
|
deviceType = "presenceSensor"
|
||||||
|
break
|
||||||
|
case "thermostat":
|
||||||
|
deviceType = "thermostat"
|
||||||
|
break
|
||||||
|
case "water sensor":
|
||||||
|
deviceType = "waterSensor"
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return deviceType
|
||||||
|
}
|
||||||
|
|
||||||
|
//Return a specific device give the device ID.
|
||||||
|
private findDevice(deviceId) {
|
||||||
|
return inputs?.find { it.id == deviceId }
|
||||||
|
}
|
||||||
|
|
||||||
|
//Return a list of device attributes
|
||||||
|
private deviceAttributeList(device) {
|
||||||
|
device.supportedAttributes.collectEntries { attribute ->
|
||||||
|
try {
|
||||||
|
[(attribute.name): device.currentValue(attribute.name)]
|
||||||
|
} catch (e) {
|
||||||
|
[(attribute.name): null]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Map device command and value.
|
||||||
|
//input command and value are from UWP,
|
||||||
|
//returns resultCommand and resultValue that corresponds with function and value in SmartApps
|
||||||
|
private mapDeviceCommands(command, value) {
|
||||||
|
log.debug "mapDeviceCommands: [${command}, ${value}]"
|
||||||
|
def resultCommand = command
|
||||||
|
def resultValue = value
|
||||||
|
switch (command) {
|
||||||
|
case "switch":
|
||||||
|
if (value == 1 || value == "1" || value == "on") {
|
||||||
|
resultCommand = "on"
|
||||||
|
resultValue = ""
|
||||||
|
} else if (value == 0 || value == "0" || value == "off") {
|
||||||
|
resultCommand = "off"
|
||||||
|
resultValue = ""
|
||||||
|
}
|
||||||
|
break
|
||||||
|
// light attributes
|
||||||
|
case "level":
|
||||||
|
resultCommand = "setLevel"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "hue":
|
||||||
|
resultCommand = "setHue"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "saturation":
|
||||||
|
resultCommand = "setSaturation"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "ct":
|
||||||
|
resultCommand = "setColorTemperature"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "color":
|
||||||
|
resultCommand = "setColor"
|
||||||
|
resultValue = value
|
||||||
|
// thermostat attributes
|
||||||
|
case "hvacMode":
|
||||||
|
resultCommand = "setThermostatMode"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "fanMode":
|
||||||
|
resultCommand = "setThermostatFanMode"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "awayMode":
|
||||||
|
resultCommand = "setAwayMode"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "coolingSetpoint":
|
||||||
|
resultCommand = "setCoolingSetpoint"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "heatingSetpoint":
|
||||||
|
resultCommand = "setHeatingSetpoint"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
case "thermostatSetpoint":
|
||||||
|
resultCommand = "thermostatSetpoint"
|
||||||
|
resultValue = value
|
||||||
|
break
|
||||||
|
// lock attributes
|
||||||
|
case "locked":
|
||||||
|
if (value == 1 || value == "1" || value == "lock") {
|
||||||
|
resultCommand = "lock"
|
||||||
|
resultValue = ""
|
||||||
|
} else if (value == 0 || value == "0" || value == "unlock") {
|
||||||
|
resultCommand = "unlock"
|
||||||
|
resultValue = ""
|
||||||
|
}
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
return [resultCommand, resultValue]
|
||||||
|
}
|
||||||
|
|
||||||
@@ -57,7 +57,7 @@ def authPage(){
|
|||||||
atomicState.accessToken = state.accessToken
|
atomicState.accessToken = state.accessToken
|
||||||
}
|
}
|
||||||
|
|
||||||
def redirectUrl = oauthInitUrl()
|
def redirectUrl = oauthInitUrl()
|
||||||
def uninstallAllowed = false
|
def uninstallAllowed = false
|
||||||
def oauthTokenProvided = false
|
def oauthTokenProvided = false
|
||||||
if(atomicState.authToken){
|
if(atomicState.authToken){
|
||||||
@@ -78,9 +78,9 @@ def authPage(){
|
|||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
return dynamicPage(name: "auth", title: "Step 1 of 2 - Completed", nextPage:"deviceList", uninstall:uninstallAllowed) {
|
return dynamicPage(name: "auth", title: "Step 1 of 2 - Completed", nextPage:"deviceList", uninstall:uninstallAllowed) {
|
||||||
section(){
|
section(){
|
||||||
paragraph "You are logged in to myplantlink.com, tap next to continue", image: iconUrl
|
paragraph "You are logged in to myplantlink.com, tap next to continue", image: iconUrl
|
||||||
href(url:redirectUrl, title:"Or", description:"tap to switch accounts")
|
href(url:redirectUrl, title:"Or", description:"tap to switch accounts")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -137,36 +137,44 @@ def dock_sensor(device_serial, expected_plant_name) {
|
|||||||
contentType: "application/json",
|
contentType: "application/json",
|
||||||
]
|
]
|
||||||
log.debug "Creating new plant on myplantlink.com - ${expected_plant_name}"
|
log.debug "Creating new plant on myplantlink.com - ${expected_plant_name}"
|
||||||
httpPost(docking_params) { docking_response ->
|
try {
|
||||||
if (parse_api_response(docking_response, "Docking a link")) {
|
httpPost(docking_params) { docking_response ->
|
||||||
if (docking_response.data.plants.size() == 0) {
|
if (parse_api_response(docking_response, "Docking a link")) {
|
||||||
log.debug "creating plant for - ${expected_plant_name}"
|
if (docking_response.data.plants.size() == 0) {
|
||||||
plant_post_body_map["name"] = expected_plant_name
|
log.debug "creating plant for - ${expected_plant_name}"
|
||||||
plant_post_body_map['links_key'] = [docking_response.data.key]
|
plant_post_body_map["name"] = expected_plant_name
|
||||||
def plant_post_body_json_builder = new JsonBuilder(plant_post_body_map)
|
plant_post_body_map['links_key'] = [docking_response.data.key]
|
||||||
plant_post_params["body"] = plant_post_body_json_builder.toString()
|
def plant_post_body_json_builder = new JsonBuilder(plant_post_body_map)
|
||||||
httpPost(plant_post_params) { plant_post_response ->
|
plant_post_params["body"] = plant_post_body_json_builder.toString()
|
||||||
if(parse_api_response(plant_post_response, 'creating plant')){
|
try {
|
||||||
def attached_map = atomicState.attached_sensors
|
httpPost(plant_post_params) { plant_post_response ->
|
||||||
attached_map[device_serial] = plant_post_response.data
|
if(parse_api_response(plant_post_response, 'creating plant')){
|
||||||
atomicState.attached_sensors = attached_map
|
def attached_map = atomicState.attached_sensors
|
||||||
|
attached_map[device_serial] = plant_post_response.data
|
||||||
|
atomicState.attached_sensors = attached_map
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception f) {
|
||||||
|
log.debug "call failed $f"
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
def plant = docking_response.data.plants[0]
|
||||||
|
def attached_map = atomicState.attached_sensors
|
||||||
|
attached_map[device_serial] = plant
|
||||||
|
atomicState.attached_sensors = attached_map
|
||||||
|
checkAndUpdatePlantIfNeeded(plant, expected_plant_name)
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
def plant = docking_response.data.plants[0]
|
|
||||||
def attached_map = atomicState.attached_sensors
|
|
||||||
attached_map[device_serial] = plant
|
|
||||||
atomicState.attached_sensors = attached_map
|
|
||||||
checkAndUpdatePlantIfNeeded(plant, expected_plant_name)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.debug "call failed $e"
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
def checkAndUpdatePlantIfNeeded(plant, expected_plant_name){
|
def checkAndUpdatePlantIfNeeded(plant, expected_plant_name){
|
||||||
def plant_put_params = [
|
def plant_put_params = [
|
||||||
uri : appSettings.https_plantLinkServer,
|
uri : appSettings.https_plantLinkServer,
|
||||||
headers : ["Content-Type": "application/json", "Authorization": "Bearer ${atomicState.authToken}"],
|
headers : ["Content-Type": "application/json", "Authorization": "Bearer ${atomicState.authToken}"],
|
||||||
contentType : "application/json"
|
contentType : "application/json"
|
||||||
]
|
]
|
||||||
@@ -174,12 +182,16 @@ def checkAndUpdatePlantIfNeeded(plant, expected_plant_name){
|
|||||||
log.debug "updating plant for - ${expected_plant_name}"
|
log.debug "updating plant for - ${expected_plant_name}"
|
||||||
plant_put_params["path"] = "/api/v1/plants/${plant.key}"
|
plant_put_params["path"] = "/api/v1/plants/${plant.key}"
|
||||||
def plant_put_body_map = [
|
def plant_put_body_map = [
|
||||||
name: expected_plant_name
|
name: expected_plant_name
|
||||||
]
|
]
|
||||||
def plant_put_body_json_builder = new JsonBuilder(plant_put_body_map)
|
def plant_put_body_json_builder = new JsonBuilder(plant_put_body_map)
|
||||||
plant_put_params["body"] = plant_put_body_json_builder.toString()
|
plant_put_params["body"] = plant_put_body_json_builder.toString()
|
||||||
httpPut(plant_put_params) { plant_put_response ->
|
try {
|
||||||
parse_api_response(plant_put_response, 'updating plant name')
|
httpPut(plant_put_params) { plant_put_response ->
|
||||||
|
parse_api_response(plant_put_response, 'updating plant name')
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.debug "call failed $e"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -198,25 +210,29 @@ def moistureHandler(event){
|
|||||||
contentType: "application/json",
|
contentType: "application/json",
|
||||||
body: event.value
|
body: event.value
|
||||||
]
|
]
|
||||||
httpPost(measurement_post_params) { measurement_post_response ->
|
try {
|
||||||
if (parse_api_response(measurement_post_response, 'creating moisture measurement') &&
|
httpPost(measurement_post_params) { measurement_post_response ->
|
||||||
measurement_post_response.data.size() >0){
|
if (parse_api_response(measurement_post_response, 'creating moisture measurement') &&
|
||||||
def measurement = measurement_post_response.data[0]
|
measurement_post_response.data.size() >0){
|
||||||
def plant = measurement.plant
|
def measurement = measurement_post_response.data[0]
|
||||||
log.debug plant
|
def plant = measurement.plant
|
||||||
checkAndUpdatePlantIfNeeded(plant, expected_plant_name)
|
log.debug plant
|
||||||
plantlinksensors.each{ sensor_device ->
|
checkAndUpdatePlantIfNeeded(plant, expected_plant_name)
|
||||||
if (sensor_device.id == event.deviceId){
|
plantlinksensors.each{ sensor_device ->
|
||||||
sensor_device.setStatusIcon(plant.status)
|
if (sensor_device.id == event.deviceId){
|
||||||
if (plant.last_measurements && plant.last_measurements[0].moisture){
|
sensor_device.setStatusIcon(plant.status)
|
||||||
sensor_device.setPlantFuelLevel(plant.last_measurements[0].moisture * 100 as int)
|
if (plant.last_measurements && plant.last_measurements[0].moisture){
|
||||||
}
|
sensor_device.setPlantFuelLevel(plant.last_measurements[0].moisture * 100 as int)
|
||||||
if (plant.last_measurements && plant.last_measurements[0].battery){
|
}
|
||||||
sensor_device.setBatteryLevel(plant.last_measurements[0].battery * 100 as int)
|
if (plant.last_measurements && plant.last_measurements[0].battery){
|
||||||
|
sensor_device.setBatteryLevel(plant.last_measurements[0].battery * 100 as int)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.debug "call failed $e"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -235,8 +251,12 @@ def batteryHandler(event){
|
|||||||
contentType: "application/json",
|
contentType: "application/json",
|
||||||
body: event.value
|
body: event.value
|
||||||
]
|
]
|
||||||
httpPost(measurement_post_params) { measurement_post_response ->
|
try {
|
||||||
parse_api_response(measurement_post_response, 'creating battery measurement')
|
httpPost(measurement_post_params) { measurement_post_response ->
|
||||||
|
parse_api_response(measurement_post_response, 'creating battery measurement')
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.debug "call failed $e"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -248,7 +268,7 @@ def getDeviceSerialFromEvent(event){
|
|||||||
}
|
}
|
||||||
|
|
||||||
def oauthInitUrl(){
|
def oauthInitUrl(){
|
||||||
atomicState.oauthInitState = UUID.randomUUID().toString()
|
atomicState.oauthInitState = UUID.randomUUID().toString()
|
||||||
def oauthParams = [
|
def oauthParams = [
|
||||||
response_type: "code",
|
response_type: "code",
|
||||||
client_id: appSettings.client_id,
|
client_id: appSettings.client_id,
|
||||||
@@ -275,8 +295,12 @@ def swapToken(){
|
|||||||
]
|
]
|
||||||
|
|
||||||
def jsonMap
|
def jsonMap
|
||||||
httpPost(postParams) { resp ->
|
try {
|
||||||
jsonMap = resp.data
|
httpPost(postParams) { resp ->
|
||||||
|
jsonMap = resp.data
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.debug "call failed $e"
|
||||||
}
|
}
|
||||||
|
|
||||||
atomicState.refreshToken = jsonMap.refresh_token
|
atomicState.refreshToken = jsonMap.refresh_token
|
||||||
@@ -287,33 +311,33 @@ def swapToken(){
|
|||||||
<head>
|
<head>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
<style>
|
<style>
|
||||||
.container {
|
.container {
|
||||||
padding:25px;
|
padding:25px;
|
||||||
}
|
}
|
||||||
.flex1 {
|
.flex1 {
|
||||||
width:33%;
|
width:33%;
|
||||||
float:left;
|
float:left;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
p {
|
p {
|
||||||
font-size: 2em;
|
font-size: 2em;
|
||||||
font-family: Verdana, Geneva, sans-serif;
|
font-family: Verdana, Geneva, sans-serif;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
color: #777;
|
color: #777;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<div class="flex1"><img src="https://dashboard.myplantlink.com/images/PLlogo.png" alt="PlantLink" height="75"/></div>
|
<div class="flex1"><img src="https://dashboard.myplantlink.com/images/PLlogo.png" alt="PlantLink" height="75"/></div>
|
||||||
<div class="flex1"><img src="https://s3.amazonaws.com/smartapp-icons/Partner/support/connected-device-icn%402x.png" alt="connected to" height="25" style="padding-top:25px;" /></div>
|
<div class="flex1"><img src="https://s3.amazonaws.com/smartapp-icons/Partner/support/connected-device-icn%402x.png" alt="connected to" height="25" style="padding-top:25px;" /></div>
|
||||||
<div class="flex1"><img src="https://s3.amazonaws.com/smartapp-icons/Partner/support/st-logo%402x.png" alt="SmartThings" height="75"/></div>
|
<div class="flex1"><img src="https://s3.amazonaws.com/smartapp-icons/Partner/support/st-logo%402x.png" alt="SmartThings" height="75"/></div>
|
||||||
<br clear="all">
|
<br clear="all">
|
||||||
</div>
|
</div>
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<p>Your PlantLink Account is now connected to SmartThings!</p>
|
<p>Your PlantLink Account is now connected to SmartThings!</p>
|
||||||
<p style="color:green;">Click <strong>Done</strong> at the top right to finish setup.</p>
|
<p style="color:green;">Click <strong>Done</strong> at the top right to finish setup.</p>
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -172,18 +172,34 @@ def bulbDiscovery() {
|
|||||||
if (existingLightsDescription.isEmpty()) {
|
if (existingLightsDescription.isEmpty()) {
|
||||||
existingLightsDescription += it.value
|
existingLightsDescription += it.value
|
||||||
} else {
|
} else {
|
||||||
existingLightsDescription += ", ${it.value}"
|
existingLightsDescription += ", ${it.value}"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return dynamicPage(name:"bulbDiscovery", title:"Light Discovery Started!", nextPage:"", refreshInterval:refreshInterval, install:true, uninstall: true) {
|
if (bulbRefreshCount > 200 && numFound == 0) {
|
||||||
section("Please wait while we discover your Hue Lights. Discovery can take five minutes or more, so sit back and relax! Select your device below once discovered.") {
|
// Time out to avoid endless discovery
|
||||||
input "selectedBulbs", "enum", required:false, title:"Select Hue Lights to add (${numFound} found)", multiple:true, submitOnChange: true, options:newLights
|
state.inBulbDiscovery = false
|
||||||
paragraph title: "Previously added Hue Lights (${existingLights.size()} added)", existingLightsDescription
|
bulbRefreshCount = 0
|
||||||
|
return dynamicPage(name:"bulbDiscovery", title:"Light Discovery Failed!", nextPage:"", refreshInterval:0, install:true, uninstall: true) {
|
||||||
|
section("Failed to discover any lights, please try again later. Click Done to exit.") {
|
||||||
|
//input "selectedBulbs", "enum", required:false, title:"Select Hue Lights to add (${numFound} found)", multiple:true, submitOnChange: true, options:newLights
|
||||||
|
paragraph title: "Previously added Hue Lights (${existingLights.size()} added)", existingLightsDescription
|
||||||
|
}
|
||||||
|
section {
|
||||||
|
href "bridgeDiscovery", title: title, description: "", state: selectedHue ? "complete" : "incomplete", params: [override: true]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
section {
|
|
||||||
href "bridgeDiscovery", title: title, description: "", state: selectedHue ? "complete" : "incomplete", params: [override: true]
|
} else {
|
||||||
|
return dynamicPage(name:"bulbDiscovery", title:"Light Discovery Started!", nextPage:"", refreshInterval:refreshInterval, install:true, uninstall: true) {
|
||||||
|
section("Please wait while we discover your Hue Lights. Discovery can take five minutes or more, so sit back and relax! Select your device below once discovered.") {
|
||||||
|
input "selectedBulbs", "enum", required:false, title:"Select Hue Lights to add (${numFound} found)", multiple:true, submitOnChange: true, options:newLights
|
||||||
|
paragraph title: "Previously added Hue Lights (${existingLights.size()} added)", existingLightsDescription
|
||||||
|
}
|
||||||
|
section {
|
||||||
|
href "bridgeDiscovery", title: title, description: "", state: selectedHue ? "complete" : "incomplete", params: [override: true]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -407,7 +423,7 @@ def addBridge() {
|
|||||||
if(vbridge) {
|
if(vbridge) {
|
||||||
def d = getChildDevice(selectedHue)
|
def d = getChildDevice(selectedHue)
|
||||||
if(!d) {
|
if(!d) {
|
||||||
// compatibility with old devices
|
// compatibility with old devices
|
||||||
def newbridge = true
|
def newbridge = true
|
||||||
childDevices.each {
|
childDevices.each {
|
||||||
if (it.getDeviceDataByName("mac")) {
|
if (it.getDeviceDataByName("mac")) {
|
||||||
@@ -593,7 +609,7 @@ def locationHandler(evt) {
|
|||||||
log.trace "Location: $description"
|
log.trace "Location: $description"
|
||||||
|
|
||||||
def hub = evt?.hubId
|
def hub = evt?.hubId
|
||||||
def parsedEvent = parseLanMessage(description)
|
def parsedEvent = parseLanMessage(description)
|
||||||
parsedEvent << ["hub":hub]
|
parsedEvent << ["hub":hub]
|
||||||
|
|
||||||
if (parsedEvent?.ssdpTerm?.contains("urn:schemas-upnp-org:device:basic:1")) {
|
if (parsedEvent?.ssdpTerm?.contains("urn:schemas-upnp-org:device:basic:1")) {
|
||||||
@@ -819,8 +835,7 @@ def parse(childDevice, description) {
|
|||||||
try {
|
try {
|
||||||
body = new groovy.json.JsonSlurper().parseText(bodyString)
|
body = new groovy.json.JsonSlurper().parseText(bodyString)
|
||||||
} catch (all) {
|
} catch (all) {
|
||||||
log.warn "Parsing Body failed - trying again..."
|
log.warn "Parsing Body failed"
|
||||||
poll()
|
|
||||||
}
|
}
|
||||||
if (body instanceof java.util.Map) {
|
if (body instanceof java.util.Map) {
|
||||||
// get (poll) reponse
|
// get (poll) reponse
|
||||||
@@ -844,7 +859,7 @@ private sendColorEvents(device, xy, hue, sat, ct, colormode = null) {
|
|||||||
|
|
||||||
def events = [:]
|
def events = [:]
|
||||||
// For now, only care about changing color temperature if requested by user
|
// For now, only care about changing color temperature if requested by user
|
||||||
if (ct != null && (colormode == "ct" || (xy == null && hue == null && sat == null))) {
|
if (ct != null && ct != 0 && (colormode == "ct" || (xy == null && hue == null && sat == null))) {
|
||||||
// for some reason setting Hue to their specified minimum off 153 yields 154, dealt with below
|
// for some reason setting Hue to their specified minimum off 153 yields 154, dealt with below
|
||||||
// 153 (6500K) to 500 (2000K)
|
// 153 (6500K) to 500 (2000K)
|
||||||
def temp = (ct == 154) ? 6500 : Math.round(1000000 / ct)
|
def temp = (ct == 154) ? 6500 : Math.round(1000000 / ct)
|
||||||
@@ -1252,7 +1267,7 @@ private getBridgeIP() {
|
|||||||
if (d) {
|
if (d) {
|
||||||
if (d.getDeviceDataByName("networkAddress"))
|
if (d.getDeviceDataByName("networkAddress"))
|
||||||
host = d.getDeviceDataByName("networkAddress")
|
host = d.getDeviceDataByName("networkAddress")
|
||||||
else
|
else
|
||||||
host = d.latestState('networkAddress').stringValue
|
host = d.latestState('networkAddress').stringValue
|
||||||
}
|
}
|
||||||
if (host == null || host == "") {
|
if (host == null || host == "") {
|
||||||
|
|||||||
Reference in New Issue
Block a user