mirror of
https://github.com/home-assistant/core.git
synced 2026-02-28 13:56:28 +00:00
168 lines
5.3 KiB
Python
168 lines
5.3 KiB
Python
"""Tests for the Indevolt number platform."""
|
|
|
|
from datetime import timedelta
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
from freezegun.api import FrozenDateTimeFactory
|
|
import pytest
|
|
from syrupy.assertion import SnapshotAssertion
|
|
|
|
from homeassistant.components.indevolt.coordinator import SCAN_INTERVAL
|
|
from homeassistant.components.number import SERVICE_SET_VALUE
|
|
from homeassistant.const import STATE_UNAVAILABLE, Platform
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
from homeassistant.helpers import entity_registry as er
|
|
|
|
from . import setup_integration
|
|
|
|
from tests.common import MockConfigEntry, async_fire_time_changed, snapshot_platform
|
|
|
|
KEY_READ_DISCHARGE_LIMIT = "6105"
|
|
KEY_WRITE_DISCHARGE_LIMIT = "1142"
|
|
|
|
KEY_READ_MAX_AC_OUTPUT_POWER = "11011"
|
|
KEY_WRITE_MAX_AC_OUTPUT_POWER = "1147"
|
|
|
|
KEY_READ_INVERTER_INPUT_LIMIT = "11009"
|
|
KEY_WRITE_INVERTER_INPUT_LIMIT = "1138"
|
|
|
|
KEY_READ_FEEDIN_POWER_LIMIT = "11010"
|
|
KEY_WRITE_FEEDIN_POWER_LIMIT = "1146"
|
|
|
|
|
|
@pytest.mark.usefixtures("entity_registry_enabled_by_default")
|
|
@pytest.mark.parametrize("generation", [2], indirect=True)
|
|
async def test_number(
|
|
hass: HomeAssistant,
|
|
entity_registry: er.EntityRegistry,
|
|
mock_indevolt: AsyncMock,
|
|
snapshot: SnapshotAssertion,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test number entity registration and values."""
|
|
with patch("homeassistant.components.indevolt.PLATFORMS", [Platform.NUMBER]):
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
await snapshot_platform(hass, entity_registry, snapshot, mock_config_entry.entry_id)
|
|
|
|
|
|
@pytest.mark.parametrize("generation", [2], indirect=True)
|
|
@pytest.mark.parametrize(
|
|
("entity_id", "read_key", "write_key", "test_value"),
|
|
[
|
|
(
|
|
"number.cms_sf2000_discharge_limit",
|
|
KEY_READ_DISCHARGE_LIMIT,
|
|
KEY_WRITE_DISCHARGE_LIMIT,
|
|
50,
|
|
),
|
|
(
|
|
"number.cms_sf2000_max_ac_output_power",
|
|
KEY_READ_MAX_AC_OUTPUT_POWER,
|
|
KEY_WRITE_MAX_AC_OUTPUT_POWER,
|
|
1500,
|
|
),
|
|
(
|
|
"number.cms_sf2000_inverter_input_limit",
|
|
KEY_READ_INVERTER_INPUT_LIMIT,
|
|
KEY_WRITE_INVERTER_INPUT_LIMIT,
|
|
800,
|
|
),
|
|
(
|
|
"number.cms_sf2000_feed_in_power_limit",
|
|
KEY_READ_FEEDIN_POWER_LIMIT,
|
|
KEY_WRITE_FEEDIN_POWER_LIMIT,
|
|
1200,
|
|
),
|
|
],
|
|
)
|
|
async def test_number_set_values(
|
|
hass: HomeAssistant,
|
|
mock_indevolt: AsyncMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
entity_id: str,
|
|
read_key: str,
|
|
write_key: str,
|
|
test_value: int,
|
|
) -> None:
|
|
"""Test setting number values for all configurable parameters."""
|
|
with patch("homeassistant.components.indevolt.PLATFORMS", [Platform.NUMBER]):
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
# Reset mock call count for this iteration
|
|
mock_indevolt.set_data.reset_mock()
|
|
|
|
# Update mock data to reflect the new value
|
|
mock_indevolt.fetch_data.return_value[read_key] = test_value
|
|
|
|
# Call the service to set the value
|
|
await hass.services.async_call(
|
|
Platform.NUMBER,
|
|
SERVICE_SET_VALUE,
|
|
{"entity_id": entity_id, "value": test_value},
|
|
blocking=True,
|
|
)
|
|
|
|
# Verify set_data was called with correct parameters
|
|
mock_indevolt.set_data.assert_called_with(write_key, test_value)
|
|
|
|
# Verify updated state
|
|
assert (state := hass.states.get(entity_id)) is not None
|
|
assert int(float(state.state)) == test_value
|
|
|
|
|
|
@pytest.mark.parametrize("generation", [2], indirect=True)
|
|
async def test_number_set_value_error(
|
|
hass: HomeAssistant,
|
|
mock_indevolt: AsyncMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test error handling when setting number values."""
|
|
with patch("homeassistant.components.indevolt.PLATFORMS", [Platform.NUMBER]):
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
# Mock set_data to raise an error
|
|
mock_indevolt.set_data.side_effect = HomeAssistantError(
|
|
"Device communication failed"
|
|
)
|
|
|
|
# Attempt to set value
|
|
with pytest.raises(HomeAssistantError):
|
|
await hass.services.async_call(
|
|
Platform.NUMBER,
|
|
SERVICE_SET_VALUE,
|
|
{
|
|
"entity_id": "number.cms_sf2000_discharge_limit",
|
|
"value": 50,
|
|
},
|
|
blocking=True,
|
|
)
|
|
|
|
# Verify set_data was called before failing
|
|
mock_indevolt.set_data.assert_called_once()
|
|
|
|
|
|
@pytest.mark.parametrize("generation", [2], indirect=True)
|
|
async def test_number_availability(
|
|
hass: HomeAssistant,
|
|
mock_indevolt: AsyncMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
freezer: FrozenDateTimeFactory,
|
|
) -> None:
|
|
"""Test number entity availability / non-availability."""
|
|
with patch("homeassistant.components.indevolt.PLATFORMS", [Platform.NUMBER]):
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
assert (state := hass.states.get("number.cms_sf2000_discharge_limit"))
|
|
assert int(float(state.state)) == 5
|
|
|
|
# Simulate fetch_data error
|
|
mock_indevolt.fetch_data.side_effect = ConnectionError
|
|
freezer.tick(delta=timedelta(seconds=SCAN_INTERVAL))
|
|
async_fire_time_changed(hass)
|
|
await hass.async_block_till_done()
|
|
|
|
assert (state := hass.states.get("number.cms_sf2000_discharge_limit"))
|
|
assert state.state == STATE_UNAVAILABLE
|