Files
core/tests/components/tuya/__init__.py

103 lines
3.5 KiB
Python

"""Tests for the Tuya component."""
from __future__ import annotations
import pathlib
from typing import Any
from unittest.mock import patch
from freezegun.api import FrozenDateTimeFactory
from tuya_sharing import CustomerDevice, Manager
from homeassistant.components.tuya import DeviceListener
from homeassistant.core import HomeAssistant
from tests.common import MockConfigEntry
FIXTURES_DIR = pathlib.Path(__file__).parent / "fixtures"
DEVICE_MOCKS = sorted(
str(path.relative_to(FIXTURES_DIR).with_suffix(""))
for path in FIXTURES_DIR.glob("*.json")
)
class MockDeviceListener(DeviceListener):
"""Mocked DeviceListener for testing."""
async def async_send_device_update(
self,
hass: HomeAssistant,
device: CustomerDevice,
updated_status_properties: dict[str, Any] | None = None,
dp_timestamps: dict[str, int] | None = None,
) -> None:
"""Mock update device method."""
property_list: list[str] = []
if updated_status_properties:
for key, value in updated_status_properties.items():
if key not in device.status:
raise ValueError(
f"Property {key} not found in device status: {device.status}"
)
device.status[key] = value
property_list.append(key)
self.update_device(device, property_list, dp_timestamps)
await hass.async_block_till_done()
async def initialize_entry(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_devices: CustomerDevice | list[CustomerDevice],
) -> None:
"""Initialize the Tuya component with a mock manager and config entry."""
if not isinstance(mock_devices, list):
mock_devices = [mock_devices]
mock_manager.device_map = {device.id: device for device in mock_devices}
# Setup
mock_config_entry.add_to_hass(hass)
# Initialize the component
with patch("homeassistant.components.tuya.Manager", return_value=mock_manager):
await hass.config_entries.async_setup(mock_config_entry.entry_id)
await hass.async_block_till_done()
async def check_selective_state_update(
hass: HomeAssistant,
mock_device: CustomerDevice,
mock_listener: MockDeviceListener,
freezer: FrozenDateTimeFactory,
*,
entity_id: str,
dpcode: str,
initial_state: str,
updates: dict[str, Any],
expected_state: str,
last_reported: str,
) -> None:
"""Test selective state update.
This test verifies that when an update event comes with properties that do NOT
include the dpcode (e.g., a battery event for a door sensor),
the entity state is not changed and last_reported is not updated.
"""
initial_reported = "2024-01-01T00:00:00+00:00"
assert hass.states.get(entity_id).state == initial_state
assert hass.states.get(entity_id).last_reported.isoformat() == initial_reported
# Force update the dpcode and trigger device update
freezer.tick(30)
mock_device.status[dpcode] = None
await mock_listener.async_send_device_update(hass, mock_device, {})
assert hass.states.get(entity_id).state == initial_state
assert hass.states.get(entity_id).last_reported.isoformat() == initial_reported
# Trigger device update with provided updates
freezer.tick(30)
await mock_listener.async_send_device_update(hass, mock_device, updates)
assert hass.states.get(entity_id).state == expected_state
assert hass.states.get(entity_id).last_reported.isoformat() == last_reported