mirror of
https://github.com/mtan93/SmartThingsPublic.git
synced 2026-03-21 21:03:47 +00:00
Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b825fc99f4 |
@@ -0,0 +1,249 @@
|
|||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
* modified by lgkahn v1 for control of the on/off light
|
||||||
|
* the lite when on/off is not the normal parameter settings of the config class and there also
|
||||||
|
* is no never lite option.. So custom code for this device only.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
metadata {
|
||||||
|
definition (name: "Enerwave ZW15SM Metering Switch", namespace: "smartthings", author: "LGKahn") {
|
||||||
|
capability "Energy Meter"
|
||||||
|
capability "Actuator"
|
||||||
|
capability "Switch"
|
||||||
|
capability "Power Meter"
|
||||||
|
capability "Polling"
|
||||||
|
capability "Refresh"
|
||||||
|
capability "Configuration"
|
||||||
|
capability "Sensor"
|
||||||
|
capability "Indicator"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
fingerprint inClusters: "0x25,0x32"
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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 {
|
||||||
|
standardTile("switch", "device.switch", width: 2, height: 2, canChangeIcon: true) {
|
||||||
|
state "on", label: '${name}', action: "switch.off", icon: "st.switches.switch.on", backgroundColor: "#79b821"
|
||||||
|
state "off", label: '${name}', action: "switch.on", icon: "st.switches.switch.off", backgroundColor: "#ffffff"
|
||||||
|
}
|
||||||
|
valueTile("power", "device.power") {
|
||||||
|
state "default", label:'${currentValue} W'
|
||||||
|
}
|
||||||
|
valueTile("energy", "device.energy") {
|
||||||
|
state "default", label:'${currentValue} kWh'
|
||||||
|
}
|
||||||
|
standardTile("reset", "device.energy", inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "default", label:'reset kWh', action:"reset"
|
||||||
|
}
|
||||||
|
standardTile("refresh", "device.power", inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "default", label:'', action:"refresh.refresh", icon:"st.secondary.refresh"
|
||||||
|
}
|
||||||
|
|
||||||
|
standardTile("indicatorStatus", "device.indicatorStatus", width: 1, height: 1, inactiveLabel: false, decoration: "flat") {
|
||||||
|
state "when off", action:"indicator.indicatorWhenOn", icon:"st.indicators.lit-when-off"
|
||||||
|
state "when on", action:"indicator.indicatorWhenOff", icon:"st.indicators.lit-when-on"
|
||||||
|
|
||||||
|
}
|
||||||
|
main(["switch","power","energy"])
|
||||||
|
details(["switch","power","energy","indicatorStatus","refresh","reset"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def updated() {
|
||||||
|
try {
|
||||||
|
if (!state.MSR) {
|
||||||
|
response(zwave.manufacturerSpecificV2.manufacturerSpecificGet().format())
|
||||||
|
}
|
||||||
|
} catch (e) { log.debug e }
|
||||||
|
}
|
||||||
|
|
||||||
|
def parse(String description) {
|
||||||
|
def result = null
|
||||||
|
//log.debug "in parse desc = $description"
|
||||||
|
|
||||||
|
if(description == "updated") return
|
||||||
|
def cmd = zwave.parse(description, [0x20: 1, 0x32: 1, 0x72: 2])
|
||||||
|
if (cmd) {
|
||||||
|
result = zwaveEvent(cmd)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.meterv1.MeterReport cmd) {
|
||||||
|
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.basicv1.BasicReport cmd)
|
||||||
|
{
|
||||||
|
def evt = createEvent(name: "switch", value: cmd.value ? "on" : "off", type: "physical")
|
||||||
|
if (evt.isStateChange) {
|
||||||
|
[evt, response(["delay 3000", zwave.meterV2.meterGet(scale: 2).format()])]
|
||||||
|
} else {
|
||||||
|
evt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.switchbinaryv1.SwitchBinaryReport cmd)
|
||||||
|
{
|
||||||
|
createEvent(name: "switch", value: cmd.value ? "on" : "off", type: "digital")
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.commands.configurationv1.ConfigurationReport cmd) {
|
||||||
|
|
||||||
|
// log.debug "in config report got indicatorstatus = $state.currentIndicatorStatus"
|
||||||
|
[name: "indicatorStatus", value: state.currentIndicatorStatus, display: true]
|
||||||
|
}
|
||||||
|
|
||||||
|
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)
|
||||||
|
|
||||||
|
// retypeBasedOnMSR()
|
||||||
|
|
||||||
|
result << createEvent(descriptionText: "$device.displayName MSR: $msr", isStateChange: false)
|
||||||
|
|
||||||
|
if (msr.startsWith("0086") && !state.aeonconfig) { // Aeon Labs meter
|
||||||
|
state.aeonconfig = 1
|
||||||
|
result << response(delayBetween([
|
||||||
|
zwave.configurationV1.configurationSet(parameterNumber: 101, size: 4, scaledConfigurationValue: 4).format(), // report power in watts
|
||||||
|
zwave.configurationV1.configurationSet(parameterNumber: 111, size: 4, scaledConfigurationValue: 300).format(), // every 5 min
|
||||||
|
zwave.configurationV1.configurationSet(parameterNumber: 102, size: 4, scaledConfigurationValue: 8).format(), // report energy in kWh
|
||||||
|
zwave.configurationV1.configurationSet(parameterNumber: 112, size: 4, scaledConfigurationValue: 300).format(), // every 5 min
|
||||||
|
zwave.configurationV1.configurationSet(parameterNumber: 103, size: 4, scaledConfigurationValue: 0).format(), // no third report
|
||||||
|
//zwave.configurationV1.configurationSet(parameterNumber: 113, size: 4, scaledConfigurationValue: 300).format(), // every 5 min
|
||||||
|
zwave.meterV2.meterGet(scale: 0).format(),
|
||||||
|
zwave.meterV2.meterGet(scale: 2).format(),
|
||||||
|
]))
|
||||||
|
} else {
|
||||||
|
result << response(delayBetween([
|
||||||
|
zwave.meterV2.meterGet(scale: 0).format(),
|
||||||
|
zwave.meterV2.meterGet(scale: 2).format(),
|
||||||
|
]))
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
def zwaveEvent(physicalgraph.zwave.Command cmd) {
|
||||||
|
log.debug "$device.displayName: Unhandled: $cmd"
|
||||||
|
[:]
|
||||||
|
}
|
||||||
|
|
||||||
|
def on() {
|
||||||
|
|
||||||
|
log.debug "in on"
|
||||||
|
[
|
||||||
|
|
||||||
|
zwave.basicV1.basicSet(value: 0xFF).format(),
|
||||||
|
zwave.switchBinaryV1.switchBinaryGet().format(),
|
||||||
|
"delay 3000",
|
||||||
|
zwave.meterV2.meterGet(scale: 2).format()
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
def off() {
|
||||||
|
log.debug "in off"
|
||||||
|
[
|
||||||
|
zwave.basicV1.basicSet(value: 0x00).format(),
|
||||||
|
zwave.switchBinaryV1.switchBinaryGet().format(),
|
||||||
|
"delay 3000",
|
||||||
|
zwave.meterV2.meterGet(scale: 2).format()
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
def poll() {
|
||||||
|
delayBetween([
|
||||||
|
zwave.switchBinaryV1.switchBinaryGet().format(),
|
||||||
|
zwave.meterV2.meterGet(scale: 0).format(),
|
||||||
|
zwave.meterV2.meterGet(scale: 2).format()
|
||||||
|
])
|
||||||
|
}
|
||||||
|
|
||||||
|
def refresh() {
|
||||||
|
def value = "when off"
|
||||||
|
|
||||||
|
log.debug "in refresh for whenon got indicatorstatus = $state.currentIndicatorStatus"
|
||||||
|
sendEvent(name: "indicatorStatus", value: state.currentIndicatorStatus, display: true)
|
||||||
|
|
||||||
|
delayBetween([
|
||||||
|
zwave.switchBinaryV1.switchBinaryGet().format(),
|
||||||
|
zwave.meterV2.meterGet(scale: 0).format(),
|
||||||
|
zwave.meterV2.meterGet(scale: 2).format()
|
||||||
|
])
|
||||||
|
}
|
||||||
|
|
||||||
|
def configure() {
|
||||||
|
zwave.manufacturerSpecificV2.manufacturerSpecificGet().format()
|
||||||
|
}
|
||||||
|
|
||||||
|
def reset() {
|
||||||
|
state.currentIndicatorStatus = "when on"
|
||||||
|
return [
|
||||||
|
zwave.meterV2.meterReset().format(),
|
||||||
|
zwave.meterV2.meterGet(scale: 0).format()
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def indicatorWhenOn() {
|
||||||
|
log.debug "in when on"
|
||||||
|
state.currentIndicatorStatus = "when on"
|
||||||
|
sendEvent(name: "indicatorStatus", value: "when on", display: true)
|
||||||
|
zwave.configurationV1.configurationSet(parameterNumber: 0x01, size: 1, scaledConfigurationValue: 1).format()
|
||||||
|
}
|
||||||
|
|
||||||
|
def indicatorWhenOff() {
|
||||||
|
log.debug "in when off"
|
||||||
|
state.currentIndicatorStatus = "when off"
|
||||||
|
sendEvent(name: "indicatorStatus", value: "when off", display: true)
|
||||||
|
zwave.configurationV1.configurationSet(parameterNumber: 0x01, size: 1, scaledConfigurationValue: 0).format()
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,200 +0,0 @@
|
|||||||
/**
|
|
||||||
* 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.
|
|
||||||
*
|
|
||||||
* lg kahn smart app to turn device on when temp is between two values
|
|
||||||
* and off otherwise. Also has to be between 2 date ranges.
|
|
||||||
* also give alerts when turning on off.
|
|
||||||
* I use for a roof heater wire/coil.
|
|
||||||
*
|
|
||||||
* Author: LGKahn kahn-st@lgk.com
|
|
||||||
*/
|
|
||||||
|
|
||||||
definition(
|
|
||||||
name: "Roof Coil Controller",
|
|
||||||
namespace: "smartthings",
|
|
||||||
author: "lgkahn",
|
|
||||||
description: "Control a roof coil or othe device(s) when temperature is between two values turns on and also has to be within a date range. Automatically turns off if one of the conditions is not met. Alerting option also.",
|
|
||||||
category: "Green Living",
|
|
||||||
iconUrl: "https://s3.amazonaws.com/smartapp-icons/Meta/temp_thermo-switch.png",
|
|
||||||
iconX2Url: "https://s3.amazonaws.com/smartapp-icons/Meta/temp_thermo-switch@2x.png"
|
|
||||||
)
|
|
||||||
|
|
||||||
preferences {
|
|
||||||
section("Choose a temperature sensor... "){
|
|
||||||
input "sensor", "capability.temperatureMeasurement", title: "Sensor"
|
|
||||||
}
|
|
||||||
section("Select the outlet(s)... "){
|
|
||||||
input "outlets", "capability.switch", title: "Outlets", multiple: true
|
|
||||||
}
|
|
||||||
section("Turn on when temp is above ..."){
|
|
||||||
input "onSetPoint", "decimal", title: "Set On Temp"
|
|
||||||
}
|
|
||||||
section("Turn off when temp is above ..."){
|
|
||||||
input "offSetPoint", "decimal", title: "Set Off Temp"
|
|
||||||
}
|
|
||||||
section("Start after Date format (yyyymmdd)..."){
|
|
||||||
input "startDate", "number", title: "Date?"
|
|
||||||
}
|
|
||||||
section("End after Date format (yyyymmdd)..."){
|
|
||||||
input "endDate", "number", title: "Date?"
|
|
||||||
}
|
|
||||||
section("Time Zone Offset ie -5 etc...."){
|
|
||||||
input "tzOffset", "number", title: "Offset?", range: "-12..12"
|
|
||||||
}
|
|
||||||
section( "Notifications" ) {
|
|
||||||
input("recipients", "contact", title: "Send notifications to") {
|
|
||||||
input "sendPushMessage", "enum", title: "Send a push notification?", options: ["Yes", "No"], required: false
|
|
||||||
input "phone1", "phone", title: "Send a Text Message?", required: false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
def installed()
|
|
||||||
{
|
|
||||||
log.debug "in coil controller installed ... currenttemp = $sensor.currentTemperature"
|
|
||||||
subscribe(sensor, "temperature", temperatureHandler)
|
|
||||||
}
|
|
||||||
|
|
||||||
def updated()
|
|
||||||
{
|
|
||||||
log.debug "in coil controller updated ... currenttemp = $sensor.currentTemperature"
|
|
||||||
unsubscribe()
|
|
||||||
subscribe(sensor, "temperature", temperatureHandler)
|
|
||||||
// for debugging and testing uncomment temperatureHandlerTest()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def temperatureHandler(evt)
|
|
||||||
{
|
|
||||||
def currenttemp = sensor.currentTemperature
|
|
||||||
/*
|
|
||||||
log.debug "in temp handler"
|
|
||||||
log.debug "current temp = $currenttemp"
|
|
||||||
log.debug "onSetPoint = $onSetPoint"
|
|
||||||
log.debug "offSetPoint = $offSetPoint"
|
|
||||||
log.debug "set offset = $tzOffset"
|
|
||||||
*/
|
|
||||||
def today = new Date();
|
|
||||||
|
|
||||||
def ltf = new java.text.SimpleDateFormat("yyyyMMdd")
|
|
||||||
ltf.setTimeZone(TimeZone.getTimeZone("GMT${tzOffset}"))
|
|
||||||
|
|
||||||
String date1 = ltf.format(today);
|
|
||||||
int intdate = Integer.parseInt(date1)
|
|
||||||
// log.debug "int date = $intdate"
|
|
||||||
//log.debug "enddate = $endDate"
|
|
||||||
// log.debug "startdate = $startDate"
|
|
||||||
def currSwitches = outlets.currentSwitch
|
|
||||||
def onOutlets = currSwitches.findAll { switchVal ->
|
|
||||||
switchVal == "on" ? true : false }
|
|
||||||
|
|
||||||
if (((intdate >= startDate) && (intdate <= endDate))
|
|
||||||
&& ((currenttemp > onSetPoint) && (currenttemp < offSetPoint)))
|
|
||||||
{
|
|
||||||
// dont do anything if already on
|
|
||||||
if (onOutlets.size() != outlets.size())
|
|
||||||
{
|
|
||||||
log.debug "turning outlets On as $sensor.displayName is reporting $currenttemp which is between $onSetPoint and $offSetPoint, and we are within the date range ($startDate - $endDate)!"
|
|
||||||
mysend("Turning device(s) On as $sensor.displayName is reporting a temperature of $currenttemp which is between $onSetPoint and $offSetPoint, and we are within the date range ($startDate - $endDate)!")
|
|
||||||
outlets.on()
|
|
||||||
}
|
|
||||||
else log.debug "Not turning on again, all already on!"
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// dont do anything if already off
|
|
||||||
if (onOutlets.size() != 0)
|
|
||||||
{
|
|
||||||
log.debug "turning outlets Off! as $sensor.displayName is reporting $currenttemp which is Not between $onSetPoint and $offSetPoint, or we are no longer within the date range ($startDate - $endDate)!"
|
|
||||||
mysend("Turning device(s) Off as $sensor.displayName is reporting a temperature of $currenttemp which is not between $onSetPoint and $offSetPoint, or we are no longer within the date range ($startDate - $endDate)!")
|
|
||||||
outlets.off()
|
|
||||||
}
|
|
||||||
else log.debug "All outlets already off!"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def temperatureHandlerTest()
|
|
||||||
{
|
|
||||||
//log.trace "temperature: $evt.value, $evt"
|
|
||||||
// this routine is only for testing and debugging. to test or make changes uncomment the call in update.
|
|
||||||
// this is so we dont have to wait 15 minutes till a half an hour for the event to fire for testing.
|
|
||||||
|
|
||||||
def currenttemp = sensor.currentTemperature
|
|
||||||
|
|
||||||
log.debug "in temp handler test"
|
|
||||||
log.debug "current temp = $currenttemp"
|
|
||||||
log.debug "onSetPoint = $onSetPoint"
|
|
||||||
log.debug "offSetPoint = $offSetPoint"
|
|
||||||
log.debug "set offset = $tzOffset"
|
|
||||||
def today = new Date();
|
|
||||||
|
|
||||||
def ltf = new java.text.SimpleDateFormat("yyyyMMdd")
|
|
||||||
ltf.setTimeZone(TimeZone.getTimeZone("GMT${tzOffset}"))
|
|
||||||
|
|
||||||
String date1 = ltf.format(today);
|
|
||||||
int intdate = Integer.parseInt(date1)
|
|
||||||
log.debug "int date = $intdate"
|
|
||||||
log.debug "enddate = $endDate"
|
|
||||||
log.debug "startdate = $startDate"
|
|
||||||
def currSwitches = outlets.currentSwitch
|
|
||||||
def onOutlets = currSwitches.findAll { switchVal ->
|
|
||||||
switchVal == "on" ? true : false }
|
|
||||||
|
|
||||||
log.debug "how many on = ${onOutlets.size()} "
|
|
||||||
|
|
||||||
if (((intdate >= startDate) && (intdate <= endDate))
|
|
||||||
&& ((currenttemp > onSetPoint) && (currenttemp < offSetPoint)))
|
|
||||||
{
|
|
||||||
// dont do anything if already on
|
|
||||||
if (onOutlets.size() != outlets.size())
|
|
||||||
{
|
|
||||||
log.debug "turning outlets On as $sensor.displayName is reporting $currenttemp which is between $onSetPoint and $offSetPoint, and we are within the date range ($startDate - $endDate)!"
|
|
||||||
//mysend("Turning device(s) On as $sensor.displayName is reporting a temperature of $currenttemp which is between $onSetPoint and $offSetPoint, and we are within the date range ($startDate - $endDate)!")
|
|
||||||
outlets.on()
|
|
||||||
}
|
|
||||||
else log.debug "Not turning on again, all already on!"
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// dont do anything if already off
|
|
||||||
if (onOutlets.size() != 0)
|
|
||||||
{
|
|
||||||
log.debug "turning outlets Off! as $sensor.displayName is reporting $currenttemp which is Not between $onSetPoint and $offSetPoint, or we are no longer within the date range ($startDate - $endDate)!"
|
|
||||||
//mysend("Turning device(s) Off as $sensor.displayName is reporting a temperature of $currenttemp which is not between $onSetPoint and $offSetPoint, or we are no longer within the date range ($startDate - $endDate)!")
|
|
||||||
outlets.off()
|
|
||||||
}
|
|
||||||
else log.debug "All outlets already off!"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
private mysend(msg) {
|
|
||||||
if (location.contactBookEnabled) {
|
|
||||||
log.debug("sending notifications to: ${recipients?.size()}")
|
|
||||||
sendNotificationToContacts(msg, recipients)
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
if (sendPushMessage != "No") {
|
|
||||||
log.debug("sending push message")
|
|
||||||
sendPush(msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (phone1) {
|
|
||||||
log.debug("sending text message")
|
|
||||||
sendSms(phone1, msg)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
log.debug msg
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user