2016-03-09 09:25:50 +00:00
|
|
|
"""Test the entity helper."""
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=protected-access
|
2016-09-30 19:57:24 +00:00
|
|
|
import asyncio
|
2022-04-25 17:10:42 +00:00
|
|
|
import dataclasses
|
2018-08-20 15:39:53 +00:00
|
|
|
from datetime import timedelta
|
2019-12-09 15:52:24 +00:00
|
|
|
import threading
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import MagicMock, PropertyMock, patch
|
2016-09-30 19:57:24 +00:00
|
|
|
|
|
|
|
import pytest
|
2022-02-08 22:00:53 +00:00
|
|
|
import voluptuous as vol
|
2015-04-23 13:41:41 +00:00
|
|
|
|
2021-10-11 21:15:32 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_ATTRIBUTION,
|
|
|
|
ATTR_DEVICE_CLASS,
|
2022-06-28 16:38:05 +00:00
|
|
|
ATTR_FRIENDLY_NAME,
|
2021-10-11 21:15:32 +00:00
|
|
|
STATE_UNAVAILABLE,
|
|
|
|
STATE_UNKNOWN,
|
|
|
|
)
|
2021-04-10 06:19:16 +00:00
|
|
|
from homeassistant.core import Context, HomeAssistantError
|
2022-06-28 16:38:05 +00:00
|
|
|
from homeassistant.helpers import device_registry as dr, entity, entity_registry as er
|
2015-04-23 13:41:41 +00:00
|
|
|
|
2020-08-19 12:57:38 +00:00
|
|
|
from tests.common import (
|
|
|
|
MockConfigEntry,
|
|
|
|
MockEntity,
|
|
|
|
MockEntityPlatform,
|
2022-06-28 16:38:05 +00:00
|
|
|
MockPlatform,
|
2020-08-19 12:57:38 +00:00
|
|
|
get_test_home_assistant,
|
|
|
|
mock_registry,
|
|
|
|
)
|
2016-02-14 23:08:23 +00:00
|
|
|
|
2015-04-23 13:41:41 +00:00
|
|
|
|
2016-09-30 19:57:24 +00:00
|
|
|
def test_generate_entity_id_requires_hass_or_ids():
|
|
|
|
"""Ensure we require at least hass or current ids."""
|
|
|
|
with pytest.raises(ValueError):
|
2020-04-07 16:33:23 +00:00
|
|
|
entity.generate_entity_id("test.{}", "hello world")
|
2016-09-30 19:57:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_generate_entity_id_given_keys():
|
|
|
|
"""Test generating an entity id given current ids."""
|
2019-07-31 19:25:30 +00:00
|
|
|
assert (
|
|
|
|
entity.generate_entity_id(
|
2020-04-07 16:33:23 +00:00
|
|
|
"test.{}",
|
|
|
|
"overwrite hidden true",
|
|
|
|
current_ids=["test.overwrite_hidden_true"],
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
== "test.overwrite_hidden_true_2"
|
|
|
|
)
|
|
|
|
assert (
|
|
|
|
entity.generate_entity_id(
|
2020-04-07 16:33:23 +00:00
|
|
|
"test.{}", "overwrite hidden true", current_ids=["test.another_entity"]
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
== "test.overwrite_hidden_true"
|
|
|
|
)
|
2016-09-30 19:57:24 +00:00
|
|
|
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def test_async_update_support(hass):
|
2016-09-30 19:57:24 +00:00
|
|
|
"""Test async update getting called."""
|
|
|
|
sync_update = []
|
|
|
|
async_update = []
|
|
|
|
|
|
|
|
class AsyncEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""A test entity."""
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
entity_id = "sensor.test"
|
2016-09-30 19:57:24 +00:00
|
|
|
|
|
|
|
def update(self):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Update entity."""
|
2016-09-30 19:57:24 +00:00
|
|
|
sync_update.append([1])
|
|
|
|
|
|
|
|
ent = AsyncEntity()
|
2017-05-26 15:28:07 +00:00
|
|
|
ent.hass = hass
|
2016-09-30 19:57:24 +00:00
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
await ent.async_update_ha_state(True)
|
2016-09-30 19:57:24 +00:00
|
|
|
|
|
|
|
assert len(sync_update) == 1
|
|
|
|
assert len(async_update) == 0
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def async_update_func():
|
2016-10-01 04:38:39 +00:00
|
|
|
"""Async update."""
|
|
|
|
async_update.append(1)
|
|
|
|
|
|
|
|
ent.async_update = async_update_func
|
2016-09-30 19:57:24 +00:00
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
await ent.async_update_ha_state(True)
|
2016-09-30 19:57:24 +00:00
|
|
|
|
|
|
|
assert len(sync_update) == 1
|
|
|
|
assert len(async_update) == 1
|
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class TestHelpersEntity:
|
2016-03-09 09:25:50 +00:00
|
|
|
"""Test homeassistant.helpers.entity module."""
|
2015-04-23 13:41:41 +00:00
|
|
|
|
2016-09-30 19:57:24 +00:00
|
|
|
def setup_method(self, method):
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Set up things to be run when tests are started."""
|
2015-04-23 13:41:41 +00:00
|
|
|
self.entity = entity.Entity()
|
2019-07-31 19:25:30 +00:00
|
|
|
self.entity.entity_id = "test.overwrite_hidden_true"
|
2016-02-14 23:08:23 +00:00
|
|
|
self.hass = self.entity.hass = get_test_home_assistant()
|
2017-03-04 23:10:36 +00:00
|
|
|
self.entity.schedule_update_ha_state()
|
|
|
|
self.hass.block_till_done()
|
2015-04-23 13:41:41 +00:00
|
|
|
|
2016-09-30 19:57:24 +00:00
|
|
|
def teardown_method(self, method):
|
2016-03-09 09:25:50 +00:00
|
|
|
"""Stop everything that was started."""
|
2016-09-30 19:57:24 +00:00
|
|
|
self.hass.stop()
|
2015-04-23 13:41:41 +00:00
|
|
|
|
2016-06-22 16:13:18 +00:00
|
|
|
def test_generate_entity_id_given_hass(self):
|
|
|
|
"""Test generating an entity id given hass object."""
|
2019-07-31 19:25:30 +00:00
|
|
|
fmt = "test.{}"
|
|
|
|
assert (
|
|
|
|
entity.generate_entity_id(fmt, "overwrite hidden true", hass=self.hass)
|
|
|
|
== "test.overwrite_hidden_true_2"
|
|
|
|
)
|
2016-10-01 04:38:39 +00:00
|
|
|
|
2017-02-11 04:46:15 +00:00
|
|
|
def test_device_class(self):
|
|
|
|
"""Test device class attribute."""
|
|
|
|
state = self.hass.states.get(self.entity.entity_id)
|
|
|
|
assert state.attributes.get(ATTR_DEVICE_CLASS) is None
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.entity.Entity.device_class", new="test_class"
|
|
|
|
):
|
2017-03-04 23:10:36 +00:00
|
|
|
self.entity.schedule_update_ha_state()
|
|
|
|
self.hass.block_till_done()
|
2017-02-11 04:46:15 +00:00
|
|
|
state = self.hass.states.get(self.entity.entity_id)
|
2019-07-31 19:25:30 +00:00
|
|
|
assert state.attributes.get(ATTR_DEVICE_CLASS) == "test_class"
|
2017-03-14 16:26:55 +00:00
|
|
|
|
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
async def test_warn_slow_update(hass, caplog):
|
2017-03-14 16:26:55 +00:00
|
|
|
"""Warn we log when entity update takes a long time."""
|
|
|
|
update_call = False
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def async_update():
|
2017-03-14 16:26:55 +00:00
|
|
|
"""Mock async update."""
|
|
|
|
nonlocal update_call
|
2020-10-05 13:28:15 +00:00
|
|
|
await asyncio.sleep(0.00001)
|
2017-03-14 16:26:55 +00:00
|
|
|
update_call = True
|
|
|
|
|
|
|
|
mock_entity = entity.Entity()
|
|
|
|
mock_entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity.entity_id = "comp_test.test_entity"
|
2017-03-14 16:26:55 +00:00
|
|
|
mock_entity.async_update = async_update
|
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
fast_update_time = 0.0000001
|
2017-03-14 16:26:55 +00:00
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
with patch.object(entity, "SLOW_UPDATE_WARNING", fast_update_time):
|
|
|
|
await mock_entity.async_update_ha_state(True)
|
|
|
|
assert str(fast_update_time) in caplog.text
|
|
|
|
assert mock_entity.entity_id in caplog.text
|
2017-03-14 16:26:55 +00:00
|
|
|
assert update_call
|
|
|
|
|
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
async def test_warn_slow_update_with_exception(hass, caplog):
|
2017-03-14 16:26:55 +00:00
|
|
|
"""Warn we log when entity update takes a long time and trow exception."""
|
|
|
|
update_call = False
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def async_update():
|
2017-03-14 16:26:55 +00:00
|
|
|
"""Mock async update."""
|
|
|
|
nonlocal update_call
|
|
|
|
update_call = True
|
2020-10-05 13:28:15 +00:00
|
|
|
await asyncio.sleep(0.00001)
|
2017-03-14 16:26:55 +00:00
|
|
|
raise AssertionError("Fake update error")
|
|
|
|
|
|
|
|
mock_entity = entity.Entity()
|
|
|
|
mock_entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity.entity_id = "comp_test.test_entity"
|
2017-03-14 16:26:55 +00:00
|
|
|
mock_entity.async_update = async_update
|
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
fast_update_time = 0.0000001
|
2017-03-14 16:26:55 +00:00
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
with patch.object(entity, "SLOW_UPDATE_WARNING", fast_update_time):
|
|
|
|
await mock_entity.async_update_ha_state(True)
|
|
|
|
assert str(fast_update_time) in caplog.text
|
|
|
|
assert mock_entity.entity_id in caplog.text
|
2017-03-14 16:26:55 +00:00
|
|
|
assert update_call
|
2017-09-12 08:01:03 +00:00
|
|
|
|
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
async def test_warn_slow_device_update_disabled(hass, caplog):
|
2017-10-22 15:40:00 +00:00
|
|
|
"""Disable slow update warning with async_device_update."""
|
|
|
|
update_call = False
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def async_update():
|
2017-10-22 15:40:00 +00:00
|
|
|
"""Mock async update."""
|
|
|
|
nonlocal update_call
|
2020-10-05 13:28:15 +00:00
|
|
|
await asyncio.sleep(0.00001)
|
2017-10-22 15:40:00 +00:00
|
|
|
update_call = True
|
|
|
|
|
|
|
|
mock_entity = entity.Entity()
|
|
|
|
mock_entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity.entity_id = "comp_test.test_entity"
|
2017-10-22 15:40:00 +00:00
|
|
|
mock_entity.async_update = async_update
|
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
fast_update_time = 0.0000001
|
2017-10-22 15:40:00 +00:00
|
|
|
|
2020-10-05 13:28:15 +00:00
|
|
|
with patch.object(entity, "SLOW_UPDATE_WARNING", fast_update_time):
|
|
|
|
await mock_entity.async_device_update(warning=False)
|
|
|
|
assert str(fast_update_time) not in caplog.text
|
|
|
|
assert mock_entity.entity_id not in caplog.text
|
2017-10-22 15:40:00 +00:00
|
|
|
assert update_call
|
|
|
|
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def test_async_schedule_update_ha_state(hass):
|
2017-09-12 08:01:03 +00:00
|
|
|
"""Warn we log when entity update takes a long time and trow exception."""
|
|
|
|
update_call = False
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def async_update():
|
2017-09-12 08:01:03 +00:00
|
|
|
"""Mock async update."""
|
|
|
|
nonlocal update_call
|
|
|
|
update_call = True
|
|
|
|
|
|
|
|
mock_entity = entity.Entity()
|
|
|
|
mock_entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity.entity_id = "comp_test.test_entity"
|
2017-09-12 08:01:03 +00:00
|
|
|
mock_entity.async_update = async_update
|
|
|
|
|
|
|
|
mock_entity.async_schedule_update_ha_state(True)
|
2020-04-07 16:33:23 +00:00
|
|
|
await hass.async_block_till_done()
|
2017-09-12 08:01:03 +00:00
|
|
|
|
|
|
|
assert update_call is True
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
|
Switch on/off all lights, and wait for the result (#27078)
* Switch on/off all lights, and wait for the result
Reuses the parallel_updates semaphore.
This is a small crutch which serializes platforms which already do tis
for updates. Platforms which can parallelize everything, this makes it
go faster
* Fix broken unittest
With manual validation, with help from @frenck, we found out that the
assertions are wrong and the test should be failing.
The sequence requested is
OFF
ON
without cancelation, this code should result in:
off,off,off,on,on,on
testable, by adding a `await hass.async_block_till_done()` between the
off and on call.
with cancelation. there should be less off call's so
off,on,on,on
* Adding tests for async_request_call
* Process review feedback
* Switch gather with wait
* :shirt: running black
2019-10-06 15:23:12 +00:00
|
|
|
async def test_async_async_request_call_without_lock(hass):
|
|
|
|
"""Test for async_requests_call works without a lock."""
|
|
|
|
updates = []
|
|
|
|
|
|
|
|
class AsyncEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test entity."""
|
|
|
|
|
Switch on/off all lights, and wait for the result (#27078)
* Switch on/off all lights, and wait for the result
Reuses the parallel_updates semaphore.
This is a small crutch which serializes platforms which already do tis
for updates. Platforms which can parallelize everything, this makes it
go faster
* Fix broken unittest
With manual validation, with help from @frenck, we found out that the
assertions are wrong and the test should be failing.
The sequence requested is
OFF
ON
without cancelation, this code should result in:
off,off,off,on,on,on
testable, by adding a `await hass.async_block_till_done()` between the
off and on call.
with cancelation. there should be less off call's so
off,on,on,on
* Adding tests for async_request_call
* Process review feedback
* Switch gather with wait
* :shirt: running black
2019-10-06 15:23:12 +00:00
|
|
|
def __init__(self, entity_id):
|
|
|
|
"""Initialize Async test entity."""
|
|
|
|
self.entity_id = entity_id
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
async def testhelper(self, count):
|
|
|
|
"""Helper function."""
|
|
|
|
updates.append(count)
|
|
|
|
|
|
|
|
ent_1 = AsyncEntity("light.test_1")
|
|
|
|
ent_2 = AsyncEntity("light.test_2")
|
|
|
|
try:
|
|
|
|
job1 = ent_1.async_request_call(ent_1.testhelper(1))
|
|
|
|
job2 = ent_2.async_request_call(ent_2.testhelper(2))
|
|
|
|
|
|
|
|
await asyncio.wait([job1, job2])
|
|
|
|
while True:
|
|
|
|
if len(updates) >= 2:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
finally:
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert len(updates) == 2
|
|
|
|
updates.sort()
|
|
|
|
assert updates == [1, 2]
|
|
|
|
|
|
|
|
|
|
|
|
async def test_async_async_request_call_with_lock(hass):
|
|
|
|
"""Test for async_requests_call works with a semaphore."""
|
|
|
|
updates = []
|
|
|
|
|
|
|
|
test_semaphore = asyncio.Semaphore(1)
|
|
|
|
|
|
|
|
class AsyncEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test entity."""
|
|
|
|
|
Switch on/off all lights, and wait for the result (#27078)
* Switch on/off all lights, and wait for the result
Reuses the parallel_updates semaphore.
This is a small crutch which serializes platforms which already do tis
for updates. Platforms which can parallelize everything, this makes it
go faster
* Fix broken unittest
With manual validation, with help from @frenck, we found out that the
assertions are wrong and the test should be failing.
The sequence requested is
OFF
ON
without cancelation, this code should result in:
off,off,off,on,on,on
testable, by adding a `await hass.async_block_till_done()` between the
off and on call.
with cancelation. there should be less off call's so
off,on,on,on
* Adding tests for async_request_call
* Process review feedback
* Switch gather with wait
* :shirt: running black
2019-10-06 15:23:12 +00:00
|
|
|
def __init__(self, entity_id, lock):
|
|
|
|
"""Initialize Async test entity."""
|
|
|
|
self.entity_id = entity_id
|
|
|
|
self.hass = hass
|
|
|
|
self.parallel_updates = lock
|
|
|
|
|
|
|
|
async def testhelper(self, count):
|
|
|
|
"""Helper function."""
|
|
|
|
updates.append(count)
|
|
|
|
|
|
|
|
ent_1 = AsyncEntity("light.test_1", test_semaphore)
|
|
|
|
ent_2 = AsyncEntity("light.test_2", test_semaphore)
|
|
|
|
|
|
|
|
try:
|
|
|
|
assert test_semaphore.locked() is False
|
|
|
|
await test_semaphore.acquire()
|
|
|
|
assert test_semaphore.locked()
|
|
|
|
|
|
|
|
job1 = ent_1.async_request_call(ent_1.testhelper(1))
|
|
|
|
job2 = ent_2.async_request_call(ent_2.testhelper(2))
|
|
|
|
|
|
|
|
hass.async_create_task(job1)
|
|
|
|
hass.async_create_task(job2)
|
|
|
|
|
|
|
|
assert len(updates) == 0
|
|
|
|
assert updates == []
|
|
|
|
assert test_semaphore._value == 0
|
|
|
|
|
|
|
|
test_semaphore.release()
|
|
|
|
|
|
|
|
while True:
|
|
|
|
if len(updates) >= 2:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
finally:
|
|
|
|
test_semaphore.release()
|
|
|
|
|
|
|
|
assert len(updates) == 2
|
|
|
|
updates.sort()
|
|
|
|
assert updates == [1, 2]
|
|
|
|
|
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
async def test_async_parallel_updates_with_zero(hass):
|
2018-01-29 22:37:19 +00:00
|
|
|
"""Test parallel updates with 0 (disabled)."""
|
2017-10-19 08:56:25 +00:00
|
|
|
updates = []
|
2019-03-26 06:53:36 +00:00
|
|
|
test_lock = asyncio.Event()
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
class AsyncEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test entity."""
|
|
|
|
|
2017-10-19 08:56:25 +00:00
|
|
|
def __init__(self, entity_id, count):
|
|
|
|
"""Initialize Async test entity."""
|
|
|
|
self.entity_id = entity_id
|
|
|
|
self.hass = hass
|
|
|
|
self._count = count
|
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
async def async_update(self):
|
2017-10-19 08:56:25 +00:00
|
|
|
"""Test update."""
|
|
|
|
updates.append(self._count)
|
2019-03-26 06:53:36 +00:00
|
|
|
await test_lock.wait()
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
ent_1 = AsyncEntity("sensor.test_1", 1)
|
|
|
|
ent_2 = AsyncEntity("sensor.test_2", 2)
|
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
try:
|
|
|
|
ent_1.async_schedule_update_ha_state(True)
|
|
|
|
ent_2.async_schedule_update_ha_state(True)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
while True:
|
|
|
|
if len(updates) >= 2:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
assert len(updates) == 2
|
|
|
|
assert updates == [1, 2]
|
|
|
|
finally:
|
|
|
|
test_lock.set()
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
async def test_async_parallel_updates_with_zero_on_sync_update(hass):
|
|
|
|
"""Test parallel updates with 0 (disabled)."""
|
|
|
|
updates = []
|
|
|
|
test_lock = threading.Event()
|
|
|
|
|
|
|
|
class AsyncEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test entity."""
|
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
def __init__(self, entity_id, count):
|
|
|
|
"""Initialize Async test entity."""
|
|
|
|
self.entity_id = entity_id
|
|
|
|
self.hass = hass
|
|
|
|
self._count = count
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Test update."""
|
|
|
|
updates.append(self._count)
|
|
|
|
if not test_lock.wait(timeout=1):
|
|
|
|
# if timeout populate more data to fail the test
|
|
|
|
updates.append(self._count)
|
|
|
|
|
|
|
|
ent_1 = AsyncEntity("sensor.test_1", 1)
|
|
|
|
ent_2 = AsyncEntity("sensor.test_2", 2)
|
|
|
|
|
|
|
|
try:
|
|
|
|
ent_1.async_schedule_update_ha_state(True)
|
|
|
|
ent_2.async_schedule_update_ha_state(True)
|
|
|
|
|
|
|
|
while True:
|
|
|
|
if len(updates) >= 2:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
|
|
|
|
assert len(updates) == 2
|
|
|
|
assert updates == [1, 2]
|
|
|
|
finally:
|
|
|
|
test_lock.set()
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
|
|
|
|
|
|
|
|
async def test_async_parallel_updates_with_one(hass):
|
2018-01-29 22:37:19 +00:00
|
|
|
"""Test parallel updates with 1 (sequential)."""
|
2017-10-19 08:56:25 +00:00
|
|
|
updates = []
|
2019-03-26 06:53:36 +00:00
|
|
|
test_lock = asyncio.Lock()
|
|
|
|
test_semaphore = asyncio.Semaphore(1)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
class AsyncEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test entity."""
|
|
|
|
|
2017-10-19 08:56:25 +00:00
|
|
|
def __init__(self, entity_id, count):
|
|
|
|
"""Initialize Async test entity."""
|
|
|
|
self.entity_id = entity_id
|
|
|
|
self.hass = hass
|
|
|
|
self._count = count
|
2018-01-29 22:37:19 +00:00
|
|
|
self.parallel_updates = test_semaphore
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
async def async_update(self):
|
2017-10-19 08:56:25 +00:00
|
|
|
"""Test update."""
|
|
|
|
updates.append(self._count)
|
2019-03-26 06:53:36 +00:00
|
|
|
await test_lock.acquire()
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
ent_1 = AsyncEntity("sensor.test_1", 1)
|
|
|
|
ent_2 = AsyncEntity("sensor.test_2", 2)
|
|
|
|
ent_3 = AsyncEntity("sensor.test_3", 3)
|
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
await test_lock.acquire()
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
try:
|
|
|
|
ent_1.async_schedule_update_ha_state(True)
|
|
|
|
ent_2.async_schedule_update_ha_state(True)
|
|
|
|
ent_3.async_schedule_update_ha_state(True)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
while True:
|
|
|
|
if len(updates) >= 1:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
assert len(updates) == 1
|
|
|
|
assert updates == [1]
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
updates.clear()
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
while True:
|
|
|
|
if len(updates) >= 1:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
assert len(updates) == 1
|
|
|
|
assert updates == [2]
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
updates.clear()
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
while True:
|
|
|
|
if len(updates) >= 1:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
assert len(updates) == 1
|
|
|
|
assert updates == [3]
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
updates.clear()
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
finally:
|
|
|
|
# we may have more than one lock need to release in case test failed
|
|
|
|
for _ in updates:
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
test_lock.release()
|
|
|
|
|
|
|
|
|
|
|
|
async def test_async_parallel_updates_with_two(hass):
|
2018-01-29 22:37:19 +00:00
|
|
|
"""Test parallel updates with 2 (parallel)."""
|
2017-10-19 08:56:25 +00:00
|
|
|
updates = []
|
2019-03-26 06:53:36 +00:00
|
|
|
test_lock = asyncio.Lock()
|
|
|
|
test_semaphore = asyncio.Semaphore(2)
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
class AsyncEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test entity."""
|
|
|
|
|
2017-10-19 08:56:25 +00:00
|
|
|
def __init__(self, entity_id, count):
|
|
|
|
"""Initialize Async test entity."""
|
|
|
|
self.entity_id = entity_id
|
|
|
|
self.hass = hass
|
|
|
|
self._count = count
|
2018-01-29 22:37:19 +00:00
|
|
|
self.parallel_updates = test_semaphore
|
2017-10-19 08:56:25 +00:00
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def async_update(self):
|
2017-10-19 08:56:25 +00:00
|
|
|
"""Test update."""
|
|
|
|
updates.append(self._count)
|
2020-04-07 16:33:23 +00:00
|
|
|
await test_lock.acquire()
|
2017-10-19 08:56:25 +00:00
|
|
|
|
|
|
|
ent_1 = AsyncEntity("sensor.test_1", 1)
|
|
|
|
ent_2 = AsyncEntity("sensor.test_2", 2)
|
|
|
|
ent_3 = AsyncEntity("sensor.test_3", 3)
|
|
|
|
ent_4 = AsyncEntity("sensor.test_4", 4)
|
|
|
|
|
2019-03-26 06:53:36 +00:00
|
|
|
await test_lock.acquire()
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
|
|
ent_1.async_schedule_update_ha_state(True)
|
|
|
|
ent_2.async_schedule_update_ha_state(True)
|
|
|
|
ent_3.async_schedule_update_ha_state(True)
|
|
|
|
ent_4.async_schedule_update_ha_state(True)
|
|
|
|
|
|
|
|
while True:
|
|
|
|
if len(updates) >= 2:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
|
|
|
|
assert len(updates) == 2
|
|
|
|
assert updates == [1, 2]
|
|
|
|
|
|
|
|
updates.clear()
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
|
|
|
|
while True:
|
|
|
|
if len(updates) >= 2:
|
|
|
|
break
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
|
|
|
|
assert len(updates) == 2
|
|
|
|
assert updates == [3, 4]
|
|
|
|
|
|
|
|
updates.clear()
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
finally:
|
|
|
|
# we may have more than one lock need to release in case test failed
|
|
|
|
for _ in updates:
|
|
|
|
test_lock.release()
|
|
|
|
await asyncio.sleep(0)
|
|
|
|
test_lock.release()
|
2018-01-23 06:54:41 +00:00
|
|
|
|
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
async def test_async_remove_no_platform(hass):
|
2018-01-23 06:54:41 +00:00
|
|
|
"""Test async_remove method when no platform set."""
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
ent.entity_id = "test.test"
|
2020-04-07 16:33:23 +00:00
|
|
|
await ent.async_update_ha_state()
|
2018-01-23 06:54:41 +00:00
|
|
|
assert len(hass.states.async_entity_ids()) == 1
|
2020-04-07 16:33:23 +00:00
|
|
|
await ent.async_remove()
|
2018-01-23 06:54:41 +00:00
|
|
|
assert len(hass.states.async_entity_ids()) == 0
|
2018-07-24 12:12:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_async_remove_runs_callbacks(hass):
|
|
|
|
"""Test async_remove method when no platform set."""
|
|
|
|
result = []
|
|
|
|
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
ent.entity_id = "test.test"
|
2018-07-24 12:12:53 +00:00
|
|
|
ent.async_on_remove(lambda: result.append(1))
|
|
|
|
await ent.async_remove()
|
|
|
|
assert len(result) == 1
|
2018-08-20 15:39:53 +00:00
|
|
|
|
|
|
|
|
2022-03-08 04:42:16 +00:00
|
|
|
async def test_async_remove_ignores_in_flight_polling(hass):
|
|
|
|
"""Test in flight polling is ignored after removing."""
|
|
|
|
result = []
|
|
|
|
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
|
|
|
ent.entity_id = "test.test"
|
|
|
|
ent.async_on_remove(lambda: result.append(1))
|
|
|
|
ent.async_write_ha_state()
|
|
|
|
assert hass.states.get("test.test").state == STATE_UNKNOWN
|
|
|
|
await ent.async_remove()
|
|
|
|
assert len(result) == 1
|
|
|
|
assert hass.states.get("test.test") is None
|
|
|
|
ent.async_write_ha_state()
|
|
|
|
|
|
|
|
|
2018-08-20 15:39:53 +00:00
|
|
|
async def test_set_context(hass):
|
|
|
|
"""Test setting context."""
|
|
|
|
context = Context()
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
ent.entity_id = "hello.world"
|
2018-08-20 15:39:53 +00:00
|
|
|
ent.async_set_context(context)
|
|
|
|
await ent.async_update_ha_state()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert hass.states.get("hello.world").context == context
|
2018-08-20 15:39:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_set_context_expired(hass):
|
|
|
|
"""Test setting context."""
|
|
|
|
context = Context()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.object(
|
|
|
|
entity.Entity, "context_recent_time", new_callable=PropertyMock
|
|
|
|
) as recent:
|
2018-08-20 15:39:53 +00:00
|
|
|
recent.return_value = timedelta(seconds=-5)
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
ent.entity_id = "hello.world"
|
2018-08-20 15:39:53 +00:00
|
|
|
ent.async_set_context(context)
|
|
|
|
await ent.async_update_ha_state()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert hass.states.get("hello.world").context != context
|
2018-08-20 15:39:53 +00:00
|
|
|
assert ent._context is None
|
|
|
|
assert ent._context_set is None
|
2019-08-22 21:12:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_warn_disabled(hass, caplog):
|
|
|
|
"""Test we warn once if we write to a disabled entity."""
|
2022-06-28 16:38:05 +00:00
|
|
|
entry = er.RegistryEntry(
|
2019-08-22 21:12:24 +00:00
|
|
|
entity_id="hello.world",
|
|
|
|
unique_id="test-unique-id",
|
|
|
|
platform="test-platform",
|
2022-06-28 16:38:05 +00:00
|
|
|
disabled_by=er.RegistryEntryDisabler.USER,
|
2019-08-22 21:12:24 +00:00
|
|
|
)
|
|
|
|
mock_registry(hass, {"hello.world": entry})
|
|
|
|
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
|
|
|
ent.entity_id = "hello.world"
|
|
|
|
ent.registry_entry = entry
|
|
|
|
ent.platform = MagicMock(platform_name="test-platform")
|
|
|
|
|
|
|
|
caplog.clear()
|
|
|
|
ent.async_write_ha_state()
|
|
|
|
assert hass.states.get("hello.world") is None
|
|
|
|
assert "Entity hello.world is incorrectly being triggered" in caplog.text
|
|
|
|
|
|
|
|
caplog.clear()
|
|
|
|
ent.async_write_ha_state()
|
|
|
|
assert hass.states.get("hello.world") is None
|
|
|
|
assert caplog.text == ""
|
2019-08-23 00:32:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_disabled_in_entity_registry(hass):
|
|
|
|
"""Test entity is removed if we disable entity registry entry."""
|
2022-06-28 16:38:05 +00:00
|
|
|
entry = er.RegistryEntry(
|
2019-08-23 00:32:43 +00:00
|
|
|
entity_id="hello.world",
|
|
|
|
unique_id="test-unique-id",
|
|
|
|
platform="test-platform",
|
2020-08-19 12:57:38 +00:00
|
|
|
disabled_by=None,
|
2019-08-23 00:32:43 +00:00
|
|
|
)
|
|
|
|
registry = mock_registry(hass, {"hello.world": entry})
|
|
|
|
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
|
|
|
ent.entity_id = "hello.world"
|
|
|
|
ent.registry_entry = entry
|
2020-08-19 12:57:38 +00:00
|
|
|
assert ent.enabled is True
|
2019-08-23 00:32:43 +00:00
|
|
|
|
2020-08-19 12:57:38 +00:00
|
|
|
ent.add_to_platform_start(hass, MagicMock(platform_name="test-platform"), None)
|
|
|
|
await ent.add_to_platform_finish()
|
|
|
|
assert hass.states.get("hello.world") is not None
|
2019-08-23 00:32:43 +00:00
|
|
|
|
2021-04-23 07:56:42 +00:00
|
|
|
entry2 = registry.async_update_entity(
|
2022-06-28 16:38:05 +00:00
|
|
|
"hello.world", disabled_by=er.RegistryEntryDisabler.USER
|
2021-04-23 07:56:42 +00:00
|
|
|
)
|
2019-08-23 00:32:43 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert entry2 != entry
|
|
|
|
assert ent.registry_entry == entry2
|
2020-08-19 12:57:38 +00:00
|
|
|
assert ent.enabled is False
|
|
|
|
assert hass.states.get("hello.world") is None
|
2019-08-23 00:32:43 +00:00
|
|
|
|
2020-08-19 12:57:38 +00:00
|
|
|
entry3 = registry.async_update_entity("hello.world", disabled_by=None)
|
2019-08-23 00:32:43 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert entry3 != entry2
|
2020-08-19 12:57:38 +00:00
|
|
|
# Entry is no longer updated, entity is no longer tracking changes
|
|
|
|
assert ent.registry_entry == entry2
|
2019-12-02 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_capability_attrs(hass):
|
|
|
|
"""Test we still include capabilities even when unavailable."""
|
|
|
|
with patch.object(
|
|
|
|
entity.Entity, "available", PropertyMock(return_value=False)
|
|
|
|
), patch.object(
|
|
|
|
entity.Entity,
|
|
|
|
"capability_attributes",
|
|
|
|
PropertyMock(return_value={"always": "there"}),
|
|
|
|
):
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
|
|
|
ent.entity_id = "hello.world"
|
|
|
|
ent.async_write_ha_state()
|
|
|
|
|
|
|
|
state = hass.states.get("hello.world")
|
|
|
|
assert state is not None
|
|
|
|
assert state.state == STATE_UNAVAILABLE
|
|
|
|
assert state.attributes["always"] == "there"
|
2020-02-06 10:37:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_warn_slow_write_state(hass, caplog):
|
|
|
|
"""Check that we log a warning if reading properties takes too long."""
|
|
|
|
mock_entity = entity.Entity()
|
|
|
|
mock_entity.hass = hass
|
|
|
|
mock_entity.entity_id = "comp_test.test_entity"
|
|
|
|
mock_entity.platform = MagicMock(platform_name="hue")
|
|
|
|
|
|
|
|
with patch("homeassistant.helpers.entity.timer", side_effect=[0, 10]):
|
|
|
|
mock_entity.async_write_ha_state()
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"Updating state for comp_test.test_entity "
|
|
|
|
"(<class 'homeassistant.helpers.entity.Entity'>) "
|
|
|
|
"took 10.000 seconds. Please create a bug report at "
|
2020-10-02 22:04:11 +00:00
|
|
|
"https://github.com/home-assistant/core/issues?"
|
2020-02-06 10:37:35 +00:00
|
|
|
"q=is%3Aopen+is%3Aissue+label%3A%22integration%3A+hue%22"
|
|
|
|
) in caplog.text
|
2020-02-11 00:32:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_warn_slow_write_state_custom_component(hass, caplog):
|
|
|
|
"""Check that we log a warning if reading properties takes too long."""
|
|
|
|
|
|
|
|
class CustomComponentEntity(entity.Entity):
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Custom component entity."""
|
|
|
|
|
2020-02-11 00:32:47 +00:00
|
|
|
__module__ = "custom_components.bla.sensor"
|
|
|
|
|
|
|
|
mock_entity = CustomComponentEntity()
|
|
|
|
mock_entity.hass = hass
|
|
|
|
mock_entity.entity_id = "comp_test.test_entity"
|
|
|
|
mock_entity.platform = MagicMock(platform_name="hue")
|
|
|
|
|
|
|
|
with patch("homeassistant.helpers.entity.timer", side_effect=[0, 10]):
|
|
|
|
mock_entity.async_write_ha_state()
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"Updating state for comp_test.test_entity "
|
|
|
|
"(<class 'custom_components.bla.sensor.test_warn_slow_write_state_custom_component.<locals>.CustomComponentEntity'>) "
|
|
|
|
"took 10.000 seconds. Please report it to the custom component author."
|
|
|
|
) in caplog.text
|
2020-08-19 12:57:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_setup_source(hass):
|
|
|
|
"""Check that we register sources correctly."""
|
|
|
|
platform = MockEntityPlatform(hass)
|
|
|
|
|
|
|
|
entity_platform = MockEntity(name="Platform Config Source")
|
|
|
|
await platform.async_add_entities([entity_platform])
|
|
|
|
|
|
|
|
platform.config_entry = MockConfigEntry()
|
|
|
|
entity_entry = MockEntity(name="Config Entry Source")
|
|
|
|
await platform.async_add_entities([entity_entry])
|
|
|
|
|
|
|
|
assert entity.entity_sources(hass) == {
|
|
|
|
"test_domain.platform_config_source": {
|
2021-09-30 14:49:16 +00:00
|
|
|
"custom_component": False,
|
2020-08-19 12:57:38 +00:00
|
|
|
"domain": "test_platform",
|
2021-09-30 14:49:16 +00:00
|
|
|
"source": entity.SOURCE_PLATFORM_CONFIG,
|
2020-08-19 12:57:38 +00:00
|
|
|
},
|
|
|
|
"test_domain.config_entry_source": {
|
|
|
|
"config_entry": platform.config_entry.entry_id,
|
2021-09-30 14:49:16 +00:00
|
|
|
"custom_component": False,
|
2020-08-19 12:57:38 +00:00
|
|
|
"domain": "test_platform",
|
2021-09-30 14:49:16 +00:00
|
|
|
"source": entity.SOURCE_CONFIG_ENTRY,
|
2020-08-19 12:57:38 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
await platform.async_reset()
|
|
|
|
|
|
|
|
assert entity.entity_sources(hass) == {}
|
2021-02-08 09:45:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_removing_entity_unavailable(hass):
|
|
|
|
"""Test removing an entity that is still registered creates an unavailable state."""
|
2022-06-28 16:38:05 +00:00
|
|
|
entry = er.RegistryEntry(
|
2021-02-08 09:45:46 +00:00
|
|
|
entity_id="hello.world",
|
|
|
|
unique_id="test-unique-id",
|
|
|
|
platform="test-platform",
|
|
|
|
disabled_by=None,
|
|
|
|
)
|
|
|
|
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
|
|
|
ent.entity_id = "hello.world"
|
|
|
|
ent.registry_entry = entry
|
|
|
|
ent.async_write_ha_state()
|
|
|
|
|
|
|
|
state = hass.states.get("hello.world")
|
|
|
|
assert state is not None
|
|
|
|
assert state.state == STATE_UNKNOWN
|
|
|
|
|
|
|
|
await ent.async_remove()
|
|
|
|
|
|
|
|
state = hass.states.get("hello.world")
|
|
|
|
assert state is not None
|
|
|
|
assert state.state == STATE_UNAVAILABLE
|
2021-04-10 06:19:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_get_supported_features_entity_registry(hass):
|
|
|
|
"""Test get_supported_features falls back to entity registry."""
|
|
|
|
entity_reg = mock_registry(hass)
|
|
|
|
entity_id = entity_reg.async_get_or_create(
|
|
|
|
"hello", "world", "5678", supported_features=456
|
|
|
|
).entity_id
|
|
|
|
assert entity.get_supported_features(hass, entity_id) == 456
|
|
|
|
|
|
|
|
|
|
|
|
async def test_get_supported_features_prioritize_state(hass):
|
|
|
|
"""Test get_supported_features gives priority to state."""
|
|
|
|
entity_reg = mock_registry(hass)
|
|
|
|
entity_id = entity_reg.async_get_or_create(
|
|
|
|
"hello", "world", "5678", supported_features=456
|
|
|
|
).entity_id
|
|
|
|
assert entity.get_supported_features(hass, entity_id) == 456
|
|
|
|
|
|
|
|
hass.states.async_set(entity_id, None, {"supported_features": 123})
|
|
|
|
|
|
|
|
assert entity.get_supported_features(hass, entity_id) == 123
|
|
|
|
|
|
|
|
|
|
|
|
async def test_get_supported_features_raises_on_unknown(hass):
|
|
|
|
"""Test get_supported_features raises on unknown entity_id."""
|
|
|
|
with pytest.raises(HomeAssistantError):
|
|
|
|
entity.get_supported_features(hass, "hello.world")
|
2021-04-27 19:48:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_float_conversion(hass):
|
|
|
|
"""Test conversion of float state to string rounds."""
|
|
|
|
assert 2.4 + 1.2 != 3.6
|
|
|
|
with patch.object(entity.Entity, "state", PropertyMock(return_value=2.4 + 1.2)):
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent.hass = hass
|
|
|
|
ent.entity_id = "hello.world"
|
|
|
|
ent.async_write_ha_state()
|
|
|
|
|
|
|
|
state = hass.states.get("hello.world")
|
|
|
|
assert state is not None
|
|
|
|
assert state.state == "3.6"
|
2021-10-11 21:15:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_attribution_attribute(hass):
|
|
|
|
"""Test attribution attribute."""
|
|
|
|
mock_entity = entity.Entity()
|
|
|
|
mock_entity.hass = hass
|
|
|
|
mock_entity.entity_id = "hello.world"
|
|
|
|
mock_entity._attr_attribution = "Home Assistant"
|
|
|
|
|
|
|
|
mock_entity.async_schedule_update_ha_state(True)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
state = hass.states.get(mock_entity.entity_id)
|
|
|
|
assert state.attributes.get(ATTR_ATTRIBUTION) == "Home Assistant"
|
2021-10-14 08:04:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_entity_category_property(hass):
|
|
|
|
"""Test entity category property."""
|
|
|
|
mock_entity1 = entity.Entity()
|
|
|
|
mock_entity1.hass = hass
|
|
|
|
mock_entity1.entity_description = entity.EntityDescription(
|
|
|
|
key="abc", entity_category="ignore_me"
|
|
|
|
)
|
|
|
|
mock_entity1.entity_id = "hello.world"
|
2021-12-01 12:54:36 +00:00
|
|
|
mock_entity1._attr_entity_category = entity.EntityCategory.CONFIG
|
2021-10-14 08:04:26 +00:00
|
|
|
assert mock_entity1.entity_category == "config"
|
|
|
|
|
|
|
|
mock_entity2 = entity.Entity()
|
|
|
|
mock_entity2.hass = hass
|
|
|
|
mock_entity2.entity_description = entity.EntityDescription(
|
2021-12-01 12:54:36 +00:00
|
|
|
key="abc", entity_category=entity.EntityCategory.CONFIG
|
2021-10-14 08:04:26 +00:00
|
|
|
)
|
|
|
|
mock_entity2.entity_id = "hello.world"
|
|
|
|
assert mock_entity2.entity_category == "config"
|
2022-02-08 22:00:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"value,expected",
|
|
|
|
(
|
|
|
|
("config", entity.EntityCategory.CONFIG),
|
|
|
|
("diagnostic", entity.EntityCategory.DIAGNOSTIC),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
def test_entity_category_schema(value, expected):
|
|
|
|
"""Test entity category schema."""
|
|
|
|
schema = vol.Schema(entity.ENTITY_CATEGORIES_SCHEMA)
|
|
|
|
result = schema(value)
|
|
|
|
assert result == expected
|
|
|
|
assert isinstance(result, entity.EntityCategory)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize("value", (None, "non_existing"))
|
|
|
|
def test_entity_category_schema_error(value):
|
|
|
|
"""Test entity category schema."""
|
|
|
|
schema = vol.Schema(entity.ENTITY_CATEGORIES_SCHEMA)
|
2022-04-01 16:40:43 +00:00
|
|
|
with pytest.raises(
|
|
|
|
vol.Invalid,
|
2022-05-02 16:33:16 +00:00
|
|
|
match=r"expected EntityCategory or one of 'config', 'diagnostic'",
|
2022-04-01 16:40:43 +00:00
|
|
|
):
|
2022-02-08 22:00:53 +00:00
|
|
|
schema(value)
|
2022-04-25 17:10:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_entity_description_fallback():
|
|
|
|
"""Test entity description has same defaults as entity."""
|
|
|
|
ent = entity.Entity()
|
|
|
|
ent_with_description = entity.Entity()
|
|
|
|
ent_with_description.entity_description = entity.EntityDescription(key="test")
|
|
|
|
|
|
|
|
for field in dataclasses.fields(entity.EntityDescription):
|
|
|
|
if field.name == "key":
|
|
|
|
continue
|
|
|
|
|
|
|
|
assert getattr(ent, field.name) == getattr(ent_with_description, field.name)
|
2022-06-28 16:38:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"has_entity_name, entity_name, expected_friendly_name",
|
|
|
|
(
|
|
|
|
(False, "Entity Blu", "Entity Blu"),
|
|
|
|
(False, None, None),
|
|
|
|
(True, "Entity Blu", "Device Bla Entity Blu"),
|
|
|
|
(True, None, "Device Bla"),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
async def test_friendly_name(
|
|
|
|
hass, has_entity_name, entity_name, expected_friendly_name
|
|
|
|
):
|
|
|
|
"""Test entity_id is influenced by entity name."""
|
|
|
|
|
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Mock setup entry method."""
|
|
|
|
async_add_entities(
|
|
|
|
[
|
|
|
|
MockEntity(
|
|
|
|
unique_id="qwer",
|
|
|
|
device_info={
|
|
|
|
"identifiers": {("hue", "1234")},
|
|
|
|
"connections": {(dr.CONNECTION_NETWORK_MAC, "abcd")},
|
|
|
|
"name": "Device Bla",
|
|
|
|
},
|
|
|
|
has_entity_name=has_entity_name,
|
|
|
|
name=entity_name,
|
|
|
|
),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
return True
|
|
|
|
|
|
|
|
platform = MockPlatform(async_setup_entry=async_setup_entry)
|
|
|
|
config_entry = MockConfigEntry(entry_id="super-mock-id")
|
|
|
|
entity_platform = MockEntityPlatform(
|
|
|
|
hass, platform_name=config_entry.domain, platform=platform
|
|
|
|
)
|
|
|
|
|
|
|
|
assert await entity_platform.async_setup_entry(config_entry)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert len(hass.states.async_entity_ids()) == 1
|
|
|
|
state = hass.states.async_all()[0]
|
|
|
|
assert state.attributes.get(ATTR_FRIENDLY_NAME) == expected_friendly_name
|