core/tests/helpers/test_entity_component.py

440 lines
15 KiB
Python
Raw Normal View History

2016-03-09 10:15:04 +00:00
"""The tests for the Entity component helper."""
# pylint: disable=protected-access
import asyncio
2016-01-31 02:55:52 +00:00
from collections import OrderedDict
import logging
import unittest
from unittest.mock import patch, Mock, MagicMock
from datetime import timedelta
2016-01-31 02:55:52 +00:00
import homeassistant.core as ha
import homeassistant.loader as loader
from homeassistant.components import group
from homeassistant.helpers.entity import Entity, generate_entity_id
from homeassistant.helpers.entity_component import (
EntityComponent, DEFAULT_SCAN_INTERVAL, SLOW_SETUP_WARNING)
from homeassistant.helpers import discovery
2016-01-31 08:55:46 +00:00
import homeassistant.util.dt as dt_util
2016-01-31 02:55:52 +00:00
2016-01-31 08:55:46 +00:00
from tests.common import (
get_test_home_assistant, MockPlatform, MockModule, fire_time_changed,
mock_coro)
2016-01-31 02:55:52 +00:00
_LOGGER = logging.getLogger(__name__)
DOMAIN = "test_domain"
class EntityTest(Entity):
2016-03-09 10:15:04 +00:00
"""Test for the Entity component."""
2016-01-31 02:55:52 +00:00
def __init__(self, **values):
2016-03-09 10:15:04 +00:00
"""Initialize an entity."""
2016-01-31 02:55:52 +00:00
self._values = values
if 'entity_id' in values:
self.entity_id = values['entity_id']
@property
def name(self):
2016-03-09 10:15:04 +00:00
"""Return the name of the entity."""
2016-01-31 02:55:52 +00:00
return self._handle('name')
@property
def should_poll(self):
2016-03-09 10:15:04 +00:00
"""Return the ste of the polling."""
2016-01-31 02:55:52 +00:00
return self._handle('should_poll')
@property
def unique_id(self):
2016-03-09 10:15:04 +00:00
"""Return the unique ID of the entity."""
2016-01-31 02:55:52 +00:00
return self._handle('unique_id')
def _handle(self, attr):
2016-03-09 10:15:04 +00:00
"""Helper for the attributes."""
2016-01-31 02:55:52 +00:00
if attr in self._values:
return self._values[attr]
return getattr(super(), attr)
class TestHelpersEntityComponent(unittest.TestCase):
2016-03-09 10:15:04 +00:00
"""Test homeassistant.helpers.entity_component module."""
2016-01-31 02:55:52 +00:00
def setUp(self): # pylint: disable=invalid-name
"""Initialize a test Home Assistant instance."""
self.hass = get_test_home_assistant()
def tearDown(self): # pylint: disable=invalid-name
"""Clean up the test Home Assistant instance."""
self.hass.stop()
def test_setting_up_group(self):
2016-03-09 10:15:04 +00:00
"""Setup the setting of a group."""
2016-01-31 02:55:52 +00:00
component = EntityComponent(_LOGGER, DOMAIN, self.hass,
group_name='everyone')
# No group after setup
assert len(self.hass.states.entity_ids()) == 0
2016-01-31 02:55:52 +00:00
component.add_entities([EntityTest(name='hello')])
# group exists
assert len(self.hass.states.entity_ids()) == 2
assert self.hass.states.entity_ids('group') == ['group.everyone']
2016-01-31 02:55:52 +00:00
group = self.hass.states.get('group.everyone')
assert group.attributes.get('entity_id') == ('test_domain.hello',)
2016-01-31 02:55:52 +00:00
# group extended
component.add_entities([EntityTest(name='hello2')])
assert len(self.hass.states.entity_ids()) == 3
2016-01-31 02:55:52 +00:00
group = self.hass.states.get('group.everyone')
assert sorted(group.attributes.get('entity_id')) == \
['test_domain.hello', 'test_domain.hello2']
2016-01-31 02:55:52 +00:00
2016-01-31 08:55:46 +00:00
def test_polling_only_updates_entities_it_should_poll(self):
2016-03-09 10:15:04 +00:00
"""Test the polling of only updated entities."""
component = EntityComponent(
_LOGGER, DOMAIN, self.hass, timedelta(seconds=20))
2016-01-31 02:55:52 +00:00
no_poll_ent = EntityTest(should_poll=False)
no_poll_ent.async_update = Mock()
2016-01-31 02:55:52 +00:00
poll_ent = EntityTest(should_poll=True)
poll_ent.async_update = Mock()
2016-01-31 02:55:52 +00:00
2016-01-31 08:55:46 +00:00
component.add_entities([no_poll_ent, poll_ent])
2016-01-31 02:55:52 +00:00
no_poll_ent.async_update.reset_mock()
poll_ent.async_update.reset_mock()
2016-01-31 02:55:52 +00:00
fire_time_changed(self.hass, dt_util.utcnow() + timedelta(seconds=20))
self.hass.block_till_done()
2016-01-31 02:55:52 +00:00
assert not no_poll_ent.async_update.called
assert poll_ent.async_update.called
2016-01-31 02:55:52 +00:00
def test_update_state_adds_entities(self):
"""Test if updating poll entities cause an entity to be added works."""
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
ent1 = EntityTest()
ent2 = EntityTest(should_poll=True)
component.add_entities([ent2])
assert 1 == len(self.hass.states.entity_ids())
ent2.update = lambda *_: component.add_entities([ent1])
2016-01-31 08:55:46 +00:00
fire_time_changed(
self.hass, dt_util.utcnow() + DEFAULT_SCAN_INTERVAL
)
self.hass.block_till_done()
2016-01-31 08:55:46 +00:00
2016-01-31 02:55:52 +00:00
assert 2 == len(self.hass.states.entity_ids())
def test_update_state_adds_entities_with_update_befor_add_true(self):
"""Test if call update befor add to state machine."""
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
ent = EntityTest()
ent.update = Mock(spec_set=True)
component.add_entities([ent], True)
self.hass.block_till_done()
assert 1 == len(self.hass.states.entity_ids())
assert ent.update.called
def test_update_state_adds_entities_with_update_befor_add_false(self):
"""Test if not call update befor add to state machine."""
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
ent = EntityTest()
ent.update = Mock(spec_set=True)
component.add_entities([ent], False)
self.hass.block_till_done()
assert 1 == len(self.hass.states.entity_ids())
assert not ent.update.called
def test_adds_entities_with_update_befor_add_true_deadlock_protect(self):
"""Test if call update befor add to state machine.
It need to run update inside executor and never call
async_add_entities with True
"""
call = []
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
@asyncio.coroutine
def async_add_entities_fake(entities, update_befor_add):
"""Fake add_entities_call."""
call.append(update_befor_add)
component._platforms['core'].async_add_entities = \
async_add_entities_fake
ent = EntityTest()
ent.update = Mock(spec_set=True)
component.add_entities([ent], True)
assert ent.update.called
assert len(call) == 1
assert not call[0]
2016-01-31 02:55:52 +00:00
def test_not_adding_duplicate_entities(self):
2016-03-09 10:15:04 +00:00
"""Test for not adding duplicate entities."""
2016-01-31 02:55:52 +00:00
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
assert 0 == len(self.hass.states.entity_ids())
component.add_entities([None, EntityTest(unique_id='not_very_unique')])
assert 1 == len(self.hass.states.entity_ids())
component.add_entities([EntityTest(unique_id='not_very_unique')])
assert 1 == len(self.hass.states.entity_ids())
def test_not_assigning_entity_id_if_prescribes_one(self):
2016-03-09 10:15:04 +00:00
"""Test for not assigning an entity ID."""
2016-01-31 02:55:52 +00:00
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
assert 'hello.world' not in self.hass.states.entity_ids()
component.add_entities([EntityTest(entity_id='hello.world')])
assert 'hello.world' in self.hass.states.entity_ids()
def test_extract_from_service_returns_all_if_no_entity_id(self):
2016-03-09 10:15:04 +00:00
"""Test the extraction of everything from service."""
2016-01-31 02:55:52 +00:00
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
component.add_entities([
EntityTest(name='test_1'),
EntityTest(name='test_2'),
])
call = ha.ServiceCall('test', 'service')
assert ['test_domain.test_1', 'test_domain.test_2'] == \
sorted(ent.entity_id for ent in
component.extract_from_service(call))
def test_extract_from_service_filter_out_non_existing_entities(self):
2016-03-09 10:15:04 +00:00
"""Test the extraction of non existing entities from service."""
2016-01-31 02:55:52 +00:00
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
component.add_entities([
EntityTest(name='test_1'),
EntityTest(name='test_2'),
])
call = ha.ServiceCall('test', 'service', {
'entity_id': ['test_domain.test_2', 'test_domain.non_exist']
})
assert ['test_domain.test_2'] == \
[ent.entity_id for ent in component.extract_from_service(call)]
def test_extract_from_service_no_group_expand(self):
"""Test not expanding a group."""
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
test_group = group.Group.create_group(
self.hass, 'test_group', ['light.Ceiling', 'light.Kitchen'])
component.add_entities([test_group])
call = ha.ServiceCall('test', 'service', {
'entity_id': ['group.test_group']
})
extracted = component.extract_from_service(call, expand_group=False)
self.assertEqual([test_group], extracted)
2016-01-31 02:55:52 +00:00
def test_setup_loads_platforms(self):
2016-03-09 10:15:04 +00:00
"""Test the loading of the platforms."""
2016-01-31 02:55:52 +00:00
component_setup = Mock(return_value=True)
platform_setup = Mock(return_value=None)
loader.set_component(
'test_component',
MockModule('test_component', setup=component_setup))
loader.set_component('test_domain.mod2',
MockPlatform(platform_setup, ['test_component']))
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
assert not component_setup.called
assert not platform_setup.called
component.setup({
DOMAIN: {
'platform': 'mod2',
}
})
self.hass.block_till_done()
2016-01-31 02:55:52 +00:00
assert component_setup.called
assert platform_setup.called
def test_setup_recovers_when_setup_raises(self):
2016-03-09 10:15:04 +00:00
"""Test the setup if exceptions are happening."""
2016-01-31 02:55:52 +00:00
platform1_setup = Mock(side_effect=Exception('Broken'))
platform2_setup = Mock(return_value=None)
loader.set_component('test_domain.mod1', MockPlatform(platform1_setup))
loader.set_component('test_domain.mod2', MockPlatform(platform2_setup))
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
assert not platform1_setup.called
assert not platform2_setup.called
component.setup(OrderedDict([
(DOMAIN, {'platform': 'mod1'}),
("{} 2".format(DOMAIN), {'platform': 'non_exist'}),
("{} 3".format(DOMAIN), {'platform': 'mod2'}),
]))
self.hass.block_till_done()
2016-01-31 02:55:52 +00:00
assert platform1_setup.called
assert platform2_setup.called
@patch('homeassistant.helpers.entity_component.EntityComponent'
2017-02-16 07:19:34 +00:00
'._async_setup_platform', return_value=mock_coro())
@patch('homeassistant.setup.async_setup_component',
2017-02-16 07:19:34 +00:00
return_value=mock_coro(True))
def test_setup_does_discovery(self, mock_setup_component, mock_setup):
2016-03-09 10:15:04 +00:00
"""Test setup for discovery."""
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
2016-01-31 02:55:52 +00:00
component.setup({})
discovery.load_platform(self.hass, DOMAIN, 'platform_test',
{'msg': 'discovery_info'})
2016-01-31 02:55:52 +00:00
self.hass.block_till_done()
2016-01-31 02:55:52 +00:00
assert mock_setup.called
assert ('platform_test', {}, {'msg': 'discovery_info'}) == \
2016-01-31 02:55:52 +00:00
mock_setup.call_args[0]
2016-01-31 08:55:46 +00:00
@patch('homeassistant.helpers.entity_component.'
'async_track_time_interval')
2016-01-31 08:55:46 +00:00
def test_set_scan_interval_via_config(self, mock_track):
2016-03-09 10:15:04 +00:00
"""Test the setting of the scan interval via configuration."""
2016-01-31 08:55:46 +00:00
def platform_setup(hass, config, add_devices, discovery_info=None):
2016-03-09 10:15:04 +00:00
"""Test the platform setup."""
2016-01-31 08:55:46 +00:00
add_devices([EntityTest(should_poll=True)])
loader.set_component('test_domain.platform',
MockPlatform(platform_setup))
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
component.setup({
DOMAIN: {
'platform': 'platform',
'scan_interval': timedelta(seconds=30),
2016-01-31 08:55:46 +00:00
}
})
self.hass.block_till_done()
2016-01-31 08:55:46 +00:00
assert mock_track.called
assert timedelta(seconds=30) == mock_track.call_args[0][2]
2016-01-31 08:55:46 +00:00
@patch('homeassistant.helpers.entity_component.'
'async_track_time_interval')
2016-01-31 08:55:46 +00:00
def test_set_scan_interval_via_platform(self, mock_track):
2016-03-09 10:15:04 +00:00
"""Test the setting of the scan interval via platform."""
2016-01-31 08:55:46 +00:00
def platform_setup(hass, config, add_devices, discovery_info=None):
2016-03-09 10:15:04 +00:00
"""Test the platform setup."""
2016-01-31 08:55:46 +00:00
add_devices([EntityTest(should_poll=True)])
platform = MockPlatform(platform_setup)
platform.SCAN_INTERVAL = timedelta(seconds=30)
2016-01-31 08:55:46 +00:00
loader.set_component('test_domain.platform', platform)
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
component.setup({
DOMAIN: {
'platform': 'platform',
}
})
self.hass.block_till_done()
2016-01-31 08:55:46 +00:00
assert mock_track.called
assert timedelta(seconds=30) == mock_track.call_args[0][2]
2016-04-23 04:34:49 +00:00
def test_set_entity_namespace_via_config(self):
"""Test setting an entity namespace."""
def platform_setup(hass, config, add_devices, discovery_info=None):
"""Test the platform setup."""
add_devices([
EntityTest(name='beer'),
EntityTest(name=None),
])
platform = MockPlatform(platform_setup)
loader.set_component('test_domain.platform', platform)
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
component.setup({
DOMAIN: {
'platform': 'platform',
'entity_namespace': 'yummy'
}
})
self.hass.block_till_done()
2016-04-23 04:34:49 +00:00
assert sorted(self.hass.states.entity_ids()) == \
['test_domain.yummy_beer', 'test_domain.yummy_unnamed_device']
def test_adding_entities_with_generator_and_thread_callback(self):
"""Test generator in add_entities that calls thread method.
We should make sure we resolve the generator to a list before passing
it into an async context.
"""
component = EntityComponent(_LOGGER, DOMAIN, self.hass)
def create_entity(number):
"""Create entity helper."""
entity = EntityTest()
entity.entity_id = generate_entity_id(component.entity_id_format,
'Number', hass=self.hass)
return entity
component.add_entities(create_entity(i) for i in range(2))
@asyncio.coroutine
def test_platform_warn_slow_setup(hass):
"""Warn we log when platform setup takes a long time."""
platform = MockPlatform()
loader.set_component('test_domain.platform', platform)
component = EntityComponent(_LOGGER, DOMAIN, hass)
with patch.object(hass.loop, 'call_later', MagicMock()) \
as mock_call:
yield from component.async_setup({
DOMAIN: {
'platform': 'platform',
}
})
assert mock_call.called
assert len(mock_call.mock_calls) == 2
timeout, logger_method = mock_call.mock_calls[0][1][:2]
assert timeout == SLOW_SETUP_WARNING
assert logger_method == _LOGGER.warning
assert mock_call().cancel.called