core/homeassistant/helpers/event.py

302 lines
9.6 KiB
Python
Raw Normal View History

2016-03-07 22:39:52 +00:00
"""Helpers for listening to events."""
import asyncio
import functools as ft
2016-02-19 05:27:50 +00:00
from datetime import timedelta
2016-10-01 08:22:13 +00:00
from ..core import HomeAssistant
from ..const import (
ATTR_NOW, EVENT_STATE_CHANGED, EVENT_TIME_CHANGED, MATCH_ALL)
2016-02-19 05:27:50 +00:00
from ..util import dt as dt_util
from ..util.async import run_callback_threadsafe
2016-10-01 08:22:13 +00:00
# PyLint does not like the use of _threaded_factory
# pylint: disable=invalid-name
2016-03-07 22:39:52 +00:00
2016-10-01 08:22:13 +00:00
def _threaded_factory(async_factory):
"""Convert an async event helper to a threaded one."""
@ft.wraps(async_factory)
def factory(*args, **kwargs):
"""Call async event helper safely."""
hass = args[0]
2016-10-01 08:22:13 +00:00
if not isinstance(hass, HomeAssistant):
raise TypeError('First parameter needs to be a hass instance')
async_remove = run_callback_threadsafe(
hass.loop, ft.partial(async_factory, *args, **kwargs)).result()
2016-10-01 08:22:13 +00:00
def remove():
"""Threadsafe removal."""
run_callback_threadsafe(hass.loop, async_remove).result()
2016-10-01 08:22:13 +00:00
return remove
return factory
def async_track_state_change(hass, entity_ids, action, from_state=None,
to_state=None):
"""Track specific state changes.
entity_ids, from_state and to_state can be string or list.
Use list to match multiple.
Returns a function that can be called to remove the listener.
Must be run within the event loop.
"""
2016-06-13 03:37:33 +00:00
from_state = _process_state_match(from_state)
to_state = _process_state_match(to_state)
# Ensure it is a lowercase list with entity ids we want to match on
if entity_ids == MATCH_ALL:
pass
elif isinstance(entity_ids, str):
entity_ids = (entity_ids.lower(),)
else:
entity_ids = tuple(entity_id.lower() for entity_id in entity_ids)
@ft.wraps(action)
@asyncio.coroutine
def state_change_listener(event):
2016-03-07 22:39:52 +00:00
"""The listener that listens for specific state changes."""
if entity_ids != MATCH_ALL and \
2016-05-30 17:19:12 +00:00
event.data.get('entity_id') not in entity_ids:
return
2016-05-30 17:19:12 +00:00
if event.data.get('old_state') is not None:
2016-02-14 06:57:40 +00:00
old_state = event.data['old_state'].state
else:
2016-05-30 17:19:12 +00:00
old_state = None
if event.data.get('new_state') is not None:
2016-02-14 06:57:40 +00:00
new_state = event.data['new_state'].state
2016-05-30 17:19:12 +00:00
else:
new_state = None
2016-02-14 06:57:40 +00:00
if _matcher(old_state, from_state) and _matcher(new_state, to_state):
hass.async_add_job(action, event.data.get('entity_id'),
event.data.get('old_state'),
event.data.get('new_state'))
return hass.bus.async_listen(EVENT_STATE_CHANGED, state_change_listener)
2016-10-01 08:22:13 +00:00
track_state_change = _threaded_factory(async_track_state_change)
def async_track_point_in_time(hass, action, point_in_time):
2016-03-07 22:39:52 +00:00
"""Add a listener that fires once after a spefic point in time."""
utc_point_in_time = dt_util.as_utc(point_in_time)
@ft.wraps(action)
@asyncio.coroutine
def utc_converter(utc_now):
2016-03-07 22:39:52 +00:00
"""Convert passed in UTC now to local now."""
hass.async_add_job(action, dt_util.as_local(utc_now))
2016-10-01 08:22:13 +00:00
return async_track_point_in_utc_time(hass, utc_converter,
utc_point_in_time)
2016-10-01 08:22:13 +00:00
track_point_in_time = _threaded_factory(async_track_point_in_time)
def async_track_point_in_utc_time(hass, action, point_in_time):
2016-03-07 22:39:52 +00:00
"""Add a listener that fires once after a specific point in UTC time."""
2015-08-04 16:13:55 +00:00
# Ensure point_in_time is UTC
point_in_time = dt_util.as_utc(point_in_time)
@ft.wraps(action)
@asyncio.coroutine
def point_in_time_listener(event):
2016-03-07 22:39:52 +00:00
"""Listen for matching time_changed events."""
now = event.data[ATTR_NOW]
2016-08-26 06:25:35 +00:00
if now < point_in_time or hasattr(point_in_time_listener, 'run'):
return
2016-08-26 06:25:35 +00:00
# Set variable so that we will never run twice.
# Because the event bus might have to wait till a thread comes
# available to execute this listener it might occur that the
# listener gets lined up twice to be executed. This will make
# sure the second time it does nothing.
point_in_time_listener.run = True
async_unsub()
hass.async_add_job(action, now)
async_unsub = hass.bus.async_listen(EVENT_TIME_CHANGED,
point_in_time_listener)
return async_unsub
2016-10-01 08:22:13 +00:00
track_point_in_utc_time = _threaded_factory(async_track_point_in_utc_time)
def async_track_sunrise(hass, action, offset=None):
2016-03-07 22:39:52 +00:00
"""Add a listener that will fire a specified offset from sunrise daily."""
from homeassistant.components import sun
offset = offset or timedelta()
def next_rise():
2016-03-07 22:39:52 +00:00
"""Return the next sunrise."""
next_time = sun.next_rising_utc(hass) + offset
while next_time < dt_util.utcnow():
next_time = next_time + timedelta(days=1)
return next_time
2016-10-01 08:22:13 +00:00
@ft.wraps(action)
@asyncio.coroutine
def sunrise_automation_listener(now):
2016-03-07 22:39:52 +00:00
"""Called when it's time for action."""
2016-08-26 06:25:35 +00:00
nonlocal remove
remove = async_track_point_in_utc_time(
hass, sunrise_automation_listener, next_rise())
hass.async_add_job(action)
2016-10-01 08:22:13 +00:00
remove = async_track_point_in_utc_time(
hass, sunrise_automation_listener, next_rise())
2016-08-26 06:25:35 +00:00
def remove_listener():
"""Remove sunset listener."""
2016-10-01 08:22:13 +00:00
remove()
2016-08-26 06:25:35 +00:00
return remove_listener
2016-10-01 08:22:13 +00:00
track_sunrise = _threaded_factory(async_track_sunrise)
def async_track_sunset(hass, action, offset=None):
2016-03-07 22:39:52 +00:00
"""Add a listener that will fire a specified offset from sunset daily."""
from homeassistant.components import sun
offset = offset or timedelta()
def next_set():
2016-03-07 22:39:52 +00:00
"""Return next sunrise."""
next_time = sun.next_setting_utc(hass) + offset
while next_time < dt_util.utcnow():
next_time = next_time + timedelta(days=1)
return next_time
2016-10-01 08:22:13 +00:00
@ft.wraps(action)
@asyncio.coroutine
def sunset_automation_listener(now):
2016-03-07 22:39:52 +00:00
"""Called when it's time for action."""
2016-08-26 06:25:35 +00:00
nonlocal remove
remove = async_track_point_in_utc_time(
hass, sunset_automation_listener, next_set())
hass.async_add_job(action)
2016-10-01 08:22:13 +00:00
remove = async_track_point_in_utc_time(
hass, sunset_automation_listener, next_set())
2016-08-26 06:25:35 +00:00
def remove_listener():
"""Remove sunset listener."""
2016-10-01 08:22:13 +00:00
remove()
2016-08-26 06:25:35 +00:00
return remove_listener
2016-10-01 08:22:13 +00:00
track_sunset = _threaded_factory(async_track_sunset)
# pylint: disable=too-many-arguments
2016-10-01 08:22:13 +00:00
def async_track_utc_time_change(hass, action, year=None, month=None, day=None,
hour=None, minute=None, second=None,
local=False):
2016-03-07 22:39:52 +00:00
"""Add a listener that will fire if time matches a pattern."""
# We do not have to wrap the function with time pattern matching logic
# if no pattern given
if all(val is None for val in (year, month, day, hour, minute, second)):
@ft.wraps(action)
def time_change_listener(event):
2016-03-07 22:39:52 +00:00
"""Fire every time event that comes in."""
action(event.data[ATTR_NOW])
2016-10-01 08:22:13 +00:00
return hass.bus.async_listen(EVENT_TIME_CHANGED, time_change_listener)
2016-06-13 03:37:33 +00:00
pmp = _process_time_match
year, month, day = pmp(year), pmp(month), pmp(day)
hour, minute, second = pmp(hour), pmp(minute), pmp(second)
@ft.wraps(action)
@asyncio.coroutine
def pattern_time_change_listener(event):
2016-03-07 22:39:52 +00:00
"""Listen for matching time_changed events."""
now = event.data[ATTR_NOW]
if local:
now = dt_util.as_local(now)
mat = _matcher
2015-11-29 21:49:05 +00:00
# pylint: disable=too-many-boolean-expressions
if mat(now.year, year) and \
mat(now.month, month) and \
mat(now.day, day) and \
mat(now.hour, hour) and \
mat(now.minute, minute) and \
mat(now.second, second):
hass.async_add_job(action, now)
2016-10-01 08:22:13 +00:00
return hass.bus.async_listen(EVENT_TIME_CHANGED,
pattern_time_change_listener)
track_utc_time_change = _threaded_factory(async_track_utc_time_change)
# pylint: disable=too-many-arguments
2016-10-01 08:22:13 +00:00
def async_track_time_change(hass, action, year=None, month=None, day=None,
hour=None, minute=None, second=None):
2016-03-07 22:39:52 +00:00
"""Add a listener that will fire if UTC time matches a pattern."""
2016-10-01 08:22:13 +00:00
return async_track_utc_time_change(hass, action, year, month, day, hour,
minute, second, local=True)
track_time_change = _threaded_factory(async_track_time_change)
2016-06-13 03:37:33 +00:00
def _process_state_match(parameter):
"""Wrap parameter in a tuple if it is not one and returns it."""
if parameter is None or parameter == MATCH_ALL:
return MATCH_ALL
elif isinstance(parameter, str) or not hasattr(parameter, '__iter__'):
return (parameter,)
else:
return tuple(parameter)
def _process_time_match(parameter):
2016-03-07 22:39:52 +00:00
"""Wrap parameter in a tuple if it is not one and returns it."""
if parameter is None or parameter == MATCH_ALL:
return MATCH_ALL
2016-01-26 20:38:07 +00:00
elif isinstance(parameter, str) and parameter.startswith('/'):
return parameter
elif isinstance(parameter, str) or not hasattr(parameter, '__iter__'):
return (parameter,)
else:
return tuple(parameter)
def _matcher(subject, pattern):
2016-03-07 22:39:52 +00:00
"""Return True if subject matches the pattern.
Pattern is either a tuple of allowed subjects or a `MATCH_ALL`.
"""
2016-01-26 20:38:07 +00:00
if isinstance(pattern, str) and pattern.startswith('/'):
try:
2016-01-26 20:38:07 +00:00
return subject % float(pattern.lstrip('/')) == 0
except ValueError:
return False
return MATCH_ALL == pattern or subject in pattern