core/tests/helpers/test_event.py

452 lines
15 KiB
Python
Raw Normal View History

2016-03-09 09:25:50 +00:00
"""Test event helpers."""
# pylint: disable=protected-access
import asyncio
import unittest
2016-02-14 21:07:21 +00:00
from datetime import datetime, timedelta
from astral import Astral
from homeassistant.bootstrap import setup_component
2015-08-17 03:44:46 +00:00
import homeassistant.core as ha
from homeassistant.const import MATCH_ALL
2016-02-14 21:07:21 +00:00
from homeassistant.helpers.event import (
track_point_in_utc_time,
track_point_in_time,
track_utc_time_change,
track_time_change,
track_state_change,
track_time_interval,
2016-02-14 21:07:21 +00:00
track_sunrise,
track_sunset,
)
from homeassistant.components import sun
2016-02-14 21:07:21 +00:00
import homeassistant.util.dt as dt_util
2016-02-14 23:08:23 +00:00
from tests.common import get_test_home_assistant
class TestEventHelpers(unittest.TestCase):
2016-03-09 09:25:50 +00:00
"""Test the Home Assistant event helpers."""
# pylint: disable=invalid-name
def setUp(self):
2016-03-09 09:25:50 +00:00
"""Setup things to be run when tests are started."""
2016-02-14 23:08:23 +00:00
self.hass = get_test_home_assistant()
# pylint: disable=invalid-name
def tearDown(self):
2016-03-09 09:25:50 +00:00
"""Stop everything that was started."""
self.hass.stop()
def test_track_point_in_time(self):
2016-03-09 09:25:50 +00:00
"""Test track point in time."""
2015-08-04 16:16:10 +00:00
before_birthday = datetime(1985, 7, 9, 12, 0, 0, tzinfo=dt_util.UTC)
birthday_paulus = datetime(1986, 7, 9, 12, 0, 0, tzinfo=dt_util.UTC)
after_birthday = datetime(1987, 7, 9, 12, 0, 0, tzinfo=dt_util.UTC)
runs = []
track_point_in_utc_time(
self.hass, lambda x: runs.append(1), birthday_paulus)
self._send_time_changed(before_birthday)
self.hass.block_till_done()
self.assertEqual(0, len(runs))
self._send_time_changed(birthday_paulus)
self.hass.block_till_done()
self.assertEqual(1, len(runs))
# A point in time tracker will only fire once, this should do nothing
self._send_time_changed(birthday_paulus)
self.hass.block_till_done()
self.assertEqual(1, len(runs))
2015-08-04 16:16:10 +00:00
track_point_in_time(
self.hass, lambda x: runs.append(1), birthday_paulus)
self._send_time_changed(after_birthday)
self.hass.block_till_done()
self.assertEqual(2, len(runs))
2016-08-26 06:25:35 +00:00
unsub = track_point_in_time(
self.hass, lambda x: runs.append(1), birthday_paulus)
unsub()
self._send_time_changed(after_birthday)
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(2, len(runs))
def test_track_time_change(self):
2016-03-09 09:25:50 +00:00
"""Test tracking time change."""
wildcard_runs = []
specific_runs = []
2016-08-26 06:25:35 +00:00
unsub = track_time_change(self.hass, lambda x: wildcard_runs.append(1))
unsub_utc = track_utc_time_change(
self.hass, lambda x: specific_runs.append(1), second=[0, 30])
self._send_time_changed(datetime(2014, 5, 24, 12, 0, 0))
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self.assertEqual(1, len(wildcard_runs))
self._send_time_changed(datetime(2014, 5, 24, 12, 0, 15))
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self.assertEqual(2, len(wildcard_runs))
self._send_time_changed(datetime(2014, 5, 24, 12, 0, 30))
self.hass.block_till_done()
self.assertEqual(2, len(specific_runs))
self.assertEqual(3, len(wildcard_runs))
2016-08-26 06:25:35 +00:00
unsub()
unsub_utc()
self._send_time_changed(datetime(2014, 5, 24, 12, 0, 30))
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(2, len(specific_runs))
self.assertEqual(3, len(wildcard_runs))
def test_track_state_change(self):
2016-02-14 06:57:40 +00:00
"""Test track_state_change."""
# 2 lists to track how often our callbacks get called
specific_runs = []
wildcard_runs = []
wildercard_runs = []
def specific_run_callback(entity_id, old_state, new_state):
specific_runs.append(1)
track_state_change(
self.hass, 'light.Bowl', specific_run_callback, 'on', 'off')
@ha.callback
def wildcard_run_callback(entity_id, old_state, new_state):
wildcard_runs.append((old_state, new_state))
track_state_change(self.hass, 'light.Bowl', wildcard_run_callback)
@asyncio.coroutine
def wildercard_run_callback(entity_id, old_state, new_state):
wildercard_runs.append((old_state, new_state))
track_state_change(self.hass, MATCH_ALL, wildercard_run_callback)
2016-02-14 06:57:40 +00:00
# Adding state to state machine
self.hass.states.set("light.Bowl", "on")
self.hass.block_till_done()
2016-02-14 06:57:40 +00:00
self.assertEqual(0, len(specific_runs))
self.assertEqual(1, len(wildcard_runs))
self.assertEqual(1, len(wildercard_runs))
2016-02-14 06:57:40 +00:00
self.assertIsNone(wildcard_runs[-1][0])
self.assertIsNotNone(wildcard_runs[-1][1])
# Set same state should not trigger a state change/listener
self.hass.states.set('light.Bowl', 'on')
self.hass.block_till_done()
self.assertEqual(0, len(specific_runs))
2016-02-14 06:57:40 +00:00
self.assertEqual(1, len(wildcard_runs))
self.assertEqual(1, len(wildercard_runs))
# State change off -> on
self.hass.states.set('light.Bowl', 'off')
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
2016-02-14 06:57:40 +00:00
self.assertEqual(2, len(wildcard_runs))
self.assertEqual(2, len(wildercard_runs))
# State change off -> off
self.hass.states.set('light.Bowl', 'off', {"some_attr": 1})
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
2016-02-14 06:57:40 +00:00
self.assertEqual(3, len(wildcard_runs))
self.assertEqual(3, len(wildercard_runs))
# State change off -> on
self.hass.states.set('light.Bowl', 'on')
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
2016-02-14 06:57:40 +00:00
self.assertEqual(4, len(wildcard_runs))
self.assertEqual(4, len(wildercard_runs))
2016-02-14 06:57:40 +00:00
self.hass.states.remove('light.bowl')
self.hass.block_till_done()
2016-02-14 06:57:40 +00:00
self.assertEqual(1, len(specific_runs))
self.assertEqual(5, len(wildcard_runs))
self.assertEqual(5, len(wildercard_runs))
2016-02-14 06:57:40 +00:00
self.assertIsNotNone(wildcard_runs[-1][0])
self.assertIsNone(wildcard_runs[-1][1])
self.assertIsNotNone(wildercard_runs[-1][0])
self.assertIsNone(wildercard_runs[-1][1])
# Set state for different entity id
self.hass.states.set('switch.kitchen', 'on')
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self.assertEqual(5, len(wildcard_runs))
self.assertEqual(6, len(wildercard_runs))
def test_track_time_interval(self):
"""Test tracking time interval."""
specific_runs = []
utc_now = dt_util.utcnow()
unsub = track_time_interval(
self.hass, lambda x: specific_runs.append(1),
timedelta(seconds=10)
)
self._send_time_changed(utc_now + timedelta(seconds=5))
self.hass.block_till_done()
self.assertEqual(0, len(specific_runs))
self._send_time_changed(utc_now + timedelta(seconds=13))
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self._send_time_changed(utc_now + timedelta(minutes=20))
self.hass.block_till_done()
self.assertEqual(2, len(specific_runs))
unsub()
self._send_time_changed(utc_now + timedelta(seconds=30))
self.hass.block_till_done()
self.assertEqual(2, len(specific_runs))
def test_track_sunrise(self):
2016-03-09 09:25:50 +00:00
"""Test track the sunrise."""
latitude = 32.87336
longitude = 117.22743
2016-03-09 09:25:50 +00:00
# Setup sun component
self.hass.config.latitude = latitude
self.hass.config.longitude = longitude
setup_component(self.hass, sun.DOMAIN, {
sun.DOMAIN: {sun.CONF_ELEVATION: 0}})
2016-03-09 09:25:50 +00:00
# Get next sunrise/sunset
astral = Astral()
utc_now = dt_util.utcnow()
mod = -1
while True:
next_rising = (astral.sunrise_utc(utc_now +
timedelta(days=mod), latitude, longitude))
if next_rising > utc_now:
break
mod += 1
2016-03-09 09:25:50 +00:00
# Track sunrise
runs = []
2016-08-26 06:25:35 +00:00
unsub = track_sunrise(self.hass, lambda: runs.append(1))
offset_runs = []
offset = timedelta(minutes=30)
2016-08-26 06:25:35 +00:00
unsub2 = track_sunrise(self.hass, lambda: offset_runs.append(1),
offset)
# run tests
self._send_time_changed(next_rising - offset)
self.hass.block_till_done()
self.assertEqual(0, len(runs))
self.assertEqual(0, len(offset_runs))
self._send_time_changed(next_rising)
self.hass.block_till_done()
self.assertEqual(1, len(runs))
self.assertEqual(0, len(offset_runs))
self._send_time_changed(next_rising + offset)
self.hass.block_till_done()
self.assertEqual(2, len(runs))
self.assertEqual(1, len(offset_runs))
2016-08-26 06:25:35 +00:00
unsub()
unsub2()
self._send_time_changed(next_rising + offset)
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(2, len(runs))
self.assertEqual(1, len(offset_runs))
def test_track_sunset(self):
2016-03-09 09:25:50 +00:00
"""Test track the sunset."""
latitude = 32.87336
longitude = 117.22743
2016-03-09 09:25:50 +00:00
# Setup sun component
self.hass.config.latitude = latitude
self.hass.config.longitude = longitude
setup_component(self.hass, sun.DOMAIN, {
sun.DOMAIN: {sun.CONF_ELEVATION: 0}})
2016-03-09 09:25:50 +00:00
# Get next sunrise/sunset
astral = Astral()
utc_now = dt_util.utcnow()
mod = -1
while True:
next_setting = (astral.sunset_utc(utc_now +
timedelta(days=mod), latitude, longitude))
if next_setting > utc_now:
break
mod += 1
2016-03-09 09:25:50 +00:00
# Track sunset
runs = []
2016-08-26 06:25:35 +00:00
unsub = track_sunset(self.hass, lambda: runs.append(1))
offset_runs = []
offset = timedelta(minutes=30)
2016-08-26 06:25:35 +00:00
unsub2 = track_sunset(self.hass, lambda: offset_runs.append(1), offset)
2016-03-09 09:25:50 +00:00
# Run tests
self._send_time_changed(next_setting - offset)
self.hass.block_till_done()
self.assertEqual(0, len(runs))
self.assertEqual(0, len(offset_runs))
self._send_time_changed(next_setting)
self.hass.block_till_done()
self.assertEqual(1, len(runs))
self.assertEqual(0, len(offset_runs))
self._send_time_changed(next_setting + offset)
self.hass.block_till_done()
self.assertEqual(2, len(runs))
self.assertEqual(1, len(offset_runs))
2016-08-26 06:25:35 +00:00
unsub()
unsub2()
self._send_time_changed(next_setting + offset)
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(2, len(runs))
self.assertEqual(1, len(offset_runs))
def _send_time_changed(self, now):
2016-03-09 09:25:50 +00:00
"""Send a time changed event."""
self.hass.bus.fire(ha.EVENT_TIME_CHANGED, {ha.ATTR_NOW: now})
2016-01-26 09:28:31 +00:00
def test_periodic_task_minute(self):
2016-03-09 09:25:50 +00:00
"""Test periodic tasks per minute."""
2016-01-26 09:28:31 +00:00
specific_runs = []
2016-08-26 06:25:35 +00:00
unsub = track_utc_time_change(
2016-01-26 09:28:31 +00:00
self.hass, lambda x: specific_runs.append(1), minute='/5')
self._send_time_changed(datetime(2014, 5, 24, 12, 0, 0))
self.hass.block_till_done()
2016-01-26 09:28:31 +00:00
self.assertEqual(1, len(specific_runs))
self._send_time_changed(datetime(2014, 5, 24, 12, 3, 0))
self.hass.block_till_done()
2016-01-26 09:28:31 +00:00
self.assertEqual(1, len(specific_runs))
self._send_time_changed(datetime(2014, 5, 24, 12, 5, 0))
self.hass.block_till_done()
2016-01-26 09:28:31 +00:00
self.assertEqual(2, len(specific_runs))
2016-01-26 16:06:50 +00:00
2016-08-26 06:25:35 +00:00
unsub()
self._send_time_changed(datetime(2014, 5, 24, 12, 5, 0))
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(2, len(specific_runs))
2016-01-26 16:06:50 +00:00
def test_periodic_task_hour(self):
2016-03-09 09:25:50 +00:00
"""Test periodic tasks per hour."""
2016-01-26 16:06:50 +00:00
specific_runs = []
2016-08-26 06:25:35 +00:00
unsub = track_utc_time_change(
2016-01-26 16:06:50 +00:00
self.hass, lambda x: specific_runs.append(1), hour='/2')
2016-01-26 16:13:18 +00:00
self._send_time_changed(datetime(2014, 5, 24, 22, 0, 0))
self.hass.block_till_done()
2016-01-26 16:06:50 +00:00
self.assertEqual(1, len(specific_runs))
2016-01-26 16:13:18 +00:00
self._send_time_changed(datetime(2014, 5, 24, 23, 0, 0))
self.hass.block_till_done()
2016-01-26 16:06:50 +00:00
self.assertEqual(1, len(specific_runs))
2016-01-26 16:13:18 +00:00
self._send_time_changed(datetime(2014, 5, 24, 0, 0, 0))
self.hass.block_till_done()
2016-01-26 16:13:18 +00:00
self.assertEqual(2, len(specific_runs))
self._send_time_changed(datetime(2014, 5, 25, 1, 0, 0))
self.hass.block_till_done()
2016-01-26 16:13:18 +00:00
self.assertEqual(2, len(specific_runs))
self._send_time_changed(datetime(2014, 5, 25, 2, 0, 0))
self.hass.block_till_done()
2016-01-26 16:13:18 +00:00
self.assertEqual(3, len(specific_runs))
2016-08-26 06:25:35 +00:00
unsub()
self._send_time_changed(datetime(2014, 5, 25, 2, 0, 0))
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(3, len(specific_runs))
def test_periodic_task_day(self):
2016-03-09 09:25:50 +00:00
"""Test periodic tasks per day."""
specific_runs = []
2016-08-26 06:25:35 +00:00
unsub = track_utc_time_change(
self.hass, lambda x: specific_runs.append(1), day='/2')
self._send_time_changed(datetime(2014, 5, 2, 0, 0, 0))
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self._send_time_changed(datetime(2014, 5, 3, 12, 0, 0))
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self._send_time_changed(datetime(2014, 5, 4, 0, 0, 0))
self.hass.block_till_done()
self.assertEqual(2, len(specific_runs))
2016-08-26 06:25:35 +00:00
unsub()
self._send_time_changed(datetime(2014, 5, 4, 0, 0, 0))
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(2, len(specific_runs))
def test_periodic_task_year(self):
2016-03-09 09:25:50 +00:00
"""Test periodic tasks per year."""
specific_runs = []
2016-08-26 06:25:35 +00:00
unsub = track_utc_time_change(
self.hass, lambda x: specific_runs.append(1), year='/2')
self._send_time_changed(datetime(2014, 5, 2, 0, 0, 0))
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self._send_time_changed(datetime(2015, 5, 2, 0, 0, 0))
self.hass.block_till_done()
self.assertEqual(1, len(specific_runs))
self._send_time_changed(datetime(2016, 5, 2, 0, 0, 0))
self.hass.block_till_done()
self.assertEqual(2, len(specific_runs))
2016-08-26 06:25:35 +00:00
unsub()
self._send_time_changed(datetime(2016, 5, 2, 0, 0, 0))
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
self.assertEqual(2, len(specific_runs))
def test_periodic_task_wrong_input(self):
2016-03-09 09:25:50 +00:00
"""Test periodic tasks with wrong input."""
specific_runs = []
track_utc_time_change(
self.hass, lambda x: specific_runs.append(1), year='/two')
self._send_time_changed(datetime(2014, 5, 2, 0, 0, 0))
self.hass.block_till_done()
2016-01-26 19:49:51 +00:00
self.assertEqual(0, len(specific_runs))