core/tests/test_core.py

1093 lines
33 KiB
Python
Raw Normal View History

2016-03-09 09:25:50 +00:00
"""Test to verify that Home Assistant core works."""
# pylint: disable=protected-access
import asyncio
import functools
import logging
import os
2014-11-23 17:51:16 +00:00
import unittest
from unittest.mock import patch, MagicMock
2015-09-13 05:56:49 +00:00
from datetime import datetime, timedelta
from tempfile import TemporaryDirectory
2014-11-23 17:51:16 +00:00
2018-12-10 11:58:51 +00:00
import voluptuous as vol
2015-08-04 16:16:10 +00:00
import pytz
2016-12-16 05:30:09 +00:00
import pytest
2015-08-04 16:16:10 +00:00
2015-08-17 03:53:17 +00:00
import homeassistant.core as ha
from homeassistant.exceptions import (InvalidEntityFormatError,
InvalidStateError)
from homeassistant.util.async_ import run_coroutine_threadsafe
2015-08-04 16:16:10 +00:00
import homeassistant.util.dt as dt_util
2016-08-09 03:42:25 +00:00
from homeassistant.util.unit_system import (METRIC_SYSTEM)
2015-08-04 16:16:10 +00:00
from homeassistant.const import (
__version__, EVENT_STATE_CHANGED, ATTR_FRIENDLY_NAME, CONF_UNIT_SYSTEM,
ATTR_NOW, EVENT_TIME_CHANGED, EVENT_TIMER_OUT_OF_SYNC, ATTR_SECONDS,
EVENT_HOMEASSISTANT_STOP, EVENT_HOMEASSISTANT_CLOSE,
2018-12-10 11:58:51 +00:00
EVENT_SERVICE_REGISTERED, EVENT_SERVICE_REMOVED, EVENT_CALL_SERVICE)
2015-08-04 16:16:10 +00:00
from tests.common import get_test_home_assistant, async_mock_service
2016-02-14 23:08:23 +00:00
2015-08-04 16:16:10 +00:00
PST = pytz.timezone('America/Los_Angeles')
2014-11-23 17:51:16 +00:00
def test_split_entity_id():
"""Test split_entity_id."""
assert ha.split_entity_id('domain.object_id') == ['domain', 'object_id']
2016-08-09 03:21:40 +00:00
def test_async_add_job_schedule_callback():
"""Test that we schedule coroutines and add jobs to the job pool."""
hass = MagicMock()
job = MagicMock()
ha.HomeAssistant.async_add_job(hass, ha.callback(job))
assert len(hass.loop.call_soon.mock_calls) == 1
assert len(hass.loop.create_task.mock_calls) == 0
assert len(hass.add_job.mock_calls) == 0
def test_async_add_job_schedule_partial_callback():
"""Test that we schedule partial coros and add jobs to the job pool."""
hass = MagicMock()
job = MagicMock()
partial = functools.partial(ha.callback(job))
ha.HomeAssistant.async_add_job(hass, partial)
assert len(hass.loop.call_soon.mock_calls) == 1
assert len(hass.loop.create_task.mock_calls) == 0
assert len(hass.add_job.mock_calls) == 0
def test_async_add_job_schedule_coroutinefunction(loop):
"""Test that we schedule coroutines and add jobs to the job pool."""
hass = MagicMock(loop=MagicMock(wraps=loop))
async def job():
pass
ha.HomeAssistant.async_add_job(hass, job)
assert len(hass.loop.call_soon.mock_calls) == 0
assert len(hass.loop.create_task.mock_calls) == 1
assert len(hass.add_job.mock_calls) == 0
def test_async_add_job_schedule_partial_coroutinefunction(loop):
"""Test that we schedule partial coros and add jobs to the job pool."""
hass = MagicMock(loop=MagicMock(wraps=loop))
async def job():
pass
partial = functools.partial(job)
ha.HomeAssistant.async_add_job(hass, partial)
assert len(hass.loop.call_soon.mock_calls) == 0
assert len(hass.loop.create_task.mock_calls) == 1
assert len(hass.add_job.mock_calls) == 0
def test_async_add_job_add_threaded_job_to_pool():
"""Test that we schedule coroutines and add jobs to the job pool."""
hass = MagicMock()
def job():
pass
ha.HomeAssistant.async_add_job(hass, job)
assert len(hass.loop.call_soon.mock_calls) == 0
assert len(hass.loop.create_task.mock_calls) == 0
assert len(hass.loop.run_in_executor.mock_calls) == 1
def test_async_create_task_schedule_coroutine(loop):
"""Test that we schedule coroutines and add jobs to the job pool."""
hass = MagicMock(loop=MagicMock(wraps=loop))
async def job():
pass
ha.HomeAssistant.async_create_task(hass, job())
assert len(hass.loop.call_soon.mock_calls) == 0
assert len(hass.loop.create_task.mock_calls) == 1
assert len(hass.add_job.mock_calls) == 0
def test_async_run_job_calls_callback():
"""Test that the callback annotation is respected."""
hass = MagicMock()
calls = []
def job():
calls.append(1)
ha.HomeAssistant.async_run_job(hass, ha.callback(job))
assert len(calls) == 1
assert len(hass.async_add_job.mock_calls) == 0
def test_async_run_job_delegates_non_async():
"""Test that the callback annotation is respected."""
hass = MagicMock()
calls = []
def job():
calls.append(1)
ha.HomeAssistant.async_run_job(hass, job)
assert len(calls) == 0
assert len(hass.async_add_job.mock_calls) == 1
2016-08-09 03:21:40 +00:00
def test_stage_shutdown():
"""Simulate a shutdown, test calling stuff."""
hass = get_test_home_assistant()
test_stop = []
test_close = []
test_all = []
hass.bus.listen(
EVENT_HOMEASSISTANT_STOP, lambda event: test_stop.append(event))
hass.bus.listen(
EVENT_HOMEASSISTANT_CLOSE, lambda event: test_close.append(event))
hass.bus.listen('*', lambda event: test_all.append(event))
hass.stop()
assert len(test_stop) == 1
assert len(test_close) == 1
assert len(test_all) == 1
2014-11-23 17:51:16 +00:00
class TestHomeAssistant(unittest.TestCase):
2016-03-09 09:25:50 +00:00
"""Test the Home Assistant core classes."""
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def setUp(self):
2018-08-19 20:29:08 +00:00
"""Set up things to be run when tests are started."""
self.hass = get_test_home_assistant()
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def tearDown(self):
2016-03-09 10:15:04 +00:00
"""Stop everything that was started."""
self.hass.stop()
2014-11-23 20:57:29 +00:00
def test_pending_sheduler(self):
"""Add a coro to pending tasks."""
call_count = []
@asyncio.coroutine
def test_coro():
"""Test Coro."""
call_count.append('call')
for _ in range(3):
self.hass.add_job(test_coro())
run_coroutine_threadsafe(
asyncio.wait(self.hass._pending_tasks, loop=self.hass.loop),
loop=self.hass.loop
).result()
assert len(self.hass._pending_tasks) == 3
assert len(call_count) == 3
def test_async_add_job_pending_tasks_coro(self):
"""Add a coro to pending tasks."""
call_count = []
@asyncio.coroutine
def test_coro():
"""Test Coro."""
call_count.append('call')
for _ in range(2):
self.hass.add_job(test_coro())
@asyncio.coroutine
def wait_finish_callback():
"""Wait until all stuff is scheduled."""
yield from asyncio.sleep(0, loop=self.hass.loop)
yield from asyncio.sleep(0, loop=self.hass.loop)
run_coroutine_threadsafe(
wait_finish_callback(), self.hass.loop).result()
assert len(self.hass._pending_tasks) == 2
self.hass.block_till_done()
assert len(call_count) == 2
def test_async_add_job_pending_tasks_executor(self):
2018-01-27 19:58:27 +00:00
"""Run an executor in pending tasks."""
call_count = []
def test_executor():
"""Test executor."""
call_count.append('call')
@asyncio.coroutine
def wait_finish_callback():
"""Wait until all stuff is scheduled."""
yield from asyncio.sleep(0, loop=self.hass.loop)
yield from asyncio.sleep(0, loop=self.hass.loop)
for _ in range(2):
self.hass.add_job(test_executor)
run_coroutine_threadsafe(
wait_finish_callback(), self.hass.loop).result()
assert len(self.hass._pending_tasks) == 2
self.hass.block_till_done()
assert len(call_count) == 2
def test_async_add_job_pending_tasks_callback(self):
"""Run a callback in pending tasks."""
call_count = []
@ha.callback
def test_callback():
"""Test callback."""
call_count.append('call')
@asyncio.coroutine
def wait_finish_callback():
"""Wait until all stuff is scheduled."""
yield from asyncio.sleep(0, loop=self.hass.loop)
yield from asyncio.sleep(0, loop=self.hass.loop)
for _ in range(2):
self.hass.add_job(test_callback)
run_coroutine_threadsafe(
wait_finish_callback(), self.hass.loop).result()
self.hass.block_till_done()
assert len(self.hass._pending_tasks) == 0
assert len(call_count) == 2
2016-12-16 05:30:09 +00:00
def test_add_job_with_none(self):
"""Try to add a job with None as function."""
with pytest.raises(ValueError):
self.hass.add_job(None, 'test_arg')
2014-11-23 17:51:16 +00:00
class TestEvent(unittest.TestCase):
2016-03-09 09:25:50 +00:00
"""A Test Event class."""
2015-08-04 16:16:10 +00:00
def test_eq(self):
2016-03-09 09:25:50 +00:00
"""Test events."""
2015-08-04 16:16:10 +00:00
now = dt_util.utcnow()
data = {'some': 'attr'}
context = ha.Context()
2015-08-04 16:16:10 +00:00
event1, event2 = [
ha.Event('some_type', data, time_fired=now, context=context)
2015-08-04 16:16:10 +00:00
for _ in range(2)
]
assert event1 == event2
2015-08-04 16:16:10 +00:00
2014-11-23 17:51:16 +00:00
def test_repr(self):
2016-03-09 10:15:04 +00:00
"""Test that repr method works."""
assert "<Event TestEvent[L]>" == \
str(ha.Event("TestEvent"))
2014-11-23 17:51:16 +00:00
assert "<Event TestEvent[R]: beer=nice>" == \
2014-11-23 17:51:16 +00:00
str(ha.Event("TestEvent",
{"beer": "nice"},
ha.EventOrigin.remote))
2014-11-23 17:51:16 +00:00
2015-08-04 16:16:10 +00:00
def test_as_dict(self):
2016-03-09 09:25:50 +00:00
"""Test as dictionary."""
2015-08-04 16:16:10 +00:00
event_type = 'some_type'
now = dt_util.utcnow()
data = {'some': 'attr'}
event = ha.Event(event_type, data, ha.EventOrigin.local, now)
expected = {
'event_type': event_type,
'data': data,
'origin': 'LOCAL',
2016-04-16 07:55:35 +00:00
'time_fired': now,
'context': {
'id': event.context.id,
'parent_id': None,
'user_id': event.context.user_id,
},
2015-08-04 16:16:10 +00:00
}
assert expected == event.as_dict()
2015-08-04 16:16:10 +00:00
2014-11-23 17:51:16 +00:00
class TestEventBus(unittest.TestCase):
2016-03-09 09:25:50 +00:00
"""Test EventBus methods."""
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def setUp(self):
2018-08-19 20:29:08 +00:00
"""Set up things to be run when tests are started."""
self.hass = get_test_home_assistant()
self.bus = self.hass.bus
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def tearDown(self):
2016-03-09 09:25:50 +00:00
"""Stop down stuff we started."""
self.hass.stop()
2014-11-23 20:57:29 +00:00
2014-11-23 17:51:16 +00:00
def test_add_remove_listener(self):
2016-03-09 09:25:50 +00:00
"""Test remove_listener method."""
self.hass.allow_pool = False
2014-11-23 17:51:16 +00:00
old_count = len(self.bus.listeners)
2016-02-14 06:57:40 +00:00
def listener(_): pass
2014-11-23 17:51:16 +00:00
unsub = self.bus.listen('test', listener)
2014-11-23 17:51:16 +00:00
assert old_count + 1 == len(self.bus.listeners)
2014-11-23 17:51:16 +00:00
# Remove listener
unsub()
assert old_count == len(self.bus.listeners)
2014-11-23 17:51:16 +00:00
# Should do nothing now
unsub()
2014-11-23 17:51:16 +00:00
2016-08-26 06:25:35 +00:00
def test_unsubscribe_listener(self):
"""Test unsubscribe listener from returned function."""
calls = []
@ha.callback
2016-08-26 06:25:35 +00:00
def listener(event):
"""Mock listener."""
calls.append(event)
unsub = self.bus.listen('test', listener)
self.bus.fire('test')
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
assert len(calls) == 1
unsub()
self.bus.fire('event')
self.hass.block_till_done()
2016-08-26 06:25:35 +00:00
assert len(calls) == 1
def test_listen_once_event_with_callback(self):
2016-03-09 09:25:50 +00:00
"""Test listen_once_event method."""
2014-11-29 07:19:59 +00:00
runs = []
@ha.callback
def event_handler(event):
runs.append(event)
self.bus.listen_once('test_event', event_handler)
self.bus.fire('test_event')
# Second time it should not increase runs
self.bus.fire('test_event')
self.hass.block_till_done()
assert 1 == len(runs)
def test_listen_once_event_with_coroutine(self):
"""Test listen_once_event method."""
runs = []
@asyncio.coroutine
def event_handler(event):
runs.append(event)
self.bus.listen_once('test_event', event_handler)
self.bus.fire('test_event')
# Second time it should not increase runs
self.bus.fire('test_event')
self.hass.block_till_done()
assert 1 == len(runs)
def test_listen_once_event_with_thread(self):
"""Test listen_once_event method."""
runs = []
def event_handler(event):
runs.append(event)
self.bus.listen_once('test_event', event_handler)
2014-11-29 07:19:59 +00:00
self.bus.fire('test_event')
# Second time it should not increase runs
self.bus.fire('test_event')
2015-08-04 16:16:10 +00:00
self.hass.block_till_done()
assert 1 == len(runs)
2014-11-29 07:19:59 +00:00
def test_thread_event_listener(self):
"""Test thread event listener."""
thread_calls = []
def thread_listener(event):
thread_calls.append(event)
self.bus.listen('test_thread', thread_listener)
self.bus.fire('test_thread')
self.hass.block_till_done()
assert len(thread_calls) == 1
def test_callback_event_listener(self):
"""Test callback event listener."""
callback_calls = []
@ha.callback
def callback_listener(event):
callback_calls.append(event)
self.bus.listen('test_callback', callback_listener)
self.bus.fire('test_callback')
self.hass.block_till_done()
assert len(callback_calls) == 1
def test_coroutine_event_listener(self):
"""Test coroutine event listener."""
coroutine_calls = []
@asyncio.coroutine
def coroutine_listener(event):
coroutine_calls.append(event)
self.bus.listen('test_coroutine', coroutine_listener)
self.bus.fire('test_coroutine')
self.hass.block_till_done()
assert len(coroutine_calls) == 1
2014-11-23 17:51:16 +00:00
def test_state_init():
"""Test state.init."""
with pytest.raises(InvalidEntityFormatError):
ha.State('invalid_entity_format', 'test_state')
2015-08-04 16:16:10 +00:00
with pytest.raises(InvalidStateError):
ha.State('domain.long_state', 't' * 256)
def test_state_domain():
"""Test domain."""
state = ha.State('some_domain.hello', 'world')
assert 'some_domain' == state.domain
def test_state_object_id():
"""Test object ID."""
state = ha.State('domain.hello', 'world')
assert 'hello' == state.object_id
def test_state_name_if_no_friendly_name_attr():
"""Test if there is no friendly name."""
state = ha.State('domain.hello_world', 'world')
assert 'hello world' == state.name
def test_state_name_if_friendly_name_attr():
"""Test if there is a friendly name."""
name = 'Some Unique Name'
state = ha.State('domain.hello_world', 'world',
{ATTR_FRIENDLY_NAME: name})
assert name == state.name
def test_state_dict_conversion():
"""Test conversion of dict."""
state = ha.State('domain.hello', 'world', {'some': 'attr'})
assert state == ha.State.from_dict(state.as_dict())
def test_state_dict_conversion_with_wrong_data():
"""Test conversion with wrong data."""
assert ha.State.from_dict(None) is None
assert ha.State.from_dict({'state': 'yes'}) is None
assert ha.State.from_dict({'entity_id': 'yes'}) is None
# Make sure invalid context data doesn't crash
wrong_context = ha.State.from_dict({
'entity_id': 'light.kitchen',
'state': 'on',
'context': {
'id': '123',
'non-existing': 'crash'
}
})
assert wrong_context is not None
assert wrong_context.context.id == '123'
def test_state_repr():
"""Test state.repr."""
assert "<state happy.happy=on @ 1984-12-08T12:00:00+00:00>" == \
str(ha.State(
"happy.happy", "on",
last_changed=datetime(1984, 12, 8, 12, 0, 0)))
assert "<state happy.happy=on; brightness=144 @ " \
"1984-12-08T12:00:00+00:00>" == \
str(ha.State("happy.happy", "on", {"brightness": 144},
datetime(1984, 12, 8, 12, 0, 0)))
2014-11-23 17:51:16 +00:00
class TestStateMachine(unittest.TestCase):
2016-03-09 10:15:04 +00:00
"""Test State machine methods."""
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def setUp(self):
2018-08-19 20:29:08 +00:00
"""Set up things to be run when tests are started."""
self.hass = get_test_home_assistant()
self.states = self.hass.states
2014-11-23 17:51:16 +00:00
self.states.set("light.Bowl", "on")
self.states.set("switch.AC", "off")
# pylint: disable=invalid-name
def tearDown(self):
2016-03-09 09:25:50 +00:00
"""Stop down stuff we started."""
self.hass.stop()
2014-11-23 20:57:29 +00:00
2014-11-23 17:51:16 +00:00
def test_is_state(self):
2016-03-09 09:25:50 +00:00
"""Test is_state method."""
assert self.states.is_state('light.Bowl', 'on')
assert not self.states.is_state('light.Bowl', 'off')
assert not self.states.is_state('light.Non_existing', 'on')
2014-11-23 17:51:16 +00:00
2014-11-29 07:19:59 +00:00
def test_entity_ids(self):
2016-03-09 09:25:50 +00:00
"""Test get_entity_ids method."""
2014-11-29 07:19:59 +00:00
ent_ids = self.states.entity_ids()
assert 2 == len(ent_ids)
assert 'light.bowl' in ent_ids
assert 'switch.ac' in ent_ids
2014-11-29 07:19:59 +00:00
ent_ids = self.states.entity_ids('light')
assert 1 == len(ent_ids)
assert 'light.bowl' in ent_ids
2014-11-29 07:19:59 +00:00
2015-08-04 16:16:10 +00:00
def test_all(self):
2016-03-09 09:25:50 +00:00
"""Test everything."""
2015-08-04 16:16:10 +00:00
states = sorted(state.entity_id for state in self.states.all())
assert ['light.bowl', 'switch.ac'] == states
2015-08-04 16:16:10 +00:00
2014-11-23 17:51:16 +00:00
def test_remove(self):
2016-03-09 09:25:50 +00:00
"""Test remove method."""
2016-02-14 06:57:40 +00:00
events = []
@ha.callback
def callback(event):
events.append(event)
self.hass.bus.listen(EVENT_STATE_CHANGED, callback)
2016-02-14 06:57:40 +00:00
assert 'light.bowl' in self.states.entity_ids()
assert self.states.remove('light.bowl')
self.hass.block_till_done()
2016-02-14 06:57:40 +00:00
assert 'light.bowl' not in self.states.entity_ids()
assert 1 == len(events)
assert 'light.bowl' == events[0].data.get('entity_id')
assert events[0].data.get('old_state') is not None
assert 'light.bowl' == events[0].data['old_state'].entity_id
assert events[0].data.get('new_state') is None
2014-11-23 17:51:16 +00:00
# If it does not exist, we should get False
assert not self.states.remove('light.Bowl')
self.hass.block_till_done()
assert 1 == len(events)
2014-11-23 17:51:16 +00:00
def test_case_insensitivty(self):
2016-03-09 09:25:50 +00:00
"""Test insensitivty."""
runs = []
@ha.callback
def callback(event):
runs.append(event)
self.hass.bus.listen(EVENT_STATE_CHANGED, callback)
self.states.set('light.BOWL', 'off')
self.hass.block_till_done()
assert self.states.is_state('light.bowl', 'off')
assert 1 == len(runs)
def test_last_changed_not_updated_on_same_state(self):
2016-03-09 09:25:50 +00:00
"""Test to not update the existing, same state."""
state = self.states.get('light.Bowl')
2015-09-13 05:56:49 +00:00
future = dt_util.utcnow() + timedelta(hours=10)
2015-09-13 05:56:49 +00:00
with patch('homeassistant.util.dt.utcnow', return_value=future):
self.states.set("light.Bowl", "on", {'attr': 'triggers_change'})
self.hass.block_till_done()
state2 = self.states.get('light.Bowl')
assert state2 is not None
assert state.last_changed == state2.last_changed
def test_force_update(self):
"""Test force update option."""
events = []
@ha.callback
def callback(event):
events.append(event)
self.hass.bus.listen(EVENT_STATE_CHANGED, callback)
self.states.set('light.bowl', 'on')
self.hass.block_till_done()
assert 0 == len(events)
self.states.set('light.bowl', 'on', None, True)
self.hass.block_till_done()
assert 1 == len(events)
2014-11-23 17:51:16 +00:00
def test_service_call_repr():
"""Test ServiceCall repr."""
call = ha.ServiceCall('homeassistant', 'start')
assert str(call) == \
"<ServiceCall homeassistant.start (c:{})>".format(call.context.id)
2016-03-09 09:25:50 +00:00
call2 = ha.ServiceCall('homeassistant', 'start', {'fast': 'yes'})
assert str(call2) == \
"<ServiceCall homeassistant.start (c:{}): fast=yes>".format(
call2.context.id)
2014-11-23 17:51:16 +00:00
class TestServiceRegistry(unittest.TestCase):
2016-03-09 10:15:04 +00:00
"""Test ServicerRegistry methods."""
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def setUp(self):
2018-08-19 20:29:08 +00:00
"""Set up things to be run when tests are started."""
self.hass = get_test_home_assistant()
self.services = self.hass.services
@ha.callback
def mock_service(call):
pass
self.services.register("Test_Domain", "TEST_SERVICE", mock_service)
2014-11-23 17:51:16 +00:00
self.calls_register = []
@ha.callback
def mock_event_register(event):
"""Mock register event."""
self.calls_register.append(event)
self.hass.bus.listen(EVENT_SERVICE_REGISTERED, mock_event_register)
# pylint: disable=invalid-name
def tearDown(self):
2016-03-09 09:25:50 +00:00
"""Stop down stuff we started."""
self.hass.stop()
2014-11-23 20:57:29 +00:00
2014-11-23 17:51:16 +00:00
def test_has_service(self):
2016-03-09 09:25:50 +00:00
"""Test has_service method."""
assert self.services.has_service("tesT_domaiN", "tesT_servicE")
assert not self.services.has_service("test_domain", "non_existing")
assert not self.services.has_service("non_existing", "test_service")
2015-08-04 16:16:10 +00:00
def test_services(self):
2016-03-09 09:25:50 +00:00
"""Test services."""
assert len(self.services.services) == 1
2015-08-04 16:16:10 +00:00
def test_call_with_blocking_done_in_time(self):
2016-03-09 09:25:50 +00:00
"""Test call with blocking."""
2015-08-04 16:16:10 +00:00
calls = []
@ha.callback
def service_handler(call):
"""Service handler."""
calls.append(call)
self.services.register(
"test_domain", "register_calls", service_handler)
self.hass.block_till_done()
assert len(self.calls_register) == 1
assert self.calls_register[-1].data['domain'] == 'test_domain'
assert self.calls_register[-1].data['service'] == 'register_calls'
2015-08-04 16:16:10 +00:00
assert self.services.call('test_domain', 'REGISTER_CALLS',
blocking=True)
assert 1 == len(calls)
2015-08-04 16:16:10 +00:00
def test_call_non_existing_with_blocking(self):
2016-03-09 09:25:50 +00:00
"""Test non-existing with blocking."""
with pytest.raises(ha.ServiceNotFound):
self.services.call('test_domain', 'i_do_not_exist', blocking=True)
2015-08-04 16:16:10 +00:00
def test_async_service(self):
"""Test registering and calling an async service."""
calls = []
@asyncio.coroutine
def service_handler(call):
"""Service handler coroutine."""
calls.append(call)
self.services.register(
'test_domain', 'register_calls', service_handler)
self.hass.block_till_done()
assert len(self.calls_register) == 1
assert self.calls_register[-1].data['domain'] == 'test_domain'
assert self.calls_register[-1].data['service'] == 'register_calls'
assert self.services.call('test_domain', 'REGISTER_CALLS',
blocking=True)
self.hass.block_till_done()
assert 1 == len(calls)
def test_callback_service(self):
"""Test registering and calling an async service."""
calls = []
@ha.callback
def service_handler(call):
"""Service handler coroutine."""
calls.append(call)
self.services.register(
'test_domain', 'register_calls', service_handler)
self.hass.block_till_done()
assert len(self.calls_register) == 1
assert self.calls_register[-1].data['domain'] == 'test_domain'
assert self.calls_register[-1].data['service'] == 'register_calls'
assert self.services.call('test_domain', 'REGISTER_CALLS',
blocking=True)
self.hass.block_till_done()
assert 1 == len(calls)
def test_remove_service(self):
"""Test remove service."""
calls_remove = []
@ha.callback
def mock_event_remove(event):
"""Mock register event."""
calls_remove.append(event)
self.hass.bus.listen(EVENT_SERVICE_REMOVED, mock_event_remove)
assert self.services.has_service('test_Domain', 'test_Service')
self.services.remove('test_Domain', 'test_Service')
self.hass.block_till_done()
assert not self.services.has_service('test_Domain', 'test_Service')
assert len(calls_remove) == 1
assert calls_remove[-1].data['domain'] == 'test_domain'
assert calls_remove[-1].data['service'] == 'test_service'
def test_remove_service_that_not_exists(self):
"""Test remove service that not exists."""
calls_remove = []
@ha.callback
def mock_event_remove(event):
"""Mock register event."""
calls_remove.append(event)
self.hass.bus.listen(EVENT_SERVICE_REMOVED, mock_event_remove)
assert not self.services.has_service('test_xxx', 'test_yyy')
self.services.remove('test_xxx', 'test_yyy')
self.hass.block_till_done()
assert len(calls_remove) == 0
2015-08-04 16:16:10 +00:00
class TestConfig(unittest.TestCase):
2016-03-09 09:25:50 +00:00
"""Test configuration methods."""
# pylint: disable=invalid-name
def setUp(self):
2018-08-19 20:29:08 +00:00
"""Set up things to be run when tests are started."""
2015-08-04 16:16:10 +00:00
self.config = ha.Config()
assert self.config.config_dir is None
2015-08-04 16:16:10 +00:00
def test_path_with_file(self):
2016-03-09 09:25:50 +00:00
"""Test get_config_path method."""
2016-08-09 03:21:40 +00:00
self.config.config_dir = '/tmp/ha-config'
assert "/tmp/ha-config/test.conf" == \
self.config.path("test.conf")
2015-08-04 16:16:10 +00:00
def test_path_with_dir_and_file(self):
2016-03-09 09:25:50 +00:00
"""Test get_config_path method."""
2016-08-09 03:21:40 +00:00
self.config.config_dir = '/tmp/ha-config'
assert "/tmp/ha-config/dir/test.conf" == \
self.config.path("dir", "test.conf")
2015-08-04 16:16:10 +00:00
def test_as_dict(self):
2016-03-09 09:25:50 +00:00
"""Test as dict."""
self.config.config_dir = '/tmp/ha-config'
2015-08-04 16:16:10 +00:00
expected = {
'latitude': None,
'longitude': None,
'elevation': None,
Add unit system support Add unit symbol constants Initial unit system object Import more constants Pydoc for unit system file Import constants for configuration validation Unit system validation method Typing for constants Inches are valid lengths too Typings Change base class to dict - needed for remote api call serialization Validation Use dictionary keys Defined unit systems Update location util to use metric instead of us fahrenheit Update constant imports Import defined unit systems Update configuration to use unit system Update schema to use unit system Update constants Add imports to core for unit system and distance Type for config Default unit system Convert distance from HASS instance Update temperature conversion to use unit system Update temperature conversion Set unit system based on configuration Set info unit system Return unit system dictionary with config dictionary Auto discover unit system Update location test for use metric Update forecast unit system Update mold indicator unit system Update thermostat unit system Update thermostat demo test Unit tests around unit system Update test common hass configuration Update configuration unit tests There should always be a unit system! Update core unit tests Constants typing Linting issues Remove unused import Update fitbit sensor to use application unit system Update google travel time to use application unit system Update configuration example Update dht sensor Update DHT temperature conversion to use the utility function Update swagger config Update my sensors metric flag Update hvac component temperature conversion HVAC conversion for temperature Pull unit from sensor type map Pull unit from sensor type map Update the temper sensor unit Update yWeather sensor unit Update hvac demo unit test Set unit test config unit system to metric Use hass unit system length for default in proximity Use the name of the system instead of temperature Use constants from const Unused import Forecasted temperature Fix calculation in case furthest distance is greater than 1000000 units Remove unneeded constants Set default length to km or miles Use constants Linting doesn't like importing just for typing Fix reference Test is expecting meters - set config to meters Use constant Use constant PyDoc for unit test Should be not in Rename to units Change unit system to be an object - not a dictionary Return tuple in conversion Move convert to temperature util Temperature conversion is now in unit system Update imports Rename to units Units is now an object Use temperature util conversion Unit system is now an object Validate and convert unit system config Return the scalar value in template distance Test is expecting meters Update unit tests around unit system Distance util returns tuple Fix location info test Set units Update unit tests Convert distance DOH Pull out the scalar from the vector Linting I really hate python linting Linting again BLARG Unit test documentation Unit test around is metric flag Break ternary statement into if/else blocks Don't use dictionary - use members is metric flag Rename constants Use is metric flag Move constants to CONST file Move to const file Raise error if unit is not expected Typing No need to return unit since only performing conversion if it can work Use constants Line wrapping Raise error if invalid value Remove subscripts from conversion as they are no longer returned as tuples No longer tuples No longer tuples Check for numeric type Fix string format to use correct variable Typing Assert errors raised Remove subscript Only convert temperature if we know the unit If no unit of measurement set - default to HASS config Convert only if we know the unit Remove subscription Fix not in clause Linting fixes Wants a boolean Clearer if-block Check if the key is in the config first Missed a couple expecting tuples Backwards compatibility No like-y ternary! Error handling around state setting Pretty unit system configuration validation More tuple crap Use is metric flag Error handling around min/max temp Explode if no unit Pull unit from config Celsius has a decimal Unused import Check if it's a temperature before we try to convert it to a temperature Linting says too many statements - combine lat/long in a fairly reasonable manner Backwards compatibility unit test Better doc
2016-07-31 20:24:49 +00:00
CONF_UNIT_SYSTEM: METRIC_SYSTEM.as_dict(),
2015-08-04 16:16:10 +00:00
'location_name': None,
'time_zone': 'UTC',
'components': set(),
'config_dir': '/tmp/ha-config',
'whitelist_external_dirs': set(),
2015-10-26 04:00:22 +00:00
'version': __version__,
2015-08-04 16:16:10 +00:00
}
assert expected == self.config.as_dict()
2015-08-04 16:16:10 +00:00
def test_is_allowed_path(self):
"""Test is_allowed_path method."""
with TemporaryDirectory() as tmp_dir:
# The created dir is in /tmp. This is a symlink on OS X
# causing this test to fail unless we resolve path first.
self.config.whitelist_external_dirs = set((
os.path.realpath(tmp_dir),
))
test_file = os.path.join(tmp_dir, "test.jpg")
with open(test_file, "w") as tmp_file:
tmp_file.write("test")
valid = [
test_file,
tmp_dir,
os.path.join(tmp_dir, 'notfound321')
]
for path in valid:
assert self.config.is_allowed_path(path)
self.config.whitelist_external_dirs = set(('/home', '/var'))
unvalid = [
"/hass/config/secure",
"/etc/passwd",
"/root/secure_file",
"/var/../etc/passwd",
test_file,
]
for path in unvalid:
assert not self.config.is_allowed_path(path)
with pytest.raises(AssertionError):
self.config.is_allowed_path(None)
2015-08-04 16:16:10 +00:00
@patch('homeassistant.core.monotonic')
def test_create_timer(mock_monotonic, loop):
"""Test create timer."""
hass = MagicMock()
funcs = []
orig_callback = ha.callback
def mock_callback(func):
funcs.append(func)
return orig_callback(func)
mock_monotonic.side_effect = 10.2, 10.8, 11.3
with patch.object(ha, 'callback', mock_callback), \
patch('homeassistant.core.dt_util.utcnow',
return_value=datetime(2018, 12, 31, 3, 4, 5, 333333)):
ha._async_create_timer(hass)
assert len(funcs) == 2
fire_time_event, stop_timer = funcs
assert len(hass.loop.call_later.mock_calls) == 1
delay, callback, target = hass.loop.call_later.mock_calls[0][1]
assert abs(delay - 0.666667) < 0.001
assert callback is fire_time_event
assert abs(target - 10.866667) < 0.001
with patch('homeassistant.core.dt_util.utcnow',
return_value=datetime(2018, 12, 31, 3, 4, 6, 100000)):
callback(target)
assert len(hass.bus.async_listen_once.mock_calls) == 1
assert len(hass.bus.async_fire.mock_calls) == 1
assert len(hass.loop.call_later.mock_calls) == 2
event_type, callback = hass.bus.async_listen_once.mock_calls[0][1]
assert event_type == EVENT_HOMEASSISTANT_STOP
assert callback is stop_timer
delay, callback, target = hass.loop.call_later.mock_calls[1][1]
assert abs(delay - 0.9) < 0.001
assert callback is fire_time_event
assert abs(target - 12.2) < 0.001
event_type, event_data = hass.bus.async_fire.mock_calls[0][1]
assert event_type == EVENT_TIME_CHANGED
assert event_data[ATTR_NOW] == datetime(2018, 12, 31, 3, 4, 6, 100000)
@patch('homeassistant.core.monotonic')
def test_timer_out_of_sync(mock_monotonic, loop):
"""Test create timer."""
hass = MagicMock()
funcs = []
orig_callback = ha.callback
def mock_callback(func):
funcs.append(func)
return orig_callback(func)
mock_monotonic.side_effect = 10.2, 13.3, 13.4
with patch.object(ha, 'callback', mock_callback), \
patch('homeassistant.core.dt_util.utcnow',
return_value=datetime(2018, 12, 31, 3, 4, 5, 333333)):
ha._async_create_timer(hass)
delay, callback, target = hass.loop.call_later.mock_calls[0][1]
with patch('homeassistant.core.dt_util.utcnow',
return_value=datetime(2018, 12, 31, 3, 4, 8, 200000)):
callback(target)
event_type, event_data = hass.bus.async_fire.mock_calls[1][1]
assert event_type == EVENT_TIMER_OUT_OF_SYNC
assert abs(event_data[ATTR_SECONDS] - 2.433333) < 0.001
assert len(funcs) == 2
fire_time_event, stop_timer = funcs
assert len(hass.loop.call_later.mock_calls) == 2
delay, callback, target = hass.loop.call_later.mock_calls[1][1]
assert abs(delay - 0.8) < 0.001
assert callback is fire_time_event
assert abs(target - 14.2) < 0.001
@asyncio.coroutine
def test_hass_start_starts_the_timer(loop):
"""Test when hass starts, it starts the timer."""
hass = ha.HomeAssistant(loop=loop)
try:
with patch('homeassistant.core._async_create_timer') as mock_timer:
yield from hass.async_start()
assert hass.state == ha.CoreState.running
assert not hass._track_task
assert len(mock_timer.mock_calls) == 1
assert mock_timer.mock_calls[0][1][0] is hass
finally:
yield from hass.async_stop()
assert hass.state == ha.CoreState.not_running
@asyncio.coroutine
def test_start_taking_too_long(loop, caplog):
"""Test when async_start takes too long."""
hass = ha.HomeAssistant(loop=loop)
caplog.set_level(logging.WARNING)
try:
with patch('homeassistant.core.timeout',
side_effect=asyncio.TimeoutError), \
patch('homeassistant.core._async_create_timer') as mock_timer:
yield from hass.async_start()
assert hass.state == ha.CoreState.running
assert len(mock_timer.mock_calls) == 1
assert mock_timer.mock_calls[0][1][0] is hass
assert 'Something is blocking Home Assistant' in caplog.text
finally:
yield from hass.async_stop()
assert hass.state == ha.CoreState.not_running
@asyncio.coroutine
def test_track_task_functions(loop):
"""Test function to start/stop track task and initial state."""
hass = ha.HomeAssistant(loop=loop)
try:
assert hass._track_task
hass.async_stop_track_tasks()
assert not hass._track_task
hass.async_track_tasks()
assert hass._track_task
finally:
yield from hass.async_stop()
async def test_service_executed_with_subservices(hass):
"""Test we block correctly till all services done."""
calls = async_mock_service(hass, 'test', 'inner')
context = ha.Context()
async def handle_outer(call):
"""Handle outer service call."""
calls.append(call)
call1 = hass.services.async_call('test', 'inner', blocking=True,
context=call.context)
call2 = hass.services.async_call('test', 'inner', blocking=True,
context=call.context)
await asyncio.wait([call1, call2])
calls.append(call)
hass.services.async_register('test', 'outer', handle_outer)
await hass.services.async_call('test', 'outer', blocking=True,
context=context)
assert len(calls) == 4
assert [call.service for call in calls] == [
'outer', 'inner', 'inner', 'outer']
assert all(call.context is context for call in calls)
2018-12-10 11:58:51 +00:00
async def test_service_call_event_contains_original_data(hass):
"""Test that service call event contains original data."""
events = []
@ha.callback
def callback(event):
events.append(event)
hass.bus.async_listen(EVENT_CALL_SERVICE, callback)
calls = async_mock_service(hass, 'test', 'service', vol.Schema({
'number': vol.Coerce(int)
}))
context = ha.Context()
2018-12-10 11:58:51 +00:00
await hass.services.async_call('test', 'service', {
'number': '23'
}, blocking=True, context=context)
2018-12-10 11:58:51 +00:00
await hass.async_block_till_done()
assert len(events) == 1
assert events[0].data['service_data']['number'] == '23'
assert events[0].context is context
2018-12-10 11:58:51 +00:00
assert len(calls) == 1
assert calls[0].data['number'] == 23
assert calls[0].context is context
def test_context():
"""Test context init."""
c = ha.Context()
assert c.user_id is None
assert c.parent_id is None
assert c.id is not None
c = ha.Context(23, 100)
assert c.user_id == 23
assert c.parent_id == 100
assert c.id is not None