315 lines
10 KiB
Python
315 lines
10 KiB
Python
"""Define tests for the OpenUV config flow."""
|
|
|
|
from ipaddress import ip_address
|
|
from unittest.mock import patch
|
|
|
|
import pytest
|
|
from regenmaschine.errors import RainMachineError
|
|
|
|
from homeassistant import config_entries, setup
|
|
from homeassistant.components import zeroconf
|
|
from homeassistant.components.rainmachine import (
|
|
CONF_ALLOW_INACTIVE_ZONES_TO_RUN,
|
|
CONF_DEFAULT_ZONE_RUN_TIME,
|
|
CONF_USE_APP_RUN_TIMES,
|
|
DOMAIN,
|
|
)
|
|
from homeassistant.const import CONF_IP_ADDRESS, CONF_PASSWORD, CONF_PORT, CONF_SSL
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.data_entry_flow import FlowResultType
|
|
from homeassistant.helpers import entity_registry as er
|
|
|
|
|
|
async def test_duplicate_error(hass: HomeAssistant, config, config_entry) -> None:
|
|
"""Test that errors are shown when duplicates are added."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}, data=config
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
async def test_invalid_password(hass: HomeAssistant, config) -> None:
|
|
"""Test that an invalid password throws an error."""
|
|
with patch("regenmaschine.client.Client.load_local", side_effect=RainMachineError):
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}, data=config
|
|
)
|
|
assert result["errors"] == {CONF_PASSWORD: "invalid_auth"}
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("platform", "entity_name", "entity_id", "old_unique_id", "new_unique_id"),
|
|
[
|
|
(
|
|
"binary_sensor",
|
|
"Home Flow Sensor",
|
|
"binary_sensor.home_flow_sensor",
|
|
"60e32719b6cf_flow_sensor",
|
|
"60:e3:27:19:b6:cf_flow_sensor",
|
|
),
|
|
(
|
|
"switch",
|
|
"Home Landscaping",
|
|
"switch.home_landscaping",
|
|
"60e32719b6cf_RainMachineZone_1",
|
|
"60:e3:27:19:b6:cf_zone_1",
|
|
),
|
|
],
|
|
)
|
|
async def test_migrate_1_2(
|
|
hass: HomeAssistant,
|
|
entity_registry: er.EntityRegistry,
|
|
client,
|
|
config,
|
|
config_entry,
|
|
entity_id,
|
|
entity_name,
|
|
old_unique_id,
|
|
new_unique_id,
|
|
platform,
|
|
) -> None:
|
|
"""Test migration from version 1 to 2 (consistent unique IDs)."""
|
|
# Create entity RegistryEntry using old unique ID format:
|
|
entity_entry = entity_registry.async_get_or_create(
|
|
platform,
|
|
DOMAIN,
|
|
old_unique_id,
|
|
suggested_object_id=entity_name,
|
|
config_entry=config_entry,
|
|
original_name=entity_name,
|
|
)
|
|
assert entity_entry.entity_id == entity_id
|
|
assert entity_entry.unique_id == old_unique_id
|
|
|
|
with (
|
|
patch(
|
|
"homeassistant.components.rainmachine.async_setup_entry", return_value=True
|
|
),
|
|
patch(
|
|
"homeassistant.components.rainmachine.config_flow.Client",
|
|
return_value=client,
|
|
),
|
|
):
|
|
await setup.async_setup_component(hass, DOMAIN, {})
|
|
await hass.async_block_till_done()
|
|
|
|
# Check that new RegistryEntry is using new unique ID format
|
|
entity_entry = entity_registry.async_get(entity_id)
|
|
assert entity_entry.unique_id == new_unique_id
|
|
assert entity_registry.async_get_entity_id(platform, DOMAIN, old_unique_id) is None
|
|
|
|
|
|
async def test_options_flow(hass: HomeAssistant, config, config_entry) -> None:
|
|
"""Test config flow options."""
|
|
with patch(
|
|
"homeassistant.components.rainmachine.async_setup_entry", return_value=True
|
|
):
|
|
await hass.config_entries.async_setup(config_entry.entry_id)
|
|
result = await hass.config_entries.options.async_init(config_entry.entry_id)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "init"
|
|
|
|
result = await hass.config_entries.options.async_configure(
|
|
result["flow_id"],
|
|
user_input={
|
|
CONF_DEFAULT_ZONE_RUN_TIME: 600,
|
|
CONF_USE_APP_RUN_TIMES: False,
|
|
CONF_ALLOW_INACTIVE_ZONES_TO_RUN: False,
|
|
},
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert config_entry.options == {
|
|
CONF_DEFAULT_ZONE_RUN_TIME: 600,
|
|
CONF_USE_APP_RUN_TIMES: False,
|
|
CONF_ALLOW_INACTIVE_ZONES_TO_RUN: False,
|
|
}
|
|
|
|
|
|
async def test_show_form(hass: HomeAssistant) -> None:
|
|
"""Test that the form is served with no input."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data=None,
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
|
|
async def test_step_user(hass: HomeAssistant, config, setup_rainmachine) -> None:
|
|
"""Test that the user step works."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data=config,
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == "12345"
|
|
assert result["data"] == {
|
|
CONF_IP_ADDRESS: "192.168.1.100",
|
|
CONF_PASSWORD: "password",
|
|
CONF_PORT: 8080,
|
|
CONF_SSL: True,
|
|
CONF_DEFAULT_ZONE_RUN_TIME: 600,
|
|
}
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"source", [config_entries.SOURCE_ZEROCONF, config_entries.SOURCE_HOMEKIT]
|
|
)
|
|
async def test_step_homekit_zeroconf_ip_already_exists(
|
|
hass: HomeAssistant, client, config, config_entry, source
|
|
) -> None:
|
|
"""Test homekit and zeroconf with an ip that already exists."""
|
|
with patch(
|
|
"homeassistant.components.rainmachine.config_flow.Client", return_value=client
|
|
):
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": source},
|
|
data=zeroconf.ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.1.100"),
|
|
ip_addresses=[ip_address("192.168.1.100")],
|
|
hostname="mock_hostname",
|
|
name="mock_name",
|
|
port=None,
|
|
properties={},
|
|
type="mock_type",
|
|
),
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"source", [config_entries.SOURCE_ZEROCONF, config_entries.SOURCE_HOMEKIT]
|
|
)
|
|
async def test_step_homekit_zeroconf_ip_change(
|
|
hass: HomeAssistant, client, config_entry, source
|
|
) -> None:
|
|
"""Test zeroconf with an ip change."""
|
|
with patch(
|
|
"homeassistant.components.rainmachine.config_flow.Client", return_value=client
|
|
):
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": source},
|
|
data=zeroconf.ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.1.2"),
|
|
ip_addresses=[ip_address("192.168.1.2")],
|
|
hostname="mock_hostname",
|
|
name="mock_name",
|
|
port=None,
|
|
properties={},
|
|
type="mock_type",
|
|
),
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
assert config_entry.data[CONF_IP_ADDRESS] == "192.168.1.2"
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"source", [config_entries.SOURCE_ZEROCONF, config_entries.SOURCE_HOMEKIT]
|
|
)
|
|
async def test_step_homekit_zeroconf_new_controller_when_some_exist(
|
|
hass: HomeAssistant, client, config, source
|
|
) -> None:
|
|
"""Test homekit and zeroconf for a new controller when one already exists."""
|
|
with patch(
|
|
"homeassistant.components.rainmachine.config_flow.Client", return_value=client
|
|
):
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": source},
|
|
data=zeroconf.ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.1.100"),
|
|
ip_addresses=[ip_address("192.168.1.100")],
|
|
hostname="mock_hostname",
|
|
name="mock_name",
|
|
port=None,
|
|
properties={},
|
|
type="mock_type",
|
|
),
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
with (
|
|
patch(
|
|
"homeassistant.components.rainmachine.async_setup_entry", return_value=True
|
|
),
|
|
patch(
|
|
"homeassistant.components.rainmachine.config_flow.Client",
|
|
return_value=client,
|
|
),
|
|
):
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{
|
|
CONF_IP_ADDRESS: "192.168.1.100",
|
|
CONF_PASSWORD: "password",
|
|
CONF_PORT: 8080,
|
|
},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result2["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result2["title"] == "12345"
|
|
assert result2["data"] == {
|
|
CONF_IP_ADDRESS: "192.168.1.100",
|
|
CONF_PASSWORD: "password",
|
|
CONF_PORT: 8080,
|
|
CONF_SSL: True,
|
|
CONF_DEFAULT_ZONE_RUN_TIME: 600,
|
|
}
|
|
|
|
|
|
async def test_discovery_by_homekit_and_zeroconf_same_time(
|
|
hass: HomeAssistant, client
|
|
) -> None:
|
|
"""Test the same controller gets discovered by two different methods."""
|
|
with patch(
|
|
"homeassistant.components.rainmachine.config_flow.Client", return_value=client
|
|
):
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
|
data=zeroconf.ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.1.100"),
|
|
ip_addresses=[ip_address("192.168.1.100")],
|
|
hostname="mock_hostname",
|
|
name="mock_name",
|
|
port=None,
|
|
properties={},
|
|
type="mock_type",
|
|
),
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
with patch(
|
|
"homeassistant.components.rainmachine.config_flow.Client", return_value=client
|
|
):
|
|
result2 = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": config_entries.SOURCE_HOMEKIT},
|
|
data=zeroconf.ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.1.100"),
|
|
ip_addresses=[ip_address("192.168.1.100")],
|
|
hostname="mock_hostname",
|
|
name="mock_name",
|
|
port=None,
|
|
properties={},
|
|
type="mock_type",
|
|
),
|
|
)
|
|
|
|
assert result2["type"] is FlowResultType.ABORT
|
|
assert result2["reason"] == "already_in_progress"
|