2022-06-08 01:11:38 +02:00
|
|
|
"""Tests for Fritz!Tools sensor platform."""
|
2024-03-08 14:50:04 +01:00
|
|
|
|
2022-03-30 01:54:16 +02:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
2026-02-14 13:41:26 +01:00
|
|
|
from datetime import UTC, datetime
|
2024-09-03 16:39:06 +02:00
|
|
|
from unittest.mock import patch
|
2022-03-30 01:54:16 +02:00
|
|
|
|
2026-02-14 13:41:26 +01:00
|
|
|
from freezegun.api import FrozenDateTimeFactory
|
2022-03-30 01:54:16 +02:00
|
|
|
from fritzconnection.core.exceptions import FritzConnectionException
|
2024-09-03 16:39:06 +02:00
|
|
|
import pytest
|
2026-04-11 00:04:23 +02:00
|
|
|
from requests.exceptions import RequestException
|
2024-09-03 16:39:06 +02:00
|
|
|
from syrupy.assertion import SnapshotAssertion
|
2022-03-30 01:54:16 +02:00
|
|
|
|
2026-04-24 17:18:17 +02:00
|
|
|
from homeassistant.components.fritz.const import DOMAIN, SCAN_INTERVAL
|
2024-09-03 16:39:06 +02:00
|
|
|
from homeassistant.components.sensor import DOMAIN as SENSOR_DOMAIN
|
|
|
|
|
from homeassistant.const import STATE_UNAVAILABLE, Platform
|
2022-03-30 01:54:16 +02:00
|
|
|
from homeassistant.core import HomeAssistant
|
2024-09-03 16:39:06 +02:00
|
|
|
from homeassistant.helpers import entity_registry as er
|
2022-03-30 01:54:16 +02:00
|
|
|
|
2026-02-16 21:30:52 +01:00
|
|
|
from .const import MOCK_FB_SERVICES, MOCK_USER_DATA
|
2022-03-30 01:54:16 +02:00
|
|
|
|
2024-09-03 16:39:06 +02:00
|
|
|
from tests.common import MockConfigEntry, async_fire_time_changed, snapshot_platform
|
2022-03-30 01:54:16 +02:00
|
|
|
|
|
|
|
|
|
2024-09-03 16:39:06 +02:00
|
|
|
@pytest.mark.freeze_time(datetime(2024, 9, 1, 20, tzinfo=UTC))
|
|
|
|
|
@pytest.mark.usefixtures("entity_registry_enabled_by_default")
|
|
|
|
|
async def test_sensor_setup(
|
|
|
|
|
hass: HomeAssistant,
|
|
|
|
|
entity_registry: er.EntityRegistry,
|
|
|
|
|
fc_class_mock,
|
|
|
|
|
fh_class_mock,
|
2026-01-27 22:57:28 +01:00
|
|
|
fs_class_mock,
|
2024-09-03 16:39:06 +02:00
|
|
|
snapshot: SnapshotAssertion,
|
|
|
|
|
) -> None:
|
2024-03-25 11:33:00 +01:00
|
|
|
"""Test setup of Fritz!Tools sensors."""
|
2022-03-30 01:54:16 +02:00
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain=DOMAIN, data=MOCK_USER_DATA)
|
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
2024-09-03 16:39:06 +02:00
|
|
|
with patch("homeassistant.components.fritz.PLATFORMS", [Platform.SENSOR]):
|
|
|
|
|
assert await hass.config_entries.async_setup(entry.entry_id)
|
|
|
|
|
await hass.async_block_till_done()
|
2022-03-30 01:54:16 +02:00
|
|
|
|
2024-09-03 16:39:06 +02:00
|
|
|
await snapshot_platform(hass, entity_registry, snapshot, entry.entry_id)
|
2022-03-30 01:54:16 +02:00
|
|
|
|
|
|
|
|
|
2023-02-13 09:53:09 +01:00
|
|
|
async def test_sensor_update_fail(
|
2026-02-14 13:41:26 +01:00
|
|
|
hass: HomeAssistant,
|
|
|
|
|
caplog: pytest.LogCaptureFixture,
|
|
|
|
|
freezer: FrozenDateTimeFactory,
|
|
|
|
|
fc_class_mock,
|
|
|
|
|
fh_class_mock,
|
2026-02-17 14:08:02 +01:00
|
|
|
fs_class_mock,
|
2023-02-13 09:53:09 +01:00
|
|
|
) -> None:
|
2024-03-25 11:33:00 +01:00
|
|
|
"""Test failed update of Fritz!Tools sensors."""
|
2022-03-30 01:54:16 +02:00
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain=DOMAIN, data=MOCK_USER_DATA)
|
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
2024-02-15 20:52:40 +01:00
|
|
|
await hass.config_entries.async_setup(entry.entry_id)
|
2022-03-30 01:54:16 +02:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
2024-11-24 18:32:37 +01:00
|
|
|
fc_class_mock().call_action_side_effect(FritzConnectionException("Boom"))
|
2026-02-14 13:41:26 +01:00
|
|
|
|
|
|
|
|
freezer.tick(SCAN_INTERVAL)
|
|
|
|
|
async_fire_time_changed(hass)
|
2024-03-29 18:16:53 -10:00
|
|
|
await hass.async_block_till_done(wait_background_tasks=True)
|
2022-03-30 01:54:16 +02:00
|
|
|
|
2025-07-23 18:53:19 +02:00
|
|
|
assert "Error while updating the data: Boom" in caplog.text
|
2024-11-24 18:32:37 +01:00
|
|
|
|
2022-03-30 01:54:16 +02:00
|
|
|
sensors = hass.states.async_all(SENSOR_DOMAIN)
|
|
|
|
|
for sensor in sensors:
|
|
|
|
|
assert sensor.state == STATE_UNAVAILABLE
|
2026-02-16 21:30:52 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.freeze_time("2026-02-14T09:30:00+00:00")
|
|
|
|
|
async def test_sensor_uptime_spike(
|
|
|
|
|
hass: HomeAssistant,
|
|
|
|
|
caplog: pytest.LogCaptureFixture,
|
|
|
|
|
freezer: FrozenDateTimeFactory,
|
|
|
|
|
fc_class_mock,
|
|
|
|
|
fh_class_mock,
|
|
|
|
|
fs_class_mock,
|
|
|
|
|
) -> None:
|
|
|
|
|
"""Test handling of uptime spikes in Fritz!Tools sensors."""
|
|
|
|
|
|
2026-04-29 13:34:25 +02:00
|
|
|
entity_id = "sensor.mock_title_uptime"
|
2026-02-16 21:30:52 +01:00
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain=DOMAIN, data=MOCK_USER_DATA)
|
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
|
|
await hass.config_entries.async_setup(entry.entry_id)
|
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
|
|
assert (state := hass.states.get(entity_id))
|
|
|
|
|
assert state.state == "2026-01-16T06:00:21+00:00"
|
|
|
|
|
|
2026-04-24 17:18:17 +02:00
|
|
|
# Simulate uptime spike by setting uptime to a value that shifts
|
|
|
|
|
# the resulting timestamp only by 1 second.
|
2026-02-16 21:30:52 +01:00
|
|
|
base_uptime = MOCK_FB_SERVICES["DeviceInfo1"]["GetInfo"]["NewUpTime"]
|
|
|
|
|
update_uptime = {
|
|
|
|
|
"DeviceInfo1": {
|
|
|
|
|
"GetInfo": {
|
2026-04-24 17:18:17 +02:00
|
|
|
"NewUpTime": base_uptime + SCAN_INTERVAL + 1,
|
2026-02-16 21:30:52 +01:00
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
fc_class_mock().override_services({**MOCK_FB_SERVICES, **update_uptime})
|
|
|
|
|
|
|
|
|
|
freezer.tick(SCAN_INTERVAL)
|
|
|
|
|
async_fire_time_changed(hass)
|
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
|
|
assert (new_state := hass.states.get(entity_id))
|
|
|
|
|
assert new_state.state == "2026-01-16T06:00:21+00:00"
|
2026-04-11 00:04:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.freeze_time(datetime(2024, 9, 1, 20, tzinfo=UTC))
|
|
|
|
|
@pytest.mark.usefixtures("entity_registry_enabled_by_default")
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
|
("side_effect", "return_values"),
|
2026-05-04 15:38:39 +02:00
|
|
|
[
|
|
|
|
|
(RequestException("boom"), None),
|
|
|
|
|
(None, [0, 0, 0]),
|
|
|
|
|
(None, []),
|
|
|
|
|
(FritzConnectionException("boom"), None),
|
|
|
|
|
],
|
2026-04-11 00:04:23 +02:00
|
|
|
)
|
|
|
|
|
async def test_sensor_cpu_temp_not_supported(
|
|
|
|
|
hass: HomeAssistant,
|
|
|
|
|
entity_registry: er.EntityRegistry,
|
|
|
|
|
side_effect,
|
|
|
|
|
return_values,
|
|
|
|
|
fc_class_mock,
|
|
|
|
|
fh_class_mock,
|
|
|
|
|
fs_class_mock,
|
|
|
|
|
snapshot: SnapshotAssertion,
|
|
|
|
|
) -> None:
|
|
|
|
|
"""Test setup of Fritz!Tools sensors."""
|
|
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain=DOMAIN, data=MOCK_USER_DATA)
|
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
|
|
with (
|
|
|
|
|
patch("homeassistant.components.fritz.PLATFORMS", [Platform.SENSOR]),
|
|
|
|
|
patch(
|
|
|
|
|
"homeassistant.components.fritz.coordinator.FritzStatus", fs_class_mock
|
|
|
|
|
) as mock_status,
|
|
|
|
|
):
|
|
|
|
|
mock_status.get_cpu_temperatures.side_effect = side_effect
|
|
|
|
|
mock_status.get_cpu_temperatures.return_value = return_values
|
|
|
|
|
assert await hass.config_entries.async_setup(entry.entry_id)
|
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
|
|
await snapshot_platform(hass, entity_registry, snapshot, entry.entry_id)
|
|
|
|
|
assert not entity_registry.async_is_registered(
|
|
|
|
|
"sensor.mock_title_cpu_temperature"
|
|
|
|
|
)
|