1
0
mirror of https://github.com/home-assistant/core.git synced 2026-05-17 05:51:33 +01:00
Files
core/tests/components/tuya/test_services.py
T
Fredrik Mårtensson 99dc368c79 Add feeder meal plan actions to tuya (#161488)
Co-authored-by: Norbert Rittel <norbert@rittel.de>
Co-authored-by: epenet <6771947+epenet@users.noreply.github.com>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
2026-04-14 23:02:31 +02:00

251 lines
7.8 KiB
Python

"""Tests for Tuya services."""
from __future__ import annotations
import pytest
from syrupy.assertion import SnapshotAssertion
from tuya_device_handlers.device_wrapper.service_feeder_schedule import FeederSchedule
from tuya_sharing import CustomerDevice, Manager
from homeassistant.components.tuya.const import DOMAIN
from homeassistant.components.tuya.services import Service
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError, ServiceValidationError
from homeassistant.helpers import device_registry as dr
from . import initialize_entry
from tests.common import MockConfigEntry
DECODED_MEAL_PLAN: list[FeederSchedule] = [
{
"days": [
"monday",
"tuesday",
"wednesday",
"thursday",
"friday",
"saturday",
"sunday",
],
"time": "09:00",
"portion": 1,
"enabled": True,
},
{
"days": [
"monday",
"tuesday",
"wednesday",
"thursday",
"friday",
"saturday",
"sunday",
],
"time": "09:30",
"portion": 1,
"enabled": True,
},
]
@pytest.mark.parametrize("mock_device_code", ["cwwsq_wfkzyy0evslzsmoi"])
async def test_get_feeder_meal_plan(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_device: CustomerDevice,
snapshot: SnapshotAssertion,
device_registry: dr.DeviceRegistry,
) -> None:
"""Test GET_FEEDER_MEAL_PLAN with valid meal plan data."""
await initialize_entry(hass, mock_manager, mock_config_entry, mock_device)
device_entry = device_registry.async_get_device(
identifiers={(DOMAIN, mock_device.id)}
)
assert device_entry is not None
device_id = device_entry.id
result = await hass.services.async_call(
DOMAIN,
Service.GET_FEEDER_MEAL_PLAN,
{"device_id": device_id},
blocking=True,
return_response=True,
)
assert result == snapshot
@pytest.mark.parametrize("mock_device_code", ["cwwsq_wfkzyy0evslzsmoi"])
async def test_get_feeder_meal_plan_invalid_meal_plan(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_device: CustomerDevice,
device_registry: dr.DeviceRegistry,
) -> None:
"""Test GET_FEEDER_MEAL_PLAN error when meal plan data is missing."""
await initialize_entry(hass, mock_manager, mock_config_entry, mock_device)
device_entry = device_registry.async_get_device(
identifiers={(DOMAIN, mock_device.id)}
)
assert device_entry is not None
device_id = device_entry.id
mock_device.status.pop("meal_plan", None)
with pytest.raises(
HomeAssistantError,
match="Unable to parse meal plan data",
):
await hass.services.async_call(
DOMAIN,
Service.GET_FEEDER_MEAL_PLAN,
{"device_id": device_id},
blocking=True,
return_response=True,
)
@pytest.mark.parametrize("mock_device_code", ["cwwsq_wfkzyy0evslzsmoi"])
async def test_set_feeder_meal_plan(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_device: CustomerDevice,
device_registry: dr.DeviceRegistry,
) -> None:
"""Test SET_FEEDER_MEAL_PLAN with valid device and meal plan data."""
await initialize_entry(hass, mock_manager, mock_config_entry, mock_device)
device_entry = device_registry.async_get_device(
identifiers={(DOMAIN, mock_device.id)}
)
assert device_entry is not None
device_id = device_entry.id
await hass.services.async_call(
DOMAIN,
Service.SET_FEEDER_MEAL_PLAN,
{
"device_id": device_id,
"meal_plan": DECODED_MEAL_PLAN,
},
blocking=True,
)
mock_manager.send_commands.assert_called_once_with(
mock_device.id,
[{"code": "meal_plan", "value": "fwkAAQF/CR4BAQ=="}],
)
@pytest.mark.parametrize("mock_device_code", ["cwwsq_wfkzyy0evslzsmoi"])
async def test_set_feeder_meal_plan_unsupported_device(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_device: CustomerDevice,
device_registry: dr.DeviceRegistry,
) -> None:
"""Test SET_FEEDER_MEAL_PLAN error when device is unsupported."""
await initialize_entry(hass, mock_manager, mock_config_entry, mock_device)
device_entry = device_registry.async_get_device(
identifiers={(DOMAIN, mock_device.id)}
)
assert device_entry is not None
device_id = device_entry.id
mock_device.product_id = "unsupported_product"
with pytest.raises(
ServiceValidationError,
match=f"Feeder with ID {mock_device.id} does not support meal plan functionality",
):
await hass.services.async_call(
DOMAIN,
Service.SET_FEEDER_MEAL_PLAN,
{
"device_id": device_id,
"meal_plan": DECODED_MEAL_PLAN,
},
blocking=True,
)
@pytest.mark.parametrize("mock_device_code", ["cwwsq_wfkzyy0evslzsmoi"])
async def test_get_tuya_device_error_device_not_found(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_device: CustomerDevice,
) -> None:
"""Test service error when device ID does not exist."""
await initialize_entry(hass, mock_manager, mock_config_entry, mock_device)
with pytest.raises(
ServiceValidationError,
match="Feeder with ID invalid_device_id could not be found",
):
await hass.services.async_call(
DOMAIN,
Service.GET_FEEDER_MEAL_PLAN,
{"device_id": "invalid_device_id"},
blocking=True,
return_response=True,
)
@pytest.mark.parametrize("mock_device_code", ["cwwsq_wfkzyy0evslzsmoi"])
async def test_get_tuya_device_error_non_tuya_device(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_device: CustomerDevice,
) -> None:
"""Test service error when target device is not a Tuya device."""
await initialize_entry(hass, mock_manager, mock_config_entry, mock_device)
device_registry = dr.async_get(hass)
non_tuya_device = device_registry.async_get_or_create(
config_entry_id=mock_config_entry.entry_id,
identifiers={("other_domain", "some_id")},
name="Non-Tuya Device",
)
with pytest.raises(
ServiceValidationError,
match=f"Device with ID {non_tuya_device.id} is not a Tuya feeder",
):
await hass.services.async_call(
DOMAIN,
Service.GET_FEEDER_MEAL_PLAN,
{"device_id": non_tuya_device.id},
blocking=True,
return_response=True,
)
@pytest.mark.parametrize("mock_device_code", ["cwwsq_wfkzyy0evslzsmoi"])
async def test_get_tuya_device_error_unknown_tuya_device(
hass: HomeAssistant,
mock_manager: Manager,
mock_config_entry: MockConfigEntry,
mock_device: CustomerDevice,
) -> None:
"""Test service error when Tuya identifier is not present in manager map."""
await initialize_entry(hass, mock_manager, mock_config_entry, mock_device)
device_registry = dr.async_get(hass)
tuya_device = device_registry.async_get_or_create(
config_entry_id=mock_config_entry.entry_id,
identifiers={(DOMAIN, "unknown_tuya_id")},
name="Unknown Tuya Device",
)
with pytest.raises(
ServiceValidationError,
match=f"Feeder with ID {tuya_device.id} could not be found",
):
await hass.services.async_call(
DOMAIN,
Service.GET_FEEDER_MEAL_PLAN,
{"device_id": tuya_device.id},
blocking=True,
return_response=True,
)