Import zwave-mettering-switch
This commit is contained in:
commit
3acbce12b1
335
devicetypes/zwave-metering-switch/zwave-metering-switch.groovy
Normal file
335
devicetypes/zwave-metering-switch/zwave-metering-switch.groovy
Normal file
@ -0,0 +1,335 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2015 SmartThings
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
metadata {
|
||||||
|
definition (name: "Z-Wave Metering Switch", namespace: "smartthings", author: "SmartThings", ocfDeviceType: "oic.d.switch", runLocally: true, minHubCoreVersion: '000.017.0012', executeCommandsLocally: false, genericHandler: "Z-Wave") {
|
||||||
|
capability "Energy Meter"
|
||||||
|
capability "Actuator"
|
||||||
|
capability "Switch"
|
||||||
|
capability "Power Meter"
|
||||||
|
capability "Refresh"
|
||||||
|
capability "Configuration"
|
||||||
|
capability "Sensor"
|
||||||
|
capability "Light"
|
||||||
|
capability "Health Check"
|
||||||
|
|
||||||
|
command "reset"
|
||||||
|
|
||||||
|
fingerprint inClusters: "0x25,0x32", deviceJoinName: "Switch"
|
||||||
|
fingerprint mfr: "0086", prod: "0003", model: "0012", deviceJoinName: "Aeotec Switch" //Aeotec Micro Smart Switch
|
||||||
|
fingerprint mfr: "021F", prod: "0003", model: "0087", deviceJoinName: "Dome Outlet", ocfDeviceType: "oic.d.smartplug" //Dome On/Off Plug-in Switch
|
||||||
|
fingerprint mfr: "0086", prod: "0103", model: "0060", deviceJoinName: "Aeotec Outlet", ocfDeviceType: "oic.d.smartplug" //US //Aeotec Smart Switch 6
|
||||||
|
fingerprint mfr: "0086", prod: "0003", model: "0060", deviceJoinName: "Aeotec Outlet", ocfDeviceType: "oic.d.smartplug" //EU //Aeotec Smart Switch 6
|
||||||
|
fingerprint mfr: "0086", prod: "0203", model: "0060", deviceJoinName: "Aeotec Outlet", ocfDeviceType: "oic.d.smartplug" //AU //Aeotec Smart Switch 6
|
||||||
|
fingerprint mfr: "0086", prod: "0103", model: "0074", deviceJoinName: "Aeotec Switch" //Aeotec Nano Switch
|
||||||
|
fingerprint mfr: "0086", prod: "0003", model: "0074", deviceJoinName: "Aeotec Switch" //Aeotec Nano Switch
|
||||||
|
fingerprint mfr: "0086", prod: "0203", model: "0074", deviceJoinName: "Aeotec Switch" //AU //Aeotec Nano Switch
|
||||||
|
fingerprint mfr: "014F", prod: "574F", model: "3535", deviceJoinName: "GoControl Outlet", ocfDeviceType: "oic.d.smartplug" //GoControl Wall-Mounted Outlet
|
||||||
|
fingerprint mfr: "014F", prod: "5053", model: "3531", deviceJoinName: "GoControl Outlet", ocfDeviceType: "oic.d.smartplug" //GoControl Plug-in Switch
|
||||||
|
fingerprint mfr: "0063", prod: "4F44", model: "3031", deviceJoinName: "GE Switch" //GE Direct-Wire Outdoor Switch
|
||||||
|
fingerprint mfr: "0258", prod: "0003", model: "0087", deviceJoinName: "NEO Coolcam Outlet", ocfDeviceType: "oic.d.smartplug" //NEO Coolcam Power plug
|
||||||
|
fingerprint mfr: "010F", prod: "0602", model: "1001", deviceJoinName: "Fibaro Outlet", ocfDeviceType: "oic.d.smartplug" // EU //Fibaro Wall Plug ZW5
|
||||||
|
fingerprint mfr: "010F", prod: "1801", model: "1000", deviceJoinName: "Fibaro Outlet", ocfDeviceType: "oic.d.smartplug"// UK //Fibaro Wall Plug ZW5
|
||||||
|
fingerprint mfr: "0086", prod: "0003", model: "004E", deviceJoinName: "Aeotec Switch" //EU //Aeotec Heavy Duty Smart Switch
|
||||||
|
fingerprint mfr: "0086", prod: "0103", model: "004E", deviceJoinName: "Aeotec Switch" //US //Aeotec Heavy Duty Smart Switch
|
||||||
|
//zw:L type:1001 mfr:0258 prod:0003 model:1087 ver:3.94 zwv:4.05 lib:03 cc:5E,72,86,85,59,5A,73,70,25,27,71,32,20 role:05 ff:8700 ui:8700
|
||||||
|
fingerprint mfr: "0258", prod: "0003", model: "1087", deviceJoinName: "NEO Coolcam Outlet", ocfDeviceType: "oic.d.smartplug" //EU //NEO Coolcam Power Plug
|
||||||
|
fingerprint mfr: "027A", prod: "0101", model: "000D", deviceJoinName: "Zooz Switch" //Zooz Power Switch
|
||||||
|
fingerprint mfr: "0159", prod: "0002", model: "0054", deviceJoinName: "Qubino Outlet", ocfDeviceType: "oic.d.smartplug" //Qubino Smart Plug
|
||||||
|
fingerprint mfr: "0371", prod: "0003", model: "00AF", deviceJoinName: "Aeotec Outlet", ocfDeviceType: "oic.d.smartplug" //EU //Aeotec Smart Switch 7
|
||||||
|
fingerprint mfr: "0371", prod: "0103", model: "00AF", deviceJoinName: "Aeotec Outlet", ocfDeviceType: "oic.d.smartplug" //US //Aeotec Smart Switch 7
|
||||||
|
fingerprint mfr: "0060", prod: "0004", model: "000B", deviceJoinName: "Everspring Outlet", ocfDeviceType: "oic.d.smartplug" //US //Everspring Smart Plug
|
||||||
|
fingerprint mfr: "031E", prod: "0002", model: "0001", deviceJoinName: "Inovelli Switch" //US //Inovelli Switch Red Series
|
||||||
|
fingerprint mfr: "0154", prod: "0003", model: "000A", deviceJoinName: "POPP Outlet", ocfDeviceType: "oic.d.smartplug" //EU //POPP Smart Outdoor Plug
|
||||||
|
fingerprint mfr: "010F", prod: "1F01", model: "1000", deviceJoinName: "Fibaro Outlet", ocfDeviceType: "oic.d.smartplug" //EU //Fibaro walli Outlet //Fibaro Outlet
|
||||||
|
}
|
||||||
|
|
||||||
|
// simulator metadata
|
||||||
|
simulator {
|
||||||
|
status "on": "command: 2003, payload: FF"
|
||||||
|
status "off": "command: 2003, payload: 00"
|
||||||
|
|
||||||
|
for (int i = 0; i <= 10000; i += 1000) {
|
||||||
|
status "power ${i} W": new physicalgraph.zwave.Zwave().meterV1.meterReport(
|
||||||
|
scaledMeterValue: i, precision: 3, meterType: 4, scale: 2, size: 4).incomingMessage()
|
||||||
|
}
|
||||||
|
for (int i = 0; i <= 100; i += 10) {
|
||||||
|
status "energy ${i} kWh": new physicalgraph.zwave.Zwave().meterV1.meterReport(
|
||||||
|
scaledMeterValue: i, precision: 3, meterType: 0, scale: 0, size: 4).incomingMessage()
|
||||||
|
}
|
||||||
|
|
||||||
|
// reply messages
|
||||||
|
reply "2001FF,delay 100,2502": "command: 2503, payload: FF"
|
||||||
|
reply "200100,delay 100,2502": "command: 2503, payload: 00"
|
||||||
|
}
|
||||||
|
|
||||||
|
// tile definitions
|
||||||
|
tiles(scale: 2) {
|
||||||
|
multiAttributeTile(name:"switch", type: "generic", width: 6, height: 4, canChangeIcon: true){
|
||||||
|
tileAttribute("device.switch", key: "PRIMARY_CONTROL") {
|
||||||
|
attributeState("on", label: '${name}', action: "switch.off", icon: "st.switches.switch.on", backgroundColor: "#00A0DC", nextState:"turningOff")
|
||||||
|
attributeState("off", label: '${name}', action: "switch.on", icon: "st.switches.switch.off", backgroundColor: "#ffffff", nextState:"turningOn")
|
||||||
|
attributeState("turningOn", label:'${name}', action:"switch.off", icon:"st.switches.switch.on", backgroundColor:"#00a0dc", nextState:"turningOff")
|
||||||
|
attributeState("turningOff", label:'${name}', action:"switch.on", icon:"st.switches.switch.off", backgroundColor:"#ffffff", nextState:"turningOn")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
valueTile("power", "device.power", width: 2, height: 2) {
|
||||||
|
state "default", label:'${currentValue} W'
|
||||||
|
}
|
||||||
|
valueTile("energy", "device.energy", width: 2, height: 2) {
|
||||||
|
state "default", label:'${currentValue} kWh'
|
||||||
|
}
|
||||||
|
standardTile("reset", "device.energy", inactiveLabel: false, decoration: "flat", width: 2, height: 2) {
|
||||||
|
state "default", label:'reset kWh', action:"reset"
|
||||||
|
}
|
||||||
|
standardTile("refresh", "device.power", inactiveLabel: false, decoration: "flat", width: 2, height: 2) {
|
||||||
|
state "default", label:'', action:"refresh.refresh", icon:"st.secondary.refresh"
|
||||||
|
}
|
||||||
|
|
||||||
|
main(["switch","power","energy"])
|
||||||
|
details(["switch","power","energy","refresh","reset"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def installed() {
|
||||||
|
log.debug "installed()"
|
||||||
|
// Device-Watch simply pings if no device events received for 32min(checkInterval)
|
||||||
|
initialize()
|
||||||
|
if (zwaveInfo?.mfr?.equals("0063") || zwaveInfo?.mfr?.equals("014F")) { // These old GE devices have to be polled. GoControl Plug refresh status every 15 min.
|
||||||
|
runEvery15Minutes("poll", [forceForLocallyExecuting: true])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
// Device-Watch simply pings if no device events received for 32min(checkInterval)
|
||||||
|
initialize()
|
||||||
|
if (zwaveInfo?.mfr?.equals("0063") || zwaveInfo?.mfr?.equals("014F")) { // These old GE devices have to be polled. GoControl Plug refresh status every 15 min.
|
||||||
|
unschedule("poll", [forceForLocallyExecuting: true])
|
||||||
|
runEvery15Minutes("poll", [forceForLocallyExecuting: true])
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
if (!state.MSR) {
|
||||||
|
response(zwave.manufacturerSpecificV2.manufacturerSpecificGet().format())
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
log.debug e
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def initialize() {
|
||||||
|
sendEvent(name: "checkInterval", value: 2 * 15 * 60 + 2 * 60, displayed: false, data: [protocol: "zwave", hubHardwareId: device.hub.hardwareID])
|
||||||
|
}
|
||||||
|
|
||||||
|
def getCommandClassVersions() {
|
||||||
|
[
|
||||||
|
0x20: 1, // Basic
|
||||||
|
0x32: 3, // Meter
|
||||||
|
0x56: 1, // Crc16Encap
|
||||||
|
0x70: 1, // Configuration
|
||||||
|
0x72: 2, // ManufacturerSpecific
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
// parse events into attributes
|
||||||
|
def parse(String description) {
|
||||||
|
log.debug "parse() - description: "+description
|
||||||
|
def result = null
|
||||||
|
if (description != "updated") {
|
||||||
|
def cmd = zwave.parse(description, commandClassVersions)
|
||||||
|
if (cmd) {
|
||||||
|
result = zwaveEvent(cmd)
|
||||||
|
log.debug("'$description' parsed to $result")
|
||||||
|
} else {
|
||||||
|
log.debug("Couldn't zwave.parse '$description'")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
def handleMeterReport(cmd){
|
||||||
|
if (cmd.meterType == 1) {
|
||||||
|
if (cmd.scale == 0) {
|
||||||
|
createEvent(name: "energy", value: cmd.scaledMeterValue, unit: "kWh")
|
||||||
|
} else if (cmd.scale == 1) {
|
||||||
|
createEvent(name: "energy", value: cmd.scaledMeterValue, unit: "kVAh")
|
||||||
|
} else if (cmd.scale == 2) {
|
||||||
|
createEvent(name: "power", value: Math.round(cmd.scaledMeterValue), unit: "W")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.meterv3.MeterReport cmd) {
|
||||||
|
log.debug "v3 Meter report: "+cmd
|
||||||
|
handleMeterReport(cmd)
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.basicv1.BasicReport cmd)
|
||||||
|
{
|
||||||
|
log.debug "Basic report: "+cmd
|
||||||
|
def value = (cmd.value ? "on" : "off")
|
||||||
|
def evt = createEvent(name: "switch", value: value, type: "physical", descriptionText: "$device.displayName was turned $value")
|
||||||
|
if (evt.isStateChange) {
|
||||||
|
[evt, response(["delay 3000", meterGet(scale: 2).format()])]
|
||||||
|
} else {
|
||||||
|
evt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.switchbinaryv1.SwitchBinaryReport cmd)
|
||||||
|
{
|
||||||
|
log.debug "Switch binary report: "+cmd
|
||||||
|
def value = (cmd.value ? "on" : "off")
|
||||||
|
createEvent(name: "switch", value: value, type: "digital", descriptionText: "$device.displayName was turned $value")
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.manufacturerspecificv2.ManufacturerSpecificReport cmd) {
|
||||||
|
def result = []
|
||||||
|
|
||||||
|
def msr = String.format("%04X-%04X-%04X", cmd.manufacturerId, cmd.productTypeId, cmd.productId)
|
||||||
|
log.debug "msr: $msr"
|
||||||
|
updateDataValue("MSR", msr)
|
||||||
|
|
||||||
|
result << createEvent(descriptionText: "$device.displayName MSR: $msr", isStateChange: false)
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
|
log.debug "${device.displayName}: Unhandled: $cmd"
|
||||||
|
[:]
|
||||||
|
}
|
||||||
|
|
||||||
|
def on() {
|
||||||
|
encapSequence([
|
||||||
|
zwave.basicV1.basicSet(value: 0xFF),
|
||||||
|
zwave.switchBinaryV1.switchBinaryGet(),
|
||||||
|
meterGet(scale: 2)
|
||||||
|
], 3000)
|
||||||
|
}
|
||||||
|
|
||||||
|
def off() {
|
||||||
|
encapSequence([
|
||||||
|
zwave.basicV1.basicSet(value: 0x00),
|
||||||
|
zwave.switchBinaryV1.switchBinaryGet(),
|
||||||
|
meterGet(scale: 2)
|
||||||
|
], 3000)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PING is used by Device-Watch in attempt to reach the Device
|
||||||
|
* */
|
||||||
|
def ping() {
|
||||||
|
log.debug "ping()"
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
def poll() {
|
||||||
|
sendHubCommand(refresh())
|
||||||
|
}
|
||||||
|
|
||||||
|
def refresh() {
|
||||||
|
log.debug "refresh()"
|
||||||
|
encapSequence([
|
||||||
|
zwave.switchBinaryV1.switchBinaryGet(),
|
||||||
|
meterGet(scale: 0),
|
||||||
|
meterGet(scale: 2)
|
||||||
|
])
|
||||||
|
}
|
||||||
|
|
||||||
|
def configure() {
|
||||||
|
log.debug "configure()"
|
||||||
|
def result = []
|
||||||
|
|
||||||
|
log.debug "Configure zwaveInfo: "+zwaveInfo
|
||||||
|
|
||||||
|
if (zwaveInfo.mfr == "0086") { // Aeon Labs meter
|
||||||
|
result << response(encap(zwave.configurationV1.configurationSet(parameterNumber: 80, size: 1, scaledConfigurationValue: 2))) // basic report cc
|
||||||
|
result << response(encap(zwave.configurationV1.configurationSet(parameterNumber: 101, size: 4, scaledConfigurationValue: 12))) // report power in watts
|
||||||
|
result << response(encap(zwave.configurationV1.configurationSet(parameterNumber: 111, size: 4, scaledConfigurationValue: 300))) // every 5 min
|
||||||
|
} else if (zwaveInfo.mfr == "010F" && zwaveInfo.prod == "1801" && zwaveInfo.model == "1000") { // Fibaro Wall Plug UK
|
||||||
|
result << response(encap(zwave.configurationV1.configurationSet(parameterNumber: 11, size: 1, scaledConfigurationValue: 2))) // 2% power change results in report
|
||||||
|
result << response(encap(zwave.configurationV1.configurationSet(parameterNumber: 13, size: 2, scaledConfigurationValue: 5*60))) // report every 5 minutes
|
||||||
|
} else if (zwaveInfo.mfr == "014F" && zwaveInfo.prod == "5053" && zwaveInfo.model == "3531") {
|
||||||
|
result << response(encap(zwave.configurationV1.configurationSet(parameterNumber: 13, size: 2, scaledConfigurationValue: 15))) //report kWH every 15 min
|
||||||
|
} else if (zwaveInfo.mfr == "0154" && zwaveInfo.prod == "0003" && zwaveInfo.model == "000A") {
|
||||||
|
result << response(encap(zwave.configurationV1.configurationSet(parameterNumber: 25, size: 1, scaledConfigurationValue: 1))) //report every 1W change
|
||||||
|
}
|
||||||
|
result << response(encap(meterGet(scale: 0)))
|
||||||
|
result << response(encap(meterGet(scale: 2)))
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
def reset() {
|
||||||
|
encapSequence([
|
||||||
|
meterReset(),
|
||||||
|
meterGet(scale: 0)
|
||||||
|
])
|
||||||
|
}
|
||||||
|
|
||||||
|
def meterGet(map)
|
||||||
|
{
|
||||||
|
return zwave.meterV2.meterGet(map)
|
||||||
|
}
|
||||||
|
|
||||||
|
def meterReset()
|
||||||
|
{
|
||||||
|
return zwave.meterV2.meterReset()
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Security encapsulation support:
|
||||||
|
*/
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.securityv1.SecurityMessageEncapsulation cmd) {
|
||||||
|
def encapsulatedCommand = cmd.encapsulatedCommand(commandClassVersions)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
log.debug "Parsed SecurityMessageEncapsulation into: ${encapsulatedCommand}"
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
} else {
|
||||||
|
log.warn "Unable to extract Secure command from $cmd"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.crc16encapv1.Crc16Encap cmd) {
|
||||||
|
def version = commandClassVersions[cmd.commandClass as Integer]
|
||||||
|
def ccObj = version ? zwave.commandClass(cmd.commandClass, version) : zwave.commandClass(cmd.commandClass)
|
||||||
|
def encapsulatedCommand = ccObj?.command(cmd.command)?.parse(cmd.data)
|
||||||
|
if (encapsulatedCommand) {
|
||||||
|
log.debug "Parsed Crc16Encap into: ${encapsulatedCommand}"
|
||||||
|
zwaveEvent(encapsulatedCommand)
|
||||||
|
} else {
|
||||||
|
log.warn "Unable to extract CRC16 command from $cmd"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private secEncap(physicalgraph.zwave.Command cmd) {
|
||||||
|
log.debug "encapsulating command using Secure Encapsulation, command: $cmd"
|
||||||
|
zwave.securityV1.securityMessageEncapsulation().encapsulate(cmd).format()
|
||||||
|
}
|
||||||
|
|
||||||
|
private crcEncap(physicalgraph.zwave.Command cmd) {
|
||||||
|
log.debug "encapsulating command using CRC16 Encapsulation, command: $cmd"
|
||||||
|
zwave.crc16EncapV1.crc16Encap().encapsulate(cmd).format()
|
||||||
|
}
|
||||||
|
|
||||||
|
private encap(physicalgraph.zwave.Command cmd) {
|
||||||
|
if (zwaveInfo?.zw?.contains("s")) {
|
||||||
|
secEncap(cmd)
|
||||||
|
} else if (zwaveInfo?.cc?.contains("56")){
|
||||||
|
crcEncap(cmd)
|
||||||
|
} else {
|
||||||
|
log.debug "no encapsulation supported for command: $cmd"
|
||||||
|
cmd.format()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private encapSequence(cmds, Integer delay=250) {
|
||||||
|
delayBetween(cmds.collect{ encap(it) }, delay)
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user