Compare commits

...

1 Commits

Author SHA1 Message Date
Darin Spivey
966bd2c1df MSA-1395: Smart Family Presence is a smart app with no handlers or device type dependencies.
The presence sensors have little/no configuration, especially when it comes to alerts.  I found lots of room for improvement regarding how those alerts are carried out.  Here are the problems with the original implementation of presence alerts:

1)  Each person gets and individual alert.  If you have a family of 4, each person will receive a *separate* alert for every person.
2)  There are no exceptions.  Even if the family is all together, it still individually alerts that everyone arrived/departed.
3)  Since you can't control which person gets the push alert, #1 and #2 exacerbate the issue by people receiving alerts for themselves.

The Smart Family Presence app lets the user select how and when they receive alerts.  The code is commented to explain the edge cases, but essentially, if the family arrives or departs together, no alert should be sent.  The app tracks state for whether the family was previously apart, so that it knows when and when not to send alerts.

I have found this code extremely fun to write and configure, and I believe others out in the community could benefit from it as I saw very few smart apps related to presence sensors themselves.
2016-07-12 13:43:34 -05:00

View File

@@ -0,0 +1,251 @@
/**
* Smart Family Presence
*
* Copyright 2016 Darin Spivey
*
* 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: "Smart Family Presence",
namespace: "ddspivey",
author: "Darin Spivey",
description: "Smart arrival and departure push messages for couples/families that are traveling together. When family members arrive and depart together, there is no need to send an individual push alert for each.",
category: "Family",
iconUrl: "http://cdn.device-icons.smartthings.com/Home/home4-icn@2x.png",
iconX2Url: "http://cdn.device-icons.smartthings.com/Home/home4-icn@2x.png",
iconX3Url: "http://cdn.device-icons.smartthings.com/Home/home4-icn@2x.png")
preferences {
section("Family Members") {
input "familySensors", "capability.presenceSensor", required: true, title: "Who's in your family?", multiple: true
}
section("Threshold") {
paragraph "Set the time in seconds to allow for group arrival/departure"
input "timeThreshold", "text", required: false, title: "Default is ${defaultThreshold}."
}
section("Smart departure alerts") {
paragraph "When family members are home together, departure push alerts may not be necessary because most of the time, people are aware when their family members are leaving. This feature will only send a push alert if the the entire family was previously apart."
paragraph "For example, when my wife and I are home together, I know when she's leaving; I don't need an alert for that. If this feature is off, it will send a departure alert when she leaves."
input("smartDepartureFeature", "enum", title: "Default is On.", default:"On", options: ["On","Off"])
}
section("Verbose logging") {
paragraph "For debugging, you may log all the app's decisions to the notifications log."
input("logToNotifications", "enum", title: "Default is No.", default:"No", options: ["Yes", "No" ])
}
}
/****************************
Auto-getters and setters
*****************************/
def getDefaultThreshold() {
60
}
def setInProgress(value) {
state.inProgress = value
}
def getInProgress() {
state.inProgress == true
}
def getSmartDeparture() {
settings.smartDepartureFeature == 'On'
}
def getLogToNotifications() {
settings.logToNotifications == 'Yes'
}
def getThreshold() {
settings.timeThreshold ? settings.timeThreshold.toInteger() : defaultThreshold
}
/****************************
Framework methods
*****************************/
def installed() {
initialize()
}
def logit(msg) {
log.debug msg
if (logToNotifications) {
sendNotificationEvent("[Smart Family Presense] $msg")
}
}
def updated() {
unsubscribe()
initialize()
}
def initialize() {
subscribe(familySensors, "presence", presenceHandler)
log.debug("Subscribed ${familySensors.toString()} to presenceHandler")
/*
Regular usage shows that, during certain cases such as the hub going offline,
or power outages, the app may lose state and not send alerts. This will ensure that
it re-evaluates its state at least once per hour.
*/
logit "Scheduling a re-calibration at the top of every hour."
schedule("0 0 0/1 1/1 * ? *", reset)
reset()
}
/*****************************
Smart Family Presence methods
******************************/
def reset() {
if (inProgress) {
logit "Skipping re-calibration, execution in progress!"
return
}
logit "Re-calibrating."
state.baseCase = null
state.changedThisTime = []
wasApart()
}
def isFamilyTogether() {
// Check to see if the entire family has arrived/departed together
if (state.changedThisTime.size() == 0) {
logit "No changes."
reset()
return
}
logit "People who changed presence: ${state.changedThisTime}"
def theirState = state.baseCase
def notTogether = statusNotEquals(theirState)
if (notTogether) {
// The family is not together, send an alert as normal
logit "${notTogether.join(", ")} is not with the rest of the family (who are $theirState)"
sendPushAlert()
}
else {
/*
Special case - When everyone is gone, but they *previously* weren't together,
then technically they were apart to begin with and are still apart upon leaving
*/
if (state.wasApart && theirState == 'not present') {
logit "Family was previously apart and now all gone. Alert."
sendPushAlert()
}
else {
logit "OK! Everyone has arrived/departed together. The family is $theirState"
}
}
inProgress = false
reset()
}
def wasApart() {
// This is true if everyone is gone, or some were home
def allGone = statusEquals('not present') == familySensors
def someHome = statusEquals('present') != familySensors
if (allGone || someHome) {
state.wasApart = true
}
else {
state.wasApart = false
}
}
def presenceHandler(evt) {
def person = evt.displayName
logit "Presence Event: $person is $evt.value"
if (! inProgress) {
inProgress = true
state.baseCase = evt.value
logit "First person sensed. Checking for others to be $evt.value in ${threshold} seconds"
runIn(threshold, isFamilyTogether, [overwrite: false])
}
if (! state.changedThisTime.contains(person)) {
state.changedThisTime.push person
}
else {
// Special case - presence has changed within the threshold. Remove this person.
state.changedThisTime = state.changedThisTime - person
logit "Ignoring flapping presence event for $person"
}
}
def statusEquals(status) {
if (status == null) return
familySensors.findAll {
it.currentPresence == status
}
}
def statusNotEquals(status) {
if (status == null) return
familySensors.findAll {
it.currentPresence != status
}
}
def sendPushAlert() {
def baseCase = state.baseCase
def changedPeople = state.changedThisTime
if (baseCase == 'not present' && state.wasApart == false && smartDeparture) {
logit "Not sending departure alert because smartDeparture is $smartDeparture"
return
}
def statuses = [ 'present':[], 'not present':[] ]
for (sensor in familySensors) {
def person = sensor.toString()
if (changedPeople.contains(person)) {
def currentState = sensor.currentPresence
log.debug "$person is now $currentState"
statuses[currentState].push person
}
}
logit "Statuses: $statuses"
// Construct the message payload
def pushMsg = ""
def home = statuses.present
def notHome = statuses['not present']
String adVerb;
if (home.size() > 0) {
adVerb = home.size > 1 ? "have" : "has"
pushMsg += "${home.join(", ")} $adVerb arrived $location.name. "
}
if (notHome.size() > 0) {
adVerb = notHome.size > 1 ? "have" : "has"
pushMsg += "${notHome.join(", ")} $adVerb left $location.name"
}
sendPush pushMsg
}