2019-02-13 20:21:14 +00:00
|
|
|
"""Support for repeating alerts when conditions are met."""
|
2017-02-03 05:20:51 +00:00
|
|
|
import asyncio
|
|
|
|
import logging
|
2019-01-23 07:47:37 +00:00
|
|
|
from datetime import datetime, timedelta
|
2017-02-03 05:20:51 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-01-23 07:47:37 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-10-05 21:09:55 +00:00
|
|
|
from homeassistant.components.notify import (
|
2019-01-23 07:47:37 +00:00
|
|
|
ATTR_MESSAGE, ATTR_TITLE, ATTR_DATA, DOMAIN as DOMAIN_NOTIFY)
|
2017-02-11 19:29:37 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_ENTITY_ID, STATE_IDLE, CONF_NAME, CONF_STATE, STATE_ON, STATE_OFF,
|
|
|
|
SERVICE_TURN_ON, SERVICE_TURN_OFF, SERVICE_TOGGLE, ATTR_ENTITY_ID)
|
2017-02-03 05:20:51 +00:00
|
|
|
from homeassistant.helpers import service, event
|
2019-01-23 07:47:37 +00:00
|
|
|
from homeassistant.helpers.entity import ToggleEntity
|
2017-02-03 05:20:51 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
DOMAIN = 'alert'
|
|
|
|
ENTITY_ID_FORMAT = DOMAIN + '.{}'
|
|
|
|
|
|
|
|
CONF_CAN_ACK = 'can_acknowledge'
|
|
|
|
CONF_NOTIFIERS = 'notifiers'
|
|
|
|
CONF_REPEAT = 'repeat'
|
|
|
|
CONF_SKIP_FIRST = 'skip_first'
|
2018-11-01 08:48:11 +00:00
|
|
|
CONF_ALERT_MESSAGE = 'message'
|
|
|
|
CONF_DONE_MESSAGE = 'done_message'
|
2019-01-23 07:47:37 +00:00
|
|
|
CONF_TITLE = 'title'
|
|
|
|
CONF_DATA = 'data'
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2017-02-11 19:29:37 +00:00
|
|
|
DEFAULT_CAN_ACK = True
|
|
|
|
DEFAULT_SKIP_FIRST = False
|
|
|
|
|
2017-02-03 05:20:51 +00:00
|
|
|
ALERT_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Required(CONF_ENTITY_ID): cv.entity_id,
|
|
|
|
vol.Required(CONF_STATE, default=STATE_ON): cv.string,
|
|
|
|
vol.Required(CONF_REPEAT): vol.All(cv.ensure_list, [vol.Coerce(float)]),
|
2017-02-11 19:29:37 +00:00
|
|
|
vol.Required(CONF_CAN_ACK, default=DEFAULT_CAN_ACK): cv.boolean,
|
|
|
|
vol.Required(CONF_SKIP_FIRST, default=DEFAULT_SKIP_FIRST): cv.boolean,
|
2018-11-01 08:48:11 +00:00
|
|
|
vol.Optional(CONF_ALERT_MESSAGE): cv.template,
|
|
|
|
vol.Optional(CONF_DONE_MESSAGE): cv.template,
|
2019-01-23 07:47:37 +00:00
|
|
|
vol.Optional(CONF_TITLE): cv.template,
|
|
|
|
vol.Optional(CONF_DATA): dict,
|
2017-02-03 05:20:51 +00:00
|
|
|
vol.Required(CONF_NOTIFIERS): cv.ensure_list})
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
2019-01-22 00:36:04 +00:00
|
|
|
DOMAIN: cv.schema_with_slug_keys(ALERT_SCHEMA),
|
2017-02-03 05:20:51 +00:00
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
ALERT_SERVICE_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_ENTITY_ID): cv.entity_ids,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
def is_on(hass, entity_id):
|
|
|
|
"""Return if the alert is firing and not acknowledged."""
|
|
|
|
return hass.states.is_state(entity_id, STATE_ON)
|
|
|
|
|
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def async_setup(hass, config):
|
2017-02-11 19:29:37 +00:00
|
|
|
"""Set up the Alert component."""
|
2018-11-01 08:48:11 +00:00
|
|
|
entities = []
|
|
|
|
|
|
|
|
for object_id, cfg in config[DOMAIN].items():
|
|
|
|
if not cfg:
|
|
|
|
cfg = {}
|
|
|
|
|
|
|
|
name = cfg.get(CONF_NAME)
|
|
|
|
watched_entity_id = cfg.get(CONF_ENTITY_ID)
|
|
|
|
alert_state = cfg.get(CONF_STATE)
|
|
|
|
repeat = cfg.get(CONF_REPEAT)
|
|
|
|
skip_first = cfg.get(CONF_SKIP_FIRST)
|
|
|
|
message_template = cfg.get(CONF_ALERT_MESSAGE)
|
|
|
|
done_message_template = cfg.get(CONF_DONE_MESSAGE)
|
|
|
|
notifiers = cfg.get(CONF_NOTIFIERS)
|
|
|
|
can_ack = cfg.get(CONF_CAN_ACK)
|
2019-01-23 07:47:37 +00:00
|
|
|
title_template = cfg.get(CONF_TITLE)
|
|
|
|
data = cfg.get(CONF_DATA)
|
2018-11-01 08:48:11 +00:00
|
|
|
|
|
|
|
entities.append(Alert(hass, object_id, name,
|
|
|
|
watched_entity_id, alert_state, repeat,
|
|
|
|
skip_first, message_template,
|
|
|
|
done_message_template, notifiers,
|
2019-01-23 07:47:37 +00:00
|
|
|
can_ack, title_template, data))
|
2018-11-01 08:48:11 +00:00
|
|
|
|
|
|
|
if not entities:
|
|
|
|
return False
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def async_handle_alert_service(service_call):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Handle calls to alert services."""
|
|
|
|
alert_ids = service.extract_entity_ids(hass, service_call)
|
|
|
|
|
|
|
|
for alert_id in alert_ids:
|
2018-11-01 08:48:11 +00:00
|
|
|
for alert in entities:
|
|
|
|
if alert.entity_id != alert_id:
|
|
|
|
continue
|
|
|
|
|
|
|
|
alert.async_set_context(service_call.context)
|
|
|
|
if service_call.service == SERVICE_TURN_ON:
|
|
|
|
await alert.async_turn_on()
|
|
|
|
elif service_call.service == SERVICE_TOGGLE:
|
|
|
|
await alert.async_toggle()
|
|
|
|
else:
|
|
|
|
await alert.async_turn_off()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2017-02-11 19:29:37 +00:00
|
|
|
# Setup service calls
|
2017-02-03 05:20:51 +00:00
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_TURN_OFF, async_handle_alert_service,
|
2018-01-07 22:54:16 +00:00
|
|
|
schema=ALERT_SERVICE_SCHEMA)
|
2017-02-03 05:20:51 +00:00
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_TURN_ON, async_handle_alert_service,
|
2018-01-07 22:54:16 +00:00
|
|
|
schema=ALERT_SERVICE_SCHEMA)
|
2017-02-03 05:20:51 +00:00
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_TOGGLE, async_handle_alert_service,
|
2018-01-07 22:54:16 +00:00
|
|
|
schema=ALERT_SERVICE_SCHEMA)
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-11-01 08:48:11 +00:00
|
|
|
tasks = [alert.async_update_ha_state() for alert in entities]
|
2017-02-03 05:20:51 +00:00
|
|
|
if tasks:
|
2018-10-01 12:44:11 +00:00
|
|
|
await asyncio.wait(tasks, loop=hass.loop)
|
2017-02-03 05:20:51 +00:00
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class Alert(ToggleEntity):
|
|
|
|
"""Representation of an alert."""
|
|
|
|
|
2018-11-01 08:48:11 +00:00
|
|
|
def __init__(self, hass, entity_id, name, watched_entity_id,
|
|
|
|
state, repeat, skip_first, message_template,
|
2019-01-23 07:47:37 +00:00
|
|
|
done_message_template, notifiers, can_ack, title_template,
|
|
|
|
data):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Initialize the alert."""
|
|
|
|
self.hass = hass
|
|
|
|
self._name = name
|
|
|
|
self._alert_state = state
|
|
|
|
self._skip_first = skip_first
|
2019-01-23 07:47:37 +00:00
|
|
|
self._data = data
|
2018-11-01 08:48:11 +00:00
|
|
|
|
|
|
|
self._message_template = message_template
|
|
|
|
if self._message_template is not None:
|
|
|
|
self._message_template.hass = hass
|
|
|
|
|
|
|
|
self._done_message_template = done_message_template
|
|
|
|
if self._done_message_template is not None:
|
|
|
|
self._done_message_template.hass = hass
|
|
|
|
|
2019-01-23 07:47:37 +00:00
|
|
|
self._title_template = title_template
|
|
|
|
if self._title_template is not None:
|
|
|
|
self._title_template.hass = hass
|
|
|
|
|
2017-02-03 05:20:51 +00:00
|
|
|
self._notifiers = notifiers
|
|
|
|
self._can_ack = can_ack
|
|
|
|
|
|
|
|
self._delay = [timedelta(minutes=val) for val in repeat]
|
|
|
|
self._next_delay = 0
|
|
|
|
|
|
|
|
self._firing = False
|
|
|
|
self._ack = False
|
|
|
|
self._cancel = None
|
2017-07-01 04:07:12 +00:00
|
|
|
self._send_done_message = False
|
2017-02-03 05:20:51 +00:00
|
|
|
self.entity_id = ENTITY_ID_FORMAT.format(entity_id)
|
|
|
|
|
2017-02-11 19:29:37 +00:00
|
|
|
event.async_track_state_change(
|
|
|
|
hass, watched_entity_id, self.watched_entity_change)
|
2017-02-03 05:20:51 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the alert."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""HASS need not poll these entities."""
|
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the alert status."""
|
|
|
|
if self._firing:
|
|
|
|
if self._ack:
|
|
|
|
return STATE_OFF
|
|
|
|
return STATE_ON
|
|
|
|
return STATE_IDLE
|
|
|
|
|
|
|
|
@property
|
|
|
|
def hidden(self):
|
|
|
|
"""Hide the alert when it is not firing."""
|
|
|
|
return not self._can_ack or not self._firing
|
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def watched_entity_change(self, entity, from_state, to_state):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Determine if the alert should start or stop."""
|
2017-02-11 19:29:37 +00:00
|
|
|
_LOGGER.debug("Watched entity (%s) has changed", entity)
|
2017-02-03 05:20:51 +00:00
|
|
|
if to_state.state == self._alert_state and not self._firing:
|
2018-10-01 12:44:11 +00:00
|
|
|
await self.begin_alerting()
|
2017-02-03 05:20:51 +00:00
|
|
|
if to_state.state != self._alert_state and self._firing:
|
2018-10-01 12:44:11 +00:00
|
|
|
await self.end_alerting()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def begin_alerting(self):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Begin the alert procedures."""
|
2017-02-11 19:29:37 +00:00
|
|
|
_LOGGER.debug("Beginning Alert: %s", self._name)
|
2017-02-03 05:20:51 +00:00
|
|
|
self._ack = False
|
|
|
|
self._firing = True
|
|
|
|
self._next_delay = 0
|
|
|
|
|
|
|
|
if not self._skip_first:
|
2018-10-01 12:44:11 +00:00
|
|
|
await self._notify()
|
2017-02-03 05:20:51 +00:00
|
|
|
else:
|
2018-10-01 12:44:11 +00:00
|
|
|
await self._schedule_notify()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-07-23 12:05:38 +00:00
|
|
|
self.async_schedule_update_ha_state()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def end_alerting(self):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""End the alert procedures."""
|
2017-02-11 19:29:37 +00:00
|
|
|
_LOGGER.debug("Ending Alert: %s", self._name)
|
2017-02-03 05:20:51 +00:00
|
|
|
self._cancel()
|
|
|
|
self._ack = False
|
|
|
|
self._firing = False
|
2018-11-01 08:48:11 +00:00
|
|
|
if self._send_done_message:
|
2018-10-01 12:44:11 +00:00
|
|
|
await self._notify_done_message()
|
2018-07-23 12:05:38 +00:00
|
|
|
self.async_schedule_update_ha_state()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def _schedule_notify(self):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Schedule a notification."""
|
|
|
|
delay = self._delay[self._next_delay]
|
|
|
|
next_msg = datetime.now() + delay
|
|
|
|
self._cancel = \
|
|
|
|
event.async_track_point_in_time(self.hass, self._notify, next_msg)
|
|
|
|
self._next_delay = min(self._next_delay + 1, len(self._delay) - 1)
|
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def _notify(self, *args):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Send the alert notification."""
|
|
|
|
if not self._firing:
|
|
|
|
return
|
|
|
|
|
|
|
|
if not self._ack:
|
2017-02-11 19:29:37 +00:00
|
|
|
_LOGGER.info("Alerting: %s", self._name)
|
2017-07-01 04:07:12 +00:00
|
|
|
self._send_done_message = True
|
2018-11-01 08:48:11 +00:00
|
|
|
|
|
|
|
if self._message_template is not None:
|
|
|
|
message = self._message_template.async_render()
|
|
|
|
else:
|
|
|
|
message = self._name
|
|
|
|
|
|
|
|
await self._send_notification_message(message)
|
2018-10-01 12:44:11 +00:00
|
|
|
await self._schedule_notify()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def _notify_done_message(self, *args):
|
2017-07-01 04:07:12 +00:00
|
|
|
"""Send notification of complete alert."""
|
2018-11-01 08:48:11 +00:00
|
|
|
_LOGGER.info("Alerting: %s", self._done_message_template)
|
2017-07-01 04:07:12 +00:00
|
|
|
self._send_done_message = False
|
2018-11-01 08:48:11 +00:00
|
|
|
|
|
|
|
if self._done_message_template is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
message = self._done_message_template.async_render()
|
|
|
|
|
|
|
|
await self._send_notification_message(message)
|
|
|
|
|
|
|
|
async def _send_notification_message(self, message):
|
2019-01-23 07:47:37 +00:00
|
|
|
|
|
|
|
msg_payload = {ATTR_MESSAGE: message}
|
|
|
|
|
|
|
|
if self._title_template is not None:
|
|
|
|
title = self._title_template.async_render()
|
|
|
|
msg_payload.update({ATTR_TITLE: title})
|
|
|
|
if self._data:
|
|
|
|
msg_payload.update({ATTR_DATA: self._data})
|
|
|
|
|
|
|
|
_LOGGER.debug(msg_payload)
|
|
|
|
|
2017-07-01 04:07:12 +00:00
|
|
|
for target in self._notifiers:
|
2018-10-01 12:44:11 +00:00
|
|
|
await self.hass.services.async_call(
|
2019-01-23 07:47:37 +00:00
|
|
|
DOMAIN_NOTIFY, target, msg_payload)
|
2017-07-01 04:07:12 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def async_turn_on(self, **kwargs):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Async Unacknowledge alert."""
|
2017-02-11 19:29:37 +00:00
|
|
|
_LOGGER.debug("Reset Alert: %s", self._name)
|
2017-02-03 05:20:51 +00:00
|
|
|
self._ack = False
|
2018-10-01 12:44:11 +00:00
|
|
|
await self.async_update_ha_state()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def async_turn_off(self, **kwargs):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Async Acknowledge alert."""
|
2017-02-11 19:29:37 +00:00
|
|
|
_LOGGER.debug("Acknowledged Alert: %s", self._name)
|
2017-02-03 05:20:51 +00:00
|
|
|
self._ack = True
|
2018-10-01 12:44:11 +00:00
|
|
|
await self.async_update_ha_state()
|
2017-02-03 05:20:51 +00:00
|
|
|
|
2018-10-01 12:44:11 +00:00
|
|
|
async def async_toggle(self, **kwargs):
|
2017-02-03 05:20:51 +00:00
|
|
|
"""Async toggle alert."""
|
|
|
|
if self._ack:
|
2018-10-01 12:44:11 +00:00
|
|
|
return await self.async_turn_on()
|
|
|
|
return await self.async_turn_off()
|