mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-05-11 14:26:02 +01:00
Compare commits
30 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 82009b8e61 | |||
| 2b3a4e1278 | |||
| 825e693efd | |||
| 686c8f7337 | |||
| 11f4e42fe9 | |||
| bc459ae178 | |||
| 1392b6e1a5 | |||
| 3db96faa00 | |||
| 399fbcb676 | |||
| eed1ced71b | |||
| d080833d5c | |||
| e998528e8e | |||
| 3472ee329d | |||
| 577b127287 | |||
| d85566bb98 | |||
| 5f41af35e2 | |||
| e1a5b4dd27 | |||
| a2baa37901 | |||
| 922ab45343 | |||
| 962774996e | |||
| d79594cbcb | |||
| bf8fe4cad7 | |||
| 65752ce378 | |||
| 95f08aeb3d | |||
| cd7bc1b262 | |||
| be7f6a76a9 | |||
| 10e5b7e9d7 | |||
| 90e6dc91eb | |||
| 3ee8f86aa3 | |||
| d1a910f11f |
@@ -22,6 +22,7 @@ metadata {
|
||||
capability "Configuration"
|
||||
capability "Sensor"
|
||||
capability "Battery"
|
||||
capability "Health Check"
|
||||
|
||||
attribute "tamper", "enum", ["detected", "clear"]
|
||||
attribute "batteryStatus", "string"
|
||||
@@ -326,6 +327,9 @@ def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||
}
|
||||
|
||||
def configure() {
|
||||
// allow device user configured or default 16 min to check in; double the periodic reporting interval
|
||||
sendEvent(name: "checkInterval", value: 2* timeOptionValueMap[reportInterval] ?: 2*8*60, displayed: false)
|
||||
|
||||
// This sensor joins as a secure device if you double-click the button to include it
|
||||
log.debug "${device.displayName} is configuring its settings"
|
||||
def request = []
|
||||
|
||||
@@ -20,6 +20,9 @@ metadata {
|
||||
capability "Configuration"
|
||||
capability "Sensor"
|
||||
capability "Battery"
|
||||
capability "Health Check"
|
||||
|
||||
command "configureAfterSecure"
|
||||
|
||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x86,0x72,0x59,0x85,0x73,0x71,0x84,0x80,0x30,0x31,0x70,0x98,0x7A", outClusters:"0x5A"
|
||||
}
|
||||
@@ -245,6 +248,8 @@ def configureAfterSecure() {
|
||||
def configure() {
|
||||
// log.debug "configure()"
|
||||
//["delay 30000"] + secure(zwave.securityV1.securityCommandsSupportedGet())
|
||||
// allow device 16 min to check in; double the periodic reporting interval
|
||||
sendEvent(name: "checkInterval", value: 2*8*60, displayed: false)
|
||||
}
|
||||
|
||||
private setConfigured() {
|
||||
|
||||
@@ -20,6 +20,7 @@ metadata {
|
||||
capability "Illuminance Measurement"
|
||||
capability "Sensor"
|
||||
capability "Battery"
|
||||
capability "Health Check"
|
||||
|
||||
fingerprint deviceId: "0x2001", inClusters: "0x30,0x31,0x80,0x84,0x70,0x85,0x72,0x86"
|
||||
}
|
||||
@@ -180,6 +181,9 @@ def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||
}
|
||||
|
||||
def configure() {
|
||||
// allow device 10 min to check in; double the periodic reporting interval
|
||||
sendEvent(name: "checkInterval", value: 2*5*60, displayed: false)
|
||||
|
||||
delayBetween([
|
||||
// send binary sensor report instead of basic set for motion
|
||||
zwave.configurationV1.configurationSet(parameterNumber: 5, size: 1, scaledConfigurationValue: 2).format(),
|
||||
|
||||
+2
-1
@@ -255,7 +255,8 @@ private Map getBatteryResult(rawValue) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||
}
|
||||
}
|
||||
|
||||
+2
-1
@@ -271,7 +271,8 @@ private Map getBatteryResult(rawValue) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||
}
|
||||
}
|
||||
|
||||
+2
-1
@@ -240,7 +240,8 @@ private Map getBatteryResult(rawValue) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -338,7 +338,8 @@ private Map getBatteryResult(rawValue) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "{{ device.displayName }} battery was {{ value }}%"
|
||||
}
|
||||
}
|
||||
|
||||
+2
-1
@@ -234,7 +234,8 @@ def getTemperature(value) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||
}
|
||||
|
||||
|
||||
+2
-1
@@ -229,7 +229,8 @@ private Map getBatteryResult(rawValue) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||
}
|
||||
|
||||
|
||||
+2
-1
@@ -205,7 +205,8 @@ private Map getBatteryResult(rawValue) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||
}
|
||||
|
||||
|
||||
@@ -223,7 +223,8 @@ private Map getBatteryResult(rawValue) {
|
||||
def minVolts = 2.1
|
||||
def maxVolts = 3.0
|
||||
def pct = (volts - minVolts) / (maxVolts - minVolts)
|
||||
result.value = Math.min(100, (int) pct * 100)
|
||||
def roundedPct = Math.round(pct * 100)
|
||||
result.value = Math.min(100, roundedPct)
|
||||
result.descriptionText = "${linkText} battery was ${result.value}%"
|
||||
}
|
||||
|
||||
|
||||
+2
-3
@@ -28,7 +28,6 @@ metadata {
|
||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x98"
|
||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x86,0x72,0x98", outClusters: "0x5A,0x82"
|
||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x80,0x71,0x85,0x70,0x72,0x86,0x30,0x31,0x84,0x59,0x73,0x5A,0x8F,0x98,0x7A", outClusters:"0x20" // Philio multi+
|
||||
fingerprint deviceId: "0x0701", inClusters: "0x5E,0x72,0x5A,0x80,0x73,0x84,0x85,0x59,0x71,0x70,0x7A,0x98" // Vision door/window
|
||||
}
|
||||
|
||||
// simulator metadata
|
||||
@@ -83,12 +82,12 @@ def updated() {
|
||||
cmds = [
|
||||
command(zwave.manufacturerSpecificV2.manufacturerSpecificGet()),
|
||||
"delay 1200",
|
||||
zwave.wakeUpV1.wakeUpNoMoreInformation()
|
||||
zwave.wakeUpV1.wakeUpNoMoreInformation().format()
|
||||
]
|
||||
} else if (!state.lastbat) {
|
||||
cmds = []
|
||||
} else {
|
||||
cmds = [zwave.wakeUpV1.wakeUpNoMoreInformation()]
|
||||
cmds = [zwave.wakeUpV1.wakeUpNoMoreInformation().format()]
|
||||
}
|
||||
response(cmds)
|
||||
}
|
||||
|
||||
@@ -0,0 +1,131 @@
|
||||
/**
|
||||
* LifX Scene Manager
|
||||
*
|
||||
* Copyright 2016 Patrick Killian
|
||||
*
|
||||
* 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: "LifX Scene Manager",
|
||||
namespace: "climbingcoder",
|
||||
author: "Patrick Killian",
|
||||
description: "When your mode changes, automatically activate a Lifx lighting scene with the same name",
|
||||
category: "Convenience",
|
||||
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")
|
||||
|
||||
|
||||
preferences {
|
||||
section("Configuration") {
|
||||
input(name:"apikey", type:"text", title:"LifX API Token", required:true)
|
||||
input(name:"fadeTime", type:"number", title:"Fade Time", description:"Seconds of fade between scenes", required:true)
|
||||
}
|
||||
section("Getting a LifX API Token") {
|
||||
paragraph "1. Go to https://cloud.lifx.com/sign_in and sign in with your LifX account"
|
||||
paragraph "2. Click on your email address and then choose settings from the drop down menu"
|
||||
paragraph "3. Click 'Generate Token' and give your token any name"
|
||||
paragraph "4. Copy your token to the required field above and save it somewhere else for reference"
|
||||
}
|
||||
}
|
||||
|
||||
def installed() {
|
||||
log.debug "Installed with settings: ${settings}"
|
||||
initialize()
|
||||
}
|
||||
|
||||
def updated() {
|
||||
log.debug "Updated with settings: ${settings}"
|
||||
unsubscribe()
|
||||
initialize()
|
||||
}
|
||||
|
||||
def initialize() {
|
||||
subscribe(location, "mode", modeChangeHandler)
|
||||
}
|
||||
|
||||
|
||||
def modeChangeHandler(evt) {
|
||||
log.debug "mode changed to ${evt.value}"
|
||||
def sceneUUID = ""
|
||||
sceneUUID = findMatchingLifxScene("${evt.value}")
|
||||
if (sceneUUID == "") {
|
||||
log.debug "No matching scene found"
|
||||
} else {
|
||||
activateLifxScene("${sceneUUID}")
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns the UUID of a LifX scene that matches
|
||||
* the passed in sceneName by obtaining a list
|
||||
* of all available scenes and comparing them
|
||||
* to the passed in name
|
||||
*
|
||||
* @param sceneName - the name of the desired scene
|
||||
*
|
||||
* @return UUID - the UUID of any matching scenes found
|
||||
*/
|
||||
def findMatchingLifxScene(sceneName) {
|
||||
log.debug ("Getting list of LifX scenes")
|
||||
def params = [
|
||||
headers: ["Authorization": "Bearer ${settings.apikey}"],
|
||||
uri: "https://api.lifx.com/v1/scenes",
|
||||
body: []
|
||||
]
|
||||
def uuid = ""
|
||||
|
||||
try {
|
||||
httpGet(params) { resp ->
|
||||
log.debug "response status code: ${resp.status}"
|
||||
resp.data.each {
|
||||
log.debug("Found scene '${it.name}'")
|
||||
if ("${it.name}" == sceneName) {
|
||||
uuid = "${it.uuid}"
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (e) {
|
||||
log.error "something went wrong: $e"
|
||||
} finally {
|
||||
return uuid
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* Calls the Lifx http endpoint to activate a scene with
|
||||
* the given UUID
|
||||
*
|
||||
* @param UUID - the UUID of the scene to activate
|
||||
*/
|
||||
def activateLifxScene(UUID) {
|
||||
log.debug "Activating scene with UUID '${UUID}'"
|
||||
def params = [
|
||||
headers: ["Authorization": "Bearer ${settings.apikey}"],
|
||||
uri: "https://api.lifx.com/v1/scenes/scene_id:${UUID}/activate",
|
||||
body: [
|
||||
duration: "${settings.fadeTime}"
|
||||
]
|
||||
]
|
||||
|
||||
try {
|
||||
httpPut(params) { resp ->
|
||||
log.debug "response status code: ${resp.status}"
|
||||
}
|
||||
} catch (e) {
|
||||
log.error "something went wrong: $e"
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1,178 +0,0 @@
|
||||
/**
|
||||
* Timer Light
|
||||
*
|
||||
* Copyright 2016 Kevin Hill
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
*/
|
||||
|
||||
import groovy.time.*
|
||||
|
||||
definition(
|
||||
name: "Timer Light",
|
||||
namespace: "gr8gitsby",
|
||||
author: "Kevin Hill",
|
||||
description: "This puts outside lights on a timer",
|
||||
category: "Convenience",
|
||||
iconUrl: "https://source.unsplash.com/category/nature/60x60",
|
||||
iconX2Url: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience@2x.png",
|
||||
iconX3Url: "https://s3.amazonaws.com/smartapp-icons/Convenience/Cat-Convenience@2x.png")
|
||||
|
||||
|
||||
preferences {
|
||||
// Use a light swtich
|
||||
section("Light Switch") {
|
||||
// Capabilities link: http://docs.smartthings.com/en/latest/capabilities-reference.html
|
||||
// Input data types: http://docs.smartthings.com/en/latest/device-type-developers-guide/device-preferences.html?highlight=input#supported-input-types
|
||||
input "theSwitch", "capability.switch", title: "Pick your light switch"
|
||||
input "confirmationSwitch", "capability.switch", title: "Pick the notification switch"
|
||||
input "minutes", "number", title: "Enter the number of minutes you'd like the lights on"
|
||||
input "person", "capability.presenceSensor", title: "When who arrives?"
|
||||
|
||||
// When arrival is detected, if it is dark, turn on the outside lights
|
||||
}
|
||||
}
|
||||
|
||||
def installed() {
|
||||
log.debug "Installed with settings: ${settings}"
|
||||
subscribe(theSwitch, "switch", switchHandler)
|
||||
subscribe(confirmationSwitch, "switch", confirmationSwitchHandler)
|
||||
initialize()
|
||||
}
|
||||
|
||||
def updated() {
|
||||
log.debug "Updated with settings: ${settings}"
|
||||
unsubscribe()
|
||||
initialize()
|
||||
}
|
||||
|
||||
def initialize() {
|
||||
// This method sets up the app
|
||||
subscribe(person, "presence", presenceChange)
|
||||
state.Flashes = 4
|
||||
initialSunPosition()
|
||||
}
|
||||
|
||||
def presenceChange(evt){
|
||||
log.debug("presenceChange")
|
||||
// if someone arrives at night the lights will turn on
|
||||
if("present" == evt.value){
|
||||
timerLight()
|
||||
}
|
||||
}
|
||||
|
||||
// Function to set lights on timer for person arriving
|
||||
def timerLight() {
|
||||
log.debug(state.sunPosition)
|
||||
if(state.sunPosition == "down"){
|
||||
light.on()
|
||||
runIn(60 * minutes, turnOff)
|
||||
}
|
||||
}
|
||||
|
||||
def turnOff(){
|
||||
// This method turns off the ligt
|
||||
theSwitch.off()
|
||||
confirmationSwitch.off()
|
||||
}
|
||||
|
||||
def initialSunPosition() {
|
||||
// This method determines if sun is down at time of initializtion and run sunsetHandler() if so
|
||||
|
||||
def s = getSunriseAndSunset(zipCode: "98052")
|
||||
def now = new Date()
|
||||
def riseTime = s.sunrise
|
||||
def setTime = s.sunset
|
||||
|
||||
if(setTime.before(now)) {
|
||||
sunsetHandler()
|
||||
log.info "Sun is already down, run sunsetHandler"
|
||||
}
|
||||
else
|
||||
{ if (riseTime.after(now)) {
|
||||
sunriseHandler()
|
||||
log.info "Sun is up, run sunriseHandler"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
def sunriseHandler() {
|
||||
// method to set the sun position when the sun rises
|
||||
state.sunPosition = "up"
|
||||
}
|
||||
|
||||
def sunsetHandler() {
|
||||
// method to set the sun position when the sunsets
|
||||
state.sunPosition = "down"
|
||||
}
|
||||
|
||||
/*
|
||||
def switchHandler(evt) {
|
||||
log.debug("switch Handler Fired")
|
||||
}
|
||||
*/
|
||||
def confirmationSwitchHandler(evt) {
|
||||
state.Flashes = state.Flashes - 1
|
||||
log.debug("Flashes: ${state.Flashes}")
|
||||
log.debug("Event State: ${evt.value}")
|
||||
if(evt.value == "on" && state.Flashes > 0) {
|
||||
runIn(2, turnOffConfirmationSwitch)
|
||||
} else if(evt.value == "off" && state.Flashes > 0) {
|
||||
runIn(2, turnOnConfirmationSwitch)
|
||||
}
|
||||
|
||||
if(state.Flashes == 0){
|
||||
state.Flashes = 4
|
||||
}
|
||||
}
|
||||
|
||||
def turnOnConfirmationSwitch(){
|
||||
confirmationSwitch.on()
|
||||
}
|
||||
|
||||
def turnOffConfirmationSwitch(){
|
||||
confirmationSwitch.off()
|
||||
}
|
||||
|
||||
def switchHandler(evt){
|
||||
log.debug("light Event handler fired")
|
||||
// use Event rather than DeviceState because we may be changing DeviceState to only store changed values
|
||||
def recentStates = theSwitch.eventsSince(new Date(now() - 4000), [all:true, max: 10]).findAll{it.name == "switch"}
|
||||
log.debug "${recentStates?.size()} STATES FOUND, LAST AT ${recentStates ? recentStates[0].dateCreated : ''}"
|
||||
//sendPush("Attempting to check light presses")
|
||||
|
||||
if (evt.value == "on") {
|
||||
log.debug("==== ON ====")
|
||||
}
|
||||
|
||||
log.debug(recentStates.size())
|
||||
//if (evt.isPhysical()) {
|
||||
|
||||
if (recentStates.size() >= 2) {
|
||||
|
||||
log.debug "Outside light timer mode engaged"
|
||||
def switchattr = confirmationSwitch.currentValue("switch")
|
||||
log.debug "confirmationSwitch: $switchattr"
|
||||
|
||||
confirmationSwitch.on()
|
||||
|
||||
//alternativeSwitch.off()
|
||||
//def message = "${location.name} executed ${settings.onPhrase} because ${evt.title} was tapped twice."
|
||||
runIn(60 * minutes, turnOff)
|
||||
} else {
|
||||
log.trace "Skipping digital on/off event"
|
||||
}
|
||||
|
||||
//log.debug("switch is on, turn off in $minutes")
|
||||
//runIn(60*minutes, turnOff)
|
||||
|
||||
//}
|
||||
}
|
||||
@@ -720,7 +720,7 @@ def completionPercentage() {
|
||||
|
||||
def now = new Date().getTime()
|
||||
def timeElapsed = now - atomicState.start
|
||||
def totalRunTime = totalRunTimeMillis()
|
||||
def totalRunTime = totalRunTimeMillis() ?: 1
|
||||
def percentComplete = timeElapsed / totalRunTime * 100
|
||||
log.debug "percentComplete: ${percentComplete}"
|
||||
|
||||
|
||||
@@ -689,7 +689,7 @@ def parse(childDevice, description) {
|
||||
log.warn "Parsing Body failed - trying again..."
|
||||
poll()
|
||||
}
|
||||
if (body instanceof java.util.HashMap) {
|
||||
if (body instanceof java.util.Map) {
|
||||
//poll response
|
||||
def bulbs = getChildDevices()
|
||||
for (bulb in body) {
|
||||
|
||||
@@ -688,7 +688,7 @@ def validateCommand(device, command) {
|
||||
def capabilityCommands = getDeviceCapabilityCommands(device.capabilities)
|
||||
def currentDeviceCapability = getCapabilityName(device)
|
||||
if (currentDeviceCapability != "" && capabilityCommands[currentDeviceCapability]) {
|
||||
return command in capabilityCommands[currentDeviceCapability] ? true : false
|
||||
return (command in capabilityCommands[currentDeviceCapability] || (currentDeviceCapability == "Switch" && command == "setLevel" && device.hasCommand("setLevel"))) ? true : false
|
||||
} else {
|
||||
// Handling other device types here, which don't accept commands
|
||||
httpError(400, "Bad request.")
|
||||
@@ -823,8 +823,8 @@ def deviceHandler(evt) {
|
||||
}
|
||||
|
||||
def sendToHarmony(evt, String callbackUrl) {
|
||||
def callback = new URI(callbackUrl)
|
||||
if(isIP(callback.host)){
|
||||
def callback = new URI(callbackUrl)
|
||||
if (callback.port != -1) {
|
||||
def host = callback.port != -1 ? "${callback.host}:${callback.port}" : callback.host
|
||||
def path = callback.query ? "${callback.path}?${callback.query}".toString() : callback.path
|
||||
sendHubCommand(new physicalgraph.device.HubAction(
|
||||
@@ -852,25 +852,6 @@ def sendToHarmony(evt, String callbackUrl) {
|
||||
}
|
||||
}
|
||||
|
||||
public static boolean isIP(String str) {
|
||||
try {
|
||||
String[] parts = str.split("\\.");
|
||||
if (parts.length != 4) return false;
|
||||
for (int i = 0; i < 4; ++i) {
|
||||
int p
|
||||
try {
|
||||
p = Integer.parseInt(parts[i]);
|
||||
} catch (Exception e) {
|
||||
return false;
|
||||
}
|
||||
if (p > 255 || p < 0) return false;
|
||||
}
|
||||
return true;
|
||||
} catch (Exception e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
def listHubs() {
|
||||
location.hubs?.findAll { it.type.toString() == "PHYSICAL" }?.collect { hubItem(it) }
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user