2016-03-09 09:25:50 +00:00
|
|
|
"""Test to verify that Home Assistant core works."""
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=protected-access
|
2016-09-30 19:57:24 +00:00
|
|
|
import asyncio
|
2019-01-14 23:08:44 +00:00
|
|
|
import functools
|
2017-06-25 22:10:30 +00:00
|
|
|
import logging
|
|
|
|
import os
|
2014-11-23 17:51:16 +00:00
|
|
|
import unittest
|
2018-09-17 08:10:50 +00:00
|
|
|
from unittest.mock import patch, MagicMock
|
2015-09-13 05:56:49 +00:00
|
|
|
from datetime import datetime, timedelta
|
2017-06-25 22:10:30 +00:00
|
|
|
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
|
2017-10-25 16:05:30 +00:00
|
|
|
from homeassistant.exceptions import (InvalidEntityFormatError,
|
|
|
|
InvalidStateError)
|
2018-03-11 17:01:12 +00:00
|
|
|
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 (
|
2017-02-10 17:00:17 +00:00
|
|
|
__version__, EVENT_STATE_CHANGED, ATTR_FRIENDLY_NAME, CONF_UNIT_SYSTEM,
|
2018-10-15 09:34:36 +00:00
|
|
|
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
|
|
|
|
2018-09-11 19:40:35 +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
|
|
|
|
|
|
|
|
2016-10-05 03:44:32 +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
|
|
|
|
2016-10-05 03:44:32 +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
|
|
|
|
|
|
|
|
|
2019-01-14 23:08:44 +00:00
|
|
|
def test_async_add_job_schedule_partial_callback():
|
|
|
|
"""Test that we schedule partial coros and add jobs to the job pool."""
|
2016-10-05 03:44:32 +00:00
|
|
|
hass = MagicMock()
|
|
|
|
job = MagicMock()
|
2019-01-14 23:08:44 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
|
2019-02-09 21:13:12 +00:00
|
|
|
def test_async_add_job_schedule_coroutinefunction(loop):
|
2019-01-14 23:08:44 +00:00
|
|
|
"""Test that we schedule coroutines and add jobs to the job pool."""
|
2019-02-09 21:13:12 +00:00
|
|
|
hass = MagicMock(loop=MagicMock(wraps=loop))
|
2019-01-14 23:08:44 +00:00
|
|
|
|
|
|
|
async def job():
|
|
|
|
pass
|
2016-10-05 03:44:32 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
2019-02-09 21:13:12 +00:00
|
|
|
def test_async_add_job_schedule_partial_coroutinefunction(loop):
|
2019-01-14 23:08:44 +00:00
|
|
|
"""Test that we schedule partial coros and add jobs to the job pool."""
|
2019-02-09 21:13:12 +00:00
|
|
|
hass = MagicMock(loop=MagicMock(wraps=loop))
|
2019-01-14 23:08:44 +00:00
|
|
|
|
|
|
|
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():
|
2016-10-05 03:44:32 +00:00
|
|
|
"""Test that we schedule coroutines and add jobs to the job pool."""
|
|
|
|
hass = MagicMock()
|
2019-01-14 23:08:44 +00:00
|
|
|
|
|
|
|
def job():
|
|
|
|
pass
|
2016-10-05 03:44:32 +00:00
|
|
|
|
|
|
|
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
|
2016-11-05 16:27:55 +00:00
|
|
|
assert len(hass.loop.run_in_executor.mock_calls) == 1
|
2016-10-05 03:44:32 +00:00
|
|
|
|
|
|
|
|
2019-02-09 21:13:12 +00:00
|
|
|
def test_async_create_task_schedule_coroutine(loop):
|
2018-07-13 10:24:51 +00:00
|
|
|
"""Test that we schedule coroutines and add jobs to the job pool."""
|
2019-02-09 21:13:12 +00:00
|
|
|
hass = MagicMock(loop=MagicMock(wraps=loop))
|
2018-07-13 10:24:51 +00:00
|
|
|
|
2019-01-14 23:08:44 +00:00
|
|
|
async def job():
|
|
|
|
pass
|
|
|
|
|
|
|
|
ha.HomeAssistant.async_create_task(hass, job())
|
2018-07-13 10:24:51 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
|
2016-10-05 03:44:32 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
2017-02-13 05:24:07 +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
|
|
|
|
2016-10-30 21:18:53 +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."""
|
2016-10-31 15:47:29 +00:00
|
|
|
self.hass = get_test_home_assistant()
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=invalid-name
|
|
|
|
def tearDown(self):
|
2016-03-09 10:15:04 +00:00
|
|
|
"""Stop everything that was started."""
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.stop()
|
2014-11-23 20:57:29 +00:00
|
|
|
|
2016-11-09 16:41:17 +00:00
|
|
|
def test_pending_sheduler(self):
|
2016-11-08 09:24:50 +00:00
|
|
|
"""Add a coro to pending tasks."""
|
|
|
|
call_count = []
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def test_coro():
|
|
|
|
"""Test Coro."""
|
|
|
|
call_count.append('call')
|
|
|
|
|
2018-01-30 22:44:05 +00:00
|
|
|
for _ in range(3):
|
2016-11-09 16:41:17 +00:00
|
|
|
self.hass.add_job(test_coro())
|
2016-11-08 09:24:50 +00:00
|
|
|
|
2016-11-09 16:41:17 +00:00
|
|
|
run_coroutine_threadsafe(
|
|
|
|
asyncio.wait(self.hass._pending_tasks, loop=self.hass.loop),
|
|
|
|
loop=self.hass.loop
|
|
|
|
).result()
|
|
|
|
|
2016-11-24 22:49:29 +00:00
|
|
|
assert len(self.hass._pending_tasks) == 3
|
|
|
|
assert len(call_count) == 3
|
2016-11-08 09:24:50 +00:00
|
|
|
|
2016-11-09 16:41:17 +00:00
|
|
|
def test_async_add_job_pending_tasks_coro(self):
|
|
|
|
"""Add a coro to pending tasks."""
|
2016-11-08 09:24:50 +00:00
|
|
|
call_count = []
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def test_coro():
|
|
|
|
"""Test Coro."""
|
|
|
|
call_count.append('call')
|
|
|
|
|
2018-01-30 22:44:05 +00:00
|
|
|
for _ in range(2):
|
2016-11-08 09:24:50 +00:00
|
|
|
self.hass.add_job(test_coro())
|
|
|
|
|
2016-11-17 04:00:08 +00:00
|
|
|
@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
|
2016-11-08 09:24:50 +00:00
|
|
|
self.hass.block_till_done()
|
2016-11-17 04:00:08 +00:00
|
|
|
assert len(call_count) == 2
|
2016-11-08 09:24:50 +00:00
|
|
|
|
2016-11-09 04:01:05 +00:00
|
|
|
def test_async_add_job_pending_tasks_executor(self):
|
2018-01-27 19:58:27 +00:00
|
|
|
"""Run an executor in pending tasks."""
|
2016-11-09 04:01:05 +00:00
|
|
|
call_count = []
|
|
|
|
|
|
|
|
def test_executor():
|
|
|
|
"""Test executor."""
|
|
|
|
call_count.append('call')
|
|
|
|
|
2016-11-17 04:00:08 +00:00
|
|
|
@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)
|
|
|
|
|
2018-01-30 22:44:05 +00:00
|
|
|
for _ in range(2):
|
2016-11-09 04:01:05 +00:00
|
|
|
self.hass.add_job(test_executor)
|
|
|
|
|
2016-11-17 04:00:08 +00:00
|
|
|
run_coroutine_threadsafe(
|
|
|
|
wait_finish_callback(), self.hass.loop).result()
|
|
|
|
|
|
|
|
assert len(self.hass._pending_tasks) == 2
|
2016-11-09 04:01:05 +00:00
|
|
|
self.hass.block_till_done()
|
2016-11-17 04:00:08 +00:00
|
|
|
assert len(call_count) == 2
|
2016-11-09 04:01:05 +00:00
|
|
|
|
|
|
|
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')
|
|
|
|
|
2016-11-17 04:00:08 +00:00
|
|
|
@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)
|
|
|
|
|
2018-01-30 22:44:05 +00:00
|
|
|
for _ in range(2):
|
2016-11-09 04:01:05 +00:00
|
|
|
self.hass.add_job(test_callback)
|
|
|
|
|
2016-11-17 04:00:08 +00:00
|
|
|
run_coroutine_threadsafe(
|
|
|
|
wait_finish_callback(), self.hass.loop).result()
|
|
|
|
|
2016-11-09 04:01:05 +00:00
|
|
|
self.hass.block_till_done()
|
2016-11-17 04:00:08 +00:00
|
|
|
|
|
|
|
assert len(self.hass._pending_tasks) == 0
|
|
|
|
assert len(call_count) == 2
|
2016-11-09 04:01:05 +00:00
|
|
|
|
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'}
|
2018-08-10 16:09:01 +00:00
|
|
|
context = ha.Context()
|
2015-08-04 16:16:10 +00:00
|
|
|
event1, event2 = [
|
2018-08-10 16:09:01 +00:00
|
|
|
ha.Event('some_type', data, time_fired=now, context=context)
|
2015-08-04 16:16:10 +00:00
|
|
|
for _ in range(2)
|
|
|
|
]
|
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
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."""
|
2018-10-24 10:10:05 +00:00
|
|
|
assert "<Event TestEvent[L]>" == \
|
|
|
|
str(ha.Event("TestEvent"))
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
assert "<Event TestEvent[R]: beer=nice>" == \
|
2014-11-23 17:51:16 +00:00
|
|
|
str(ha.Event("TestEvent",
|
|
|
|
{"beer": "nice"},
|
2018-10-24 10:10:05 +00:00
|
|
|
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,
|
2018-07-29 00:53:37 +00:00
|
|
|
'context': {
|
|
|
|
'id': event.context.id,
|
|
|
|
'user_id': event.context.user_id,
|
|
|
|
},
|
2015-08-04 16:16:10 +00:00
|
|
|
}
|
2018-10-24 10:10:05 +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
|
|
|
|
2016-10-30 21:18:53 +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."""
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass = get_test_home_assistant()
|
|
|
|
self.bus = self.hass.bus
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=invalid-name
|
|
|
|
def tearDown(self):
|
2016-03-09 09:25:50 +00:00
|
|
|
"""Stop down stuff we started."""
|
2016-09-13 02:16:14 +00:00
|
|
|
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."""
|
2016-10-31 15:47:29 +00:00
|
|
|
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
|
|
|
|
2016-10-18 02:38:41 +00:00
|
|
|
unsub = self.bus.listen('test', listener)
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
assert old_count + 1 == len(self.bus.listeners)
|
2014-11-23 17:51:16 +00:00
|
|
|
|
|
|
|
# Remove listener
|
2016-10-18 02:38:41 +00:00
|
|
|
unsub()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert old_count == len(self.bus.listeners)
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2016-10-18 02:38:41 +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 = []
|
|
|
|
|
2016-10-31 15:47:29 +00:00
|
|
|
@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')
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2016-08-26 06:25:35 +00:00
|
|
|
|
|
|
|
assert len(calls) == 1
|
|
|
|
|
|
|
|
unsub()
|
|
|
|
|
|
|
|
self.bus.fire('event')
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2016-08-26 06:25:35 +00:00
|
|
|
|
|
|
|
assert len(calls) == 1
|
|
|
|
|
2016-10-18 02:38:41 +00:00
|
|
|
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 = []
|
|
|
|
|
2016-10-18 02:38:41 +00:00
|
|
|
@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()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 1 == len(runs)
|
2016-10-18 02:38:41 +00:00
|
|
|
|
|
|
|
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()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 1 == len(runs)
|
2016-10-18 02:38:41 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 1 == len(runs)
|
2014-11-29 07:19:59 +00:00
|
|
|
|
2016-10-05 03:44:32 +00:00
|
|
|
def test_thread_event_listener(self):
|
2018-05-13 10:09:28 +00:00
|
|
|
"""Test thread event listener."""
|
2016-10-05 03:44:32 +00:00
|
|
|
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):
|
2018-05-13 10:09:28 +00:00
|
|
|
"""Test callback event listener."""
|
2016-10-05 03:44:32 +00:00
|
|
|
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):
|
2018-05-13 10:09:28 +00:00
|
|
|
"""Test coroutine event listener."""
|
2016-10-05 03:44:32 +00:00
|
|
|
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
|
|
|
|
2019-02-20 07:02:56 +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
|
|
|
|
2019-02-20 07:02:56 +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
|
|
|
|
2016-10-30 21:18:53 +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."""
|
2016-10-31 15:47:29 +00:00
|
|
|
self.hass = get_test_home_assistant()
|
2016-09-13 02:16:14 +00:00
|
|
|
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")
|
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=invalid-name
|
|
|
|
def tearDown(self):
|
2016-03-09 09:25:50 +00:00
|
|
|
"""Stop down stuff we started."""
|
2016-09-13 02:16:14 +00:00
|
|
|
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."""
|
2018-10-24 10:10:05 +00:00
|
|
|
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()
|
2018-10-24 10:10:05 +00:00
|
|
|
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')
|
2018-10-24 10:10:05 +00:00
|
|
|
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())
|
2018-10-24 10:10:05 +00:00
|
|
|
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 = []
|
2016-10-31 15:47:29 +00:00
|
|
|
|
|
|
|
@ha.callback
|
|
|
|
def callback(event):
|
|
|
|
events.append(event)
|
|
|
|
|
|
|
|
self.hass.bus.listen(EVENT_STATE_CHANGED, callback)
|
2016-02-14 06:57:40 +00:00
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 'light.bowl' in self.states.entity_ids()
|
|
|
|
assert self.states.remove('light.bowl')
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2016-02-14 06:57:40 +00:00
|
|
|
|
2018-10-24 10:10:05 +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
|
2018-10-24 10:10:05 +00:00
|
|
|
assert not self.states.remove('light.Bowl')
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 1 == len(events)
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2014-12-27 07:26:39 +00:00
|
|
|
def test_case_insensitivty(self):
|
2016-03-09 09:25:50 +00:00
|
|
|
"""Test insensitivty."""
|
2014-12-27 07:26:39 +00:00
|
|
|
runs = []
|
|
|
|
|
2016-10-31 15:47:29 +00:00
|
|
|
@ha.callback
|
|
|
|
def callback(event):
|
|
|
|
runs.append(event)
|
|
|
|
|
|
|
|
self.hass.bus.listen(EVENT_STATE_CHANGED, callback)
|
2014-12-27 07:26:39 +00:00
|
|
|
|
|
|
|
self.states.set('light.BOWL', 'off')
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2014-12-27 07:26:39 +00:00
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
assert self.states.is_state('light.bowl', 'off')
|
|
|
|
assert 1 == len(runs)
|
2014-12-27 07:26:39 +00:00
|
|
|
|
2015-01-02 16:48:20 +00:00
|
|
|
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."""
|
2015-01-02 16:48:20 +00:00
|
|
|
state = self.states.get('light.Bowl')
|
|
|
|
|
2015-09-13 05:56:49 +00:00
|
|
|
future = dt_util.utcnow() + timedelta(hours=10)
|
2015-01-02 16:48:20 +00:00
|
|
|
|
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'})
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2015-01-02 16:48:20 +00:00
|
|
|
|
2016-09-13 02:16:14 +00:00
|
|
|
state2 = self.states.get('light.Bowl')
|
|
|
|
assert state2 is not None
|
|
|
|
assert state.last_changed == state2.last_changed
|
2015-01-02 16:48:20 +00:00
|
|
|
|
2016-06-26 07:33:23 +00:00
|
|
|
def test_force_update(self):
|
|
|
|
"""Test force update option."""
|
|
|
|
events = []
|
2016-10-31 15:47:29 +00:00
|
|
|
|
|
|
|
@ha.callback
|
|
|
|
def callback(event):
|
|
|
|
events.append(event)
|
|
|
|
|
|
|
|
self.hass.bus.listen(EVENT_STATE_CHANGED, callback)
|
2016-06-26 07:33:23 +00:00
|
|
|
|
|
|
|
self.states.set('light.bowl', 'on')
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 0 == len(events)
|
2016-06-26 07:33:23 +00:00
|
|
|
|
|
|
|
self.states.set('light.bowl', 'on', None, True)
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass.block_till_done()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 1 == len(events)
|
2016-06-26 07:33:23 +00:00
|
|
|
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2018-07-29 00:53:37 +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
|
|
|
|
2018-07-29 00:53:37 +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
|
|
|
|
2016-10-30 21:18:53 +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."""
|
2016-09-13 02:16:14 +00:00
|
|
|
self.hass = get_test_home_assistant()
|
|
|
|
self.services = self.hass.services
|
2016-11-05 23:36:20 +00:00
|
|
|
|
|
|
|
@ha.callback
|
|
|
|
def mock_service(call):
|
|
|
|
pass
|
|
|
|
|
|
|
|
self.services.register("Test_Domain", "TEST_SERVICE", mock_service)
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2017-03-08 06:51:34 +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)
|
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=invalid-name
|
|
|
|
def tearDown(self):
|
2016-03-09 09:25:50 +00:00
|
|
|
"""Stop down stuff we started."""
|
2016-09-13 02:16:14 +00:00
|
|
|
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."""
|
2018-10-24 10:10:05 +00:00
|
|
|
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."""
|
2018-01-07 22:54:16 +00:00
|
|
|
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 = []
|
2016-09-30 19:57:24 +00:00
|
|
|
|
2016-11-05 23:36:20 +00:00
|
|
|
@ha.callback
|
2016-09-30 19:57:24 +00:00
|
|
|
def service_handler(call):
|
|
|
|
"""Service handler."""
|
|
|
|
calls.append(call)
|
|
|
|
|
2017-03-08 06:51:34 +00:00
|
|
|
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
|
|
|
|
2018-10-24 10:10:05 +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."""
|
2018-11-30 20:28:35 +00:00
|
|
|
with pytest.raises(ha.ServiceNotFound):
|
|
|
|
self.services.call('test_domain', 'i_do_not_exist', blocking=True)
|
2015-08-04 16:16:10 +00:00
|
|
|
|
2016-09-30 19:57:24 +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)
|
|
|
|
|
2017-03-08 06:51:34 +00:00
|
|
|
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'
|
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
assert self.services.call('test_domain', 'REGISTER_CALLS',
|
|
|
|
blocking=True)
|
2016-09-30 19:57:24 +00:00
|
|
|
self.hass.block_till_done()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 1 == len(calls)
|
2016-09-30 19:57:24 +00:00
|
|
|
|
2016-10-05 03:44:32 +00:00
|
|
|
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)
|
|
|
|
|
2017-03-08 06:51:34 +00:00
|
|
|
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'
|
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
assert self.services.call('test_domain', 'REGISTER_CALLS',
|
|
|
|
blocking=True)
|
2016-10-05 03:44:32 +00:00
|
|
|
self.hass.block_till_done()
|
2018-10-24 10:10:05 +00:00
|
|
|
assert 1 == len(calls)
|
2016-10-05 03:44:32 +00:00
|
|
|
|
2017-03-08 06:51:34 +00:00
|
|
|
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."""
|
|
|
|
|
2016-10-30 21:18:53 +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."""
|
2015-08-04 16:16:10 +00:00
|
|
|
self.config = ha.Config()
|
2018-10-24 10:10:05 +00:00
|
|
|
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'
|
2018-10-24 10:10:05 +00:00
|
|
|
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'
|
2018-10-24 10:10:05 +00:00
|
|
|
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."""
|
2016-10-03 07:04:43 +00:00
|
|
|
self.config.config_dir = '/tmp/ha-config'
|
2015-08-04 16:16:10 +00:00
|
|
|
expected = {
|
|
|
|
'latitude': None,
|
|
|
|
'longitude': None,
|
2017-01-20 07:55:29 +00:00
|
|
|
'elevation': None,
|
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',
|
2017-02-09 18:21:57 +00:00
|
|
|
'components': set(),
|
2016-10-03 07:04:43 +00:00
|
|
|
'config_dir': '/tmp/ha-config',
|
2017-06-25 22:10:30 +00:00
|
|
|
'whitelist_external_dirs': set(),
|
2015-10-26 04:00:22 +00:00
|
|
|
'version': __version__,
|
2015-08-04 16:16:10 +00:00
|
|
|
}
|
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
assert expected == self.config.as_dict()
|
2015-08-04 16:16:10 +00:00
|
|
|
|
2017-06-25 22:10:30 +00:00
|
|
|
def test_is_allowed_path(self):
|
|
|
|
"""Test is_allowed_path method."""
|
|
|
|
with TemporaryDirectory() as tmp_dir:
|
2018-01-10 09:48:17 +00:00
|
|
|
# The created dir is in /tmp. This is a symlink on OS X
|
|
|
|
# causing this test to fail unless we resolve path first.
|
2017-06-25 22:10:30 +00:00
|
|
|
self.config.whitelist_external_dirs = set((
|
2018-01-10 09:48:17 +00:00
|
|
|
os.path.realpath(tmp_dir),
|
2017-06-25 22:10:30 +00:00
|
|
|
))
|
|
|
|
|
|
|
|
test_file = os.path.join(tmp_dir, "test.jpg")
|
|
|
|
with open(test_file, "w") as tmp_file:
|
|
|
|
tmp_file.write("test")
|
|
|
|
|
|
|
|
valid = [
|
|
|
|
test_file,
|
2018-03-30 02:57:19 +00:00
|
|
|
tmp_dir,
|
|
|
|
os.path.join(tmp_dir, 'notfound321')
|
2017-06-25 22:10:30 +00:00
|
|
|
]
|
|
|
|
for path in valid:
|
|
|
|
assert self.config.is_allowed_path(path)
|
|
|
|
|
2017-07-03 05:24:08 +00:00
|
|
|
self.config.whitelist_external_dirs = set(('/home', '/var'))
|
2017-06-25 22:10:30 +00:00
|
|
|
|
|
|
|
unvalid = [
|
|
|
|
"/hass/config/secure",
|
|
|
|
"/etc/passwd",
|
|
|
|
"/root/secure_file",
|
2017-07-03 05:24:08 +00:00
|
|
|
"/var/../etc/passwd",
|
2017-06-25 22:10:30 +00:00
|
|
|
test_file,
|
|
|
|
]
|
|
|
|
for path in unvalid:
|
|
|
|
assert not self.config.is_allowed_path(path)
|
|
|
|
|
2018-10-24 10:10:05 +00:00
|
|
|
with pytest.raises(AssertionError):
|
2017-08-15 13:41:37 +00:00
|
|
|
self.config.is_allowed_path(None)
|
|
|
|
|
2015-08-04 16:16:10 +00:00
|
|
|
|
2017-02-10 17:00:17 +00:00
|
|
|
@patch('homeassistant.core.monotonic')
|
|
|
|
def test_create_timer(mock_monotonic, loop):
|
2016-09-20 06:39:49 +00:00
|
|
|
"""Test create timer."""
|
2017-02-10 17:00:17 +00:00
|
|
|
hass = MagicMock()
|
|
|
|
funcs = []
|
|
|
|
orig_callback = ha.callback
|
|
|
|
|
|
|
|
def mock_callback(func):
|
|
|
|
funcs.append(func)
|
|
|
|
return orig_callback(func)
|
|
|
|
|
2018-09-17 08:10:50 +00:00
|
|
|
mock_monotonic.side_effect = 10.2, 10.8, 11.3
|
2017-04-07 04:00:58 +00:00
|
|
|
|
|
|
|
with patch.object(ha, 'callback', mock_callback), \
|
|
|
|
patch('homeassistant.core.dt_util.utcnow',
|
2018-09-14 10:28:09 +00:00
|
|
|
return_value=datetime(2018, 12, 31, 3, 4, 5, 333333)):
|
2017-02-10 17:00:17 +00:00
|
|
|
ha._async_create_timer(hass)
|
|
|
|
|
2018-09-17 08:10:50 +00:00
|
|
|
assert len(funcs) == 2
|
|
|
|
fire_time_event, stop_timer = funcs
|
|
|
|
|
2018-09-14 10:28:09 +00:00
|
|
|
assert len(hass.loop.call_later.mock_calls) == 1
|
2018-09-17 08:10:50 +00:00
|
|
|
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
|
2018-09-14 10:28:09 +00:00
|
|
|
|
|
|
|
with patch('homeassistant.core.dt_util.utcnow',
|
2018-09-17 08:10:50 +00:00
|
|
|
return_value=datetime(2018, 12, 31, 3, 4, 6, 100000)):
|
|
|
|
callback(target)
|
2017-02-10 17:00:17 +00:00
|
|
|
|
|
|
|
assert len(hass.bus.async_listen_once.mock_calls) == 1
|
|
|
|
assert len(hass.bus.async_fire.mock_calls) == 1
|
2018-09-14 10:28:09 +00:00
|
|
|
assert len(hass.loop.call_later.mock_calls) == 2
|
2017-02-10 17:00:17 +00:00
|
|
|
|
2017-04-07 04:00:58 +00:00
|
|
|
event_type, callback = hass.bus.async_listen_once.mock_calls[0][1]
|
2017-02-10 17:00:17 +00:00
|
|
|
assert event_type == EVENT_HOMEASSISTANT_STOP
|
|
|
|
assert callback is stop_timer
|
|
|
|
|
2018-09-17 08:10:50 +00:00
|
|
|
delay, callback, target = hass.loop.call_later.mock_calls[1][1]
|
|
|
|
assert abs(delay - 0.9) < 0.001
|
2017-02-10 17:00:17 +00:00
|
|
|
assert callback is fire_time_event
|
2018-09-17 08:10:50 +00:00
|
|
|
assert abs(target - 12.2) < 0.001
|
2017-02-10 17:00:17 +00:00
|
|
|
|
|
|
|
event_type, event_data = hass.bus.async_fire.mock_calls[0][1]
|
|
|
|
assert event_type == EVENT_TIME_CHANGED
|
2018-09-17 08:10:50 +00:00
|
|
|
assert event_data[ATTR_NOW] == datetime(2018, 12, 31, 3, 4, 6, 100000)
|
2017-02-10 17:00:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
@patch('homeassistant.core.monotonic')
|
|
|
|
def test_timer_out_of_sync(mock_monotonic, loop):
|
|
|
|
"""Test create timer."""
|
|
|
|
hass = MagicMock()
|
|
|
|
funcs = []
|
|
|
|
orig_callback = ha.callback
|
2016-09-20 06:39:49 +00:00
|
|
|
|
2017-02-10 17:00:17 +00:00
|
|
|
def mock_callback(func):
|
|
|
|
funcs.append(func)
|
|
|
|
return orig_callback(func)
|
2016-09-20 06:39:49 +00:00
|
|
|
|
2018-09-17 08:10:50 +00:00
|
|
|
mock_monotonic.side_effect = 10.2, 13.3, 13.4
|
2016-09-20 06:39:49 +00:00
|
|
|
|
2017-04-07 04:00:58 +00:00
|
|
|
with patch.object(ha, 'callback', mock_callback), \
|
|
|
|
patch('homeassistant.core.dt_util.utcnow',
|
2018-09-14 10:28:09 +00:00
|
|
|
return_value=datetime(2018, 12, 31, 3, 4, 5, 333333)):
|
2017-04-07 04:00:58 +00:00
|
|
|
ha._async_create_timer(hass)
|
|
|
|
|
2018-09-17 08:10:50 +00:00
|
|
|
delay, callback, target = hass.loop.call_later.mock_calls[0][1]
|
2018-09-14 10:28:09 +00:00
|
|
|
|
2018-10-15 09:34:36 +00:00
|
|
|
with patch('homeassistant.core.dt_util.utcnow',
|
|
|
|
return_value=datetime(2018, 12, 31, 3, 4, 8, 200000)):
|
2018-09-17 08:10:50 +00:00
|
|
|
callback(target)
|
|
|
|
|
2018-10-15 09:34:36 +00:00
|
|
|
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
|
2018-09-14 10:28:09 +00:00
|
|
|
|
2017-04-07 04:00:58 +00:00
|
|
|
assert len(funcs) == 2
|
|
|
|
fire_time_event, stop_timer = funcs
|
2016-09-20 06:39:49 +00:00
|
|
|
|
2018-09-14 10:28:09 +00:00
|
|
|
assert len(hass.loop.call_later.mock_calls) == 2
|
2016-09-20 06:39:49 +00:00
|
|
|
|
2018-09-17 08:10:50 +00:00
|
|
|
delay, callback, target = hass.loop.call_later.mock_calls[1][1]
|
|
|
|
assert abs(delay - 0.8) < 0.001
|
2017-02-10 17:00:17 +00:00
|
|
|
assert callback is fire_time_event
|
2018-09-17 08:10:50 +00:00
|
|
|
assert abs(target - 14.2) < 0.001
|
2017-04-08 21:53:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
@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
|
2017-04-11 16:09:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
@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()
|
2018-09-11 19:40:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_service_executed_with_subservices(hass):
|
|
|
|
"""Test we block correctly till all services done."""
|
|
|
|
calls = async_mock_service(hass, 'test', 'inner')
|
2019-02-18 21:07:44 +00:00
|
|
|
context = ha.Context()
|
2018-09-11 19:40:35 +00:00
|
|
|
|
|
|
|
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)
|
|
|
|
|
2019-02-18 21:07:44 +00:00
|
|
|
await hass.services.async_call('test', 'outer', blocking=True,
|
|
|
|
context=context)
|
2018-09-11 19:40:35 +00:00
|
|
|
|
|
|
|
assert len(calls) == 4
|
|
|
|
assert [call.service for call in calls] == [
|
|
|
|
'outer', 'inner', 'inner', 'outer']
|
2019-02-18 21:07:44 +00:00
|
|
|
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)
|
|
|
|
}))
|
|
|
|
|
2019-02-18 21:07:44 +00:00
|
|
|
context = ha.Context()
|
2018-12-10 11:58:51 +00:00
|
|
|
await hass.services.async_call('test', 'service', {
|
|
|
|
'number': '23'
|
2019-02-18 21:07:44 +00:00
|
|
|
}, 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'
|
2019-02-18 21:07:44 +00:00
|
|
|
assert events[0].context is context
|
2018-12-10 11:58:51 +00:00
|
|
|
assert len(calls) == 1
|
|
|
|
assert calls[0].data['number'] == 23
|
2019-02-18 21:07:44 +00:00
|
|
|
assert calls[0].context is context
|