mirror of
https://github.com/home-assistant/core.git
synced 2025-07-21 12:17:07 +00:00
Add additional tests for Matter update entity (#122575)
* Add additional tests for Matter update entity Extend test coverage for Matter update entity. This includes tests for error handling and state store/restore. * Improve test descriptions * Add restore test only (using mock_restore_cache_with_extra_data) * Fix test_update_state_save_and_restore test * Use homeassistant constants * Use update component constants * Use freezer to skip time for device update check We check device updates every 12h currently. Use the freezer to skip time. Still add a test which uses the service call to make sure this works too.
This commit is contained in:
parent
98a007cb2f
commit
be255613de
@ -5,12 +5,26 @@ from unittest.mock import AsyncMock, MagicMock
|
|||||||
|
|
||||||
from chip.clusters import Objects as clusters
|
from chip.clusters import Objects as clusters
|
||||||
from chip.clusters.ClusterObjects import ClusterAttributeDescriptor
|
from chip.clusters.ClusterObjects import ClusterAttributeDescriptor
|
||||||
|
from freezegun.api import FrozenDateTimeFactory
|
||||||
from matter_server.client.models.node import MatterNode
|
from matter_server.client.models.node import MatterNode
|
||||||
|
from matter_server.common.errors import UpdateCheckError, UpdateError
|
||||||
from matter_server.common.models import MatterSoftwareVersion, UpdateSource
|
from matter_server.common.models import MatterSoftwareVersion, UpdateSource
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
|
from homeassistant.components.homeassistant import (
|
||||||
|
DOMAIN as HA_DOMAIN,
|
||||||
|
SERVICE_UPDATE_ENTITY,
|
||||||
|
)
|
||||||
|
from homeassistant.components.matter.update import SCAN_INTERVAL
|
||||||
|
from homeassistant.components.update import (
|
||||||
|
ATTR_VERSION,
|
||||||
|
DOMAIN as UPDATE_DOMAIN,
|
||||||
|
SERVICE_INSTALL,
|
||||||
|
)
|
||||||
from homeassistant.const import ATTR_ENTITY_ID, STATE_OFF, STATE_ON
|
from homeassistant.const import ATTR_ENTITY_ID, STATE_OFF, STATE_ON
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant, State
|
||||||
|
from homeassistant.exceptions import HomeAssistantError
|
||||||
|
from homeassistant.helpers.restore_state import STORAGE_KEY as RESTORE_STATE_KEY
|
||||||
from homeassistant.setup import async_setup_component
|
from homeassistant.setup import async_setup_component
|
||||||
|
|
||||||
from .common import (
|
from .common import (
|
||||||
@ -19,6 +33,24 @@ from .common import (
|
|||||||
trigger_subscription_callback,
|
trigger_subscription_callback,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from tests.common import (
|
||||||
|
async_fire_time_changed,
|
||||||
|
async_mock_restore_state_shutdown_restart,
|
||||||
|
mock_restore_cache_with_extra_data,
|
||||||
|
)
|
||||||
|
|
||||||
|
TEST_SOFTWARE_VERSION = MatterSoftwareVersion(
|
||||||
|
vid=65521,
|
||||||
|
pid=32768,
|
||||||
|
software_version=2,
|
||||||
|
software_version_string="v2.0",
|
||||||
|
firmware_information="",
|
||||||
|
min_applicable_software_version=0,
|
||||||
|
max_applicable_software_version=1,
|
||||||
|
release_notes_url="http://home-assistant.io/non-existing-product",
|
||||||
|
update_source=UpdateSource.LOCAL,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def set_node_attribute_typed(
|
def set_node_attribute_typed(
|
||||||
node: MatterNode,
|
node: MatterNode,
|
||||||
@ -34,11 +66,18 @@ def set_node_attribute_typed(
|
|||||||
|
|
||||||
@pytest.fixture(name="check_node_update")
|
@pytest.fixture(name="check_node_update")
|
||||||
async def check_node_update_fixture(matter_client: MagicMock) -> AsyncMock:
|
async def check_node_update_fixture(matter_client: MagicMock) -> AsyncMock:
|
||||||
"""Fixture for a flow sensor node."""
|
"""Fixture to check for node updates."""
|
||||||
matter_client.check_node_update = AsyncMock(return_value=None)
|
matter_client.check_node_update = AsyncMock(return_value=None)
|
||||||
return matter_client.check_node_update
|
return matter_client.check_node_update
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(name="update_node")
|
||||||
|
async def update_node_fixture(matter_client: MagicMock) -> AsyncMock:
|
||||||
|
"""Fixture to install update."""
|
||||||
|
matter_client.update_node = AsyncMock(return_value=None)
|
||||||
|
return matter_client.update_node
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(name="updateable_node")
|
@pytest.fixture(name="updateable_node")
|
||||||
async def updateable_node_fixture(
|
async def updateable_node_fixture(
|
||||||
hass: HomeAssistant, matter_client: MagicMock
|
hass: HomeAssistant, matter_client: MagicMock
|
||||||
@ -63,19 +102,19 @@ async def test_update_entity(
|
|||||||
assert matter_client.check_node_update.call_count == 1
|
assert matter_client.check_node_update.call_count == 1
|
||||||
|
|
||||||
|
|
||||||
async def test_update_install(
|
async def test_update_check_service(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
matter_client: MagicMock,
|
matter_client: MagicMock,
|
||||||
check_node_update: AsyncMock,
|
check_node_update: AsyncMock,
|
||||||
updateable_node: MatterNode,
|
updateable_node: MatterNode,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test update entity exists and update check got made."""
|
"""Test check device update through service call."""
|
||||||
state = hass.states.get("update.mock_dimmable_light")
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
assert state
|
assert state
|
||||||
assert state.state == STATE_OFF
|
assert state.state == STATE_OFF
|
||||||
assert state.attributes.get("installed_version") == "v1.0"
|
assert state.attributes.get("installed_version") == "v1.0"
|
||||||
|
|
||||||
await async_setup_component(hass, "homeassistant", {})
|
await async_setup_component(hass, HA_DOMAIN, {})
|
||||||
|
|
||||||
check_node_update.return_value = MatterSoftwareVersion(
|
check_node_update.return_value = MatterSoftwareVersion(
|
||||||
vid=65521,
|
vid=65521,
|
||||||
@ -90,8 +129,8 @@ async def test_update_install(
|
|||||||
)
|
)
|
||||||
|
|
||||||
await hass.services.async_call(
|
await hass.services.async_call(
|
||||||
"homeassistant",
|
HA_DOMAIN,
|
||||||
"update_entity",
|
SERVICE_UPDATE_ENTITY,
|
||||||
{
|
{
|
||||||
ATTR_ENTITY_ID: "update.mock_dimmable_light",
|
ATTR_ENTITY_ID: "update.mock_dimmable_light",
|
||||||
},
|
},
|
||||||
@ -109,11 +148,50 @@ async def test_update_install(
|
|||||||
== "http://home-assistant.io/non-existing-product"
|
== "http://home-assistant.io/non-existing-product"
|
||||||
)
|
)
|
||||||
|
|
||||||
await async_setup_component(hass, "update", {})
|
|
||||||
|
async def test_update_install(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
matter_client: MagicMock,
|
||||||
|
check_node_update: AsyncMock,
|
||||||
|
updateable_node: MatterNode,
|
||||||
|
freezer: FrozenDateTimeFactory,
|
||||||
|
) -> None:
|
||||||
|
"""Test device update with Matter attribute changes influence progress."""
|
||||||
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
|
assert state
|
||||||
|
assert state.state == STATE_OFF
|
||||||
|
assert state.attributes.get("installed_version") == "v1.0"
|
||||||
|
|
||||||
|
check_node_update.return_value = MatterSoftwareVersion(
|
||||||
|
vid=65521,
|
||||||
|
pid=32768,
|
||||||
|
software_version=2,
|
||||||
|
software_version_string="v2.0",
|
||||||
|
firmware_information="",
|
||||||
|
min_applicable_software_version=0,
|
||||||
|
max_applicable_software_version=1,
|
||||||
|
release_notes_url="http://home-assistant.io/non-existing-product",
|
||||||
|
update_source=UpdateSource.LOCAL,
|
||||||
|
)
|
||||||
|
|
||||||
|
freezer.tick(SCAN_INTERVAL)
|
||||||
|
async_fire_time_changed(hass)
|
||||||
|
await hass.async_block_till_done(wait_background_tasks=True)
|
||||||
|
|
||||||
|
assert matter_client.check_node_update.call_count == 2
|
||||||
|
|
||||||
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
|
assert state
|
||||||
|
assert state.state == STATE_ON
|
||||||
|
assert state.attributes.get("latest_version") == "v2.0"
|
||||||
|
assert (
|
||||||
|
state.attributes.get("release_url")
|
||||||
|
== "http://home-assistant.io/non-existing-product"
|
||||||
|
)
|
||||||
|
|
||||||
await hass.services.async_call(
|
await hass.services.async_call(
|
||||||
"update",
|
UPDATE_DOMAIN,
|
||||||
"install",
|
SERVICE_INSTALL,
|
||||||
{
|
{
|
||||||
ATTR_ENTITY_ID: "update.mock_dimmable_light",
|
ATTR_ENTITY_ID: "update.mock_dimmable_light",
|
||||||
},
|
},
|
||||||
@ -169,3 +247,173 @@ async def test_update_install(
|
|||||||
state = hass.states.get("update.mock_dimmable_light")
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
assert state.state == STATE_OFF
|
assert state.state == STATE_OFF
|
||||||
assert state.attributes.get("installed_version") == "v2.0"
|
assert state.attributes.get("installed_version") == "v2.0"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_update_install_failure(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
matter_client: MagicMock,
|
||||||
|
check_node_update: AsyncMock,
|
||||||
|
update_node: AsyncMock,
|
||||||
|
updateable_node: MatterNode,
|
||||||
|
freezer: FrozenDateTimeFactory,
|
||||||
|
) -> None:
|
||||||
|
"""Test update entity service call errors."""
|
||||||
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
|
assert state
|
||||||
|
assert state.state == STATE_OFF
|
||||||
|
assert state.attributes.get("installed_version") == "v1.0"
|
||||||
|
|
||||||
|
check_node_update.return_value = MatterSoftwareVersion(
|
||||||
|
vid=65521,
|
||||||
|
pid=32768,
|
||||||
|
software_version=2,
|
||||||
|
software_version_string="v2.0",
|
||||||
|
firmware_information="",
|
||||||
|
min_applicable_software_version=0,
|
||||||
|
max_applicable_software_version=1,
|
||||||
|
release_notes_url="http://home-assistant.io/non-existing-product",
|
||||||
|
update_source=UpdateSource.LOCAL,
|
||||||
|
)
|
||||||
|
|
||||||
|
freezer.tick(SCAN_INTERVAL)
|
||||||
|
async_fire_time_changed(hass)
|
||||||
|
await hass.async_block_till_done(wait_background_tasks=True)
|
||||||
|
|
||||||
|
assert matter_client.check_node_update.call_count == 2
|
||||||
|
|
||||||
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
|
assert state
|
||||||
|
assert state.state == STATE_ON
|
||||||
|
assert state.attributes.get("latest_version") == "v2.0"
|
||||||
|
assert (
|
||||||
|
state.attributes.get("release_url")
|
||||||
|
== "http://home-assistant.io/non-existing-product"
|
||||||
|
)
|
||||||
|
|
||||||
|
update_node.side_effect = UpdateCheckError("Error finding applicable update")
|
||||||
|
|
||||||
|
with pytest.raises(HomeAssistantError):
|
||||||
|
await hass.services.async_call(
|
||||||
|
UPDATE_DOMAIN,
|
||||||
|
SERVICE_INSTALL,
|
||||||
|
{
|
||||||
|
ATTR_ENTITY_ID: "update.mock_dimmable_light",
|
||||||
|
ATTR_VERSION: "v3.0",
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
update_node.side_effect = UpdateError("Error updating node")
|
||||||
|
|
||||||
|
with pytest.raises(HomeAssistantError):
|
||||||
|
await hass.services.async_call(
|
||||||
|
UPDATE_DOMAIN,
|
||||||
|
SERVICE_INSTALL,
|
||||||
|
{
|
||||||
|
ATTR_ENTITY_ID: "update.mock_dimmable_light",
|
||||||
|
ATTR_VERSION: "v3.0",
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def test_update_state_save_and_restore(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
hass_storage: dict[str, Any],
|
||||||
|
matter_client: MagicMock,
|
||||||
|
check_node_update: AsyncMock,
|
||||||
|
updateable_node: MatterNode,
|
||||||
|
freezer: FrozenDateTimeFactory,
|
||||||
|
) -> None:
|
||||||
|
"""Test latest update information is retained across reload/restart."""
|
||||||
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
|
assert state
|
||||||
|
assert state.state == STATE_OFF
|
||||||
|
assert state.attributes.get("installed_version") == "v1.0"
|
||||||
|
|
||||||
|
check_node_update.return_value = TEST_SOFTWARE_VERSION
|
||||||
|
|
||||||
|
freezer.tick(SCAN_INTERVAL)
|
||||||
|
async_fire_time_changed(hass)
|
||||||
|
await hass.async_block_till_done(wait_background_tasks=True)
|
||||||
|
|
||||||
|
assert matter_client.check_node_update.call_count == 2
|
||||||
|
|
||||||
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
|
assert state
|
||||||
|
assert state.state == STATE_ON
|
||||||
|
assert state.attributes.get("latest_version") == "v2.0"
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
await async_mock_restore_state_shutdown_restart(hass)
|
||||||
|
|
||||||
|
assert len(hass_storage[RESTORE_STATE_KEY]["data"]) == 1
|
||||||
|
state = hass_storage[RESTORE_STATE_KEY]["data"][0]["state"]
|
||||||
|
assert state["entity_id"] == "update.mock_dimmable_light"
|
||||||
|
extra_data = hass_storage[RESTORE_STATE_KEY]["data"][0]["extra_data"]
|
||||||
|
|
||||||
|
# Check that the extra data has the format we expect.
|
||||||
|
assert extra_data == {
|
||||||
|
"software_update": {
|
||||||
|
"vid": 65521,
|
||||||
|
"pid": 32768,
|
||||||
|
"software_version": 2,
|
||||||
|
"software_version_string": "v2.0",
|
||||||
|
"firmware_information": "",
|
||||||
|
"min_applicable_software_version": 0,
|
||||||
|
"max_applicable_software_version": 1,
|
||||||
|
"release_notes_url": "http://home-assistant.io/non-existing-product",
|
||||||
|
"update_source": "local",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async def test_update_state_restore(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
matter_client: MagicMock,
|
||||||
|
check_node_update: AsyncMock,
|
||||||
|
update_node: AsyncMock,
|
||||||
|
) -> None:
|
||||||
|
"""Test latest update information extra data is restored."""
|
||||||
|
mock_restore_cache_with_extra_data(
|
||||||
|
hass,
|
||||||
|
(
|
||||||
|
(
|
||||||
|
State(
|
||||||
|
"update.mock_dimmable_light",
|
||||||
|
STATE_ON,
|
||||||
|
{
|
||||||
|
"auto_update": False,
|
||||||
|
"installed_version": "v1.0",
|
||||||
|
"in_progress": False,
|
||||||
|
"latest_version": "v2.0",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
{"software_update": TEST_SOFTWARE_VERSION.as_dict()},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
await setup_integration_with_node_fixture(hass, "dimmable-light", matter_client)
|
||||||
|
|
||||||
|
assert check_node_update.call_count == 0
|
||||||
|
|
||||||
|
state = hass.states.get("update.mock_dimmable_light")
|
||||||
|
assert state
|
||||||
|
assert state.state == STATE_ON
|
||||||
|
assert state.attributes.get("latest_version") == "v2.0"
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
UPDATE_DOMAIN,
|
||||||
|
SERVICE_INSTALL,
|
||||||
|
{
|
||||||
|
ATTR_ENTITY_ID: "update.mock_dimmable_light",
|
||||||
|
},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Validate that the integer software version from the extra data is passed
|
||||||
|
# to the update_node call.
|
||||||
|
assert update_node.call_count == 1
|
||||||
|
assert (
|
||||||
|
update_node.call_args[1]["software_version"]
|
||||||
|
== TEST_SOFTWARE_VERSION.software_version
|
||||||
|
)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user