mirror of
https://github.com/home-assistant/core.git
synced 2025-04-25 09:47:52 +00:00
2332 lines
77 KiB
Python
2332 lines
77 KiB
Python
"""Test config flow."""
|
|
|
|
from collections.abc import Awaitable, Callable
|
|
from ipaddress import ip_address
|
|
import json
|
|
from typing import Any
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
from aioesphomeapi import (
|
|
APIClient,
|
|
APIConnectionError,
|
|
DeviceInfo,
|
|
EntityInfo,
|
|
EntityState,
|
|
InvalidAuthAPIError,
|
|
InvalidEncryptionKeyAPIError,
|
|
RequiresEncryptionAPIError,
|
|
ResolveAPIError,
|
|
UserService,
|
|
)
|
|
import aiohttp
|
|
import pytest
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.components.esphome import dashboard
|
|
from homeassistant.components.esphome.const import (
|
|
CONF_ALLOW_SERVICE_CALLS,
|
|
CONF_DEVICE_NAME,
|
|
CONF_NOISE_PSK,
|
|
CONF_SUBSCRIBE_LOGS,
|
|
DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
|
|
DOMAIN,
|
|
)
|
|
from homeassistant.config_entries import ConfigFlowResult
|
|
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_PORT
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.data_entry_flow import FlowResultType
|
|
from homeassistant.helpers.service_info.dhcp import DhcpServiceInfo
|
|
from homeassistant.helpers.service_info.hassio import HassioServiceInfo
|
|
from homeassistant.helpers.service_info.mqtt import MqttServiceInfo
|
|
from homeassistant.helpers.service_info.zeroconf import ZeroconfServiceInfo
|
|
|
|
from . import VALID_NOISE_PSK
|
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
INVALID_NOISE_PSK = "lSYBYEjQI1bVL8s2Vask4YytGMj1f1epNtmoim2yuTM="
|
|
WRONG_NOISE_PSK = "GP+ciK+nVfTQ/gcz6uOdS+oKEdJgesU+jeu8Ssj2how="
|
|
|
|
|
|
@pytest.fixture(autouse=False)
|
|
def mock_setup_entry():
|
|
"""Mock setting up a config entry."""
|
|
with patch("homeassistant.components.esphome.async_setup_entry", return_value=True):
|
|
yield
|
|
|
|
|
|
def get_flow_context(hass: HomeAssistant, result: ConfigFlowResult) -> dict[str, Any]:
|
|
"""Get the flow context from the result of async_init or async_configure."""
|
|
flow = next(
|
|
flow
|
|
for flow in hass.config_entries.flow.async_progress()
|
|
if flow["flow_id"] == result["flow_id"]
|
|
)
|
|
|
|
return flow["context"]
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_connection_works(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test we can finish a config flow."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data=None,
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 80},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 80,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert result["options"] == {
|
|
CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS
|
|
}
|
|
assert result["title"] == "test"
|
|
assert result["result"].unique_id == "11:22:33:44:55:aa"
|
|
|
|
assert len(mock_client.connect.mock_calls) == 1
|
|
assert len(mock_client.device_info.mock_calls) == 1
|
|
assert len(mock_client.disconnect.mock_calls) == 1
|
|
assert mock_client.host == "127.0.0.1"
|
|
assert mock_client.port == 80
|
|
assert mock_client.password == ""
|
|
assert mock_client.noise_psk is None
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_connection_updates_host(hass: HomeAssistant) -> None:
|
|
"""Test setup up the same name updates the host."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "test.local", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data=None,
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 80},
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_updates"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "unknown",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
assert entry.data[CONF_HOST] == "127.0.0.1"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_sets_unique_id(hass: HomeAssistant) -> None:
|
|
"""Test that the user flow sets the unique id."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={
|
|
"mac": "1122334455aa",
|
|
},
|
|
type="mock_type",
|
|
)
|
|
discovery_result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
|
|
assert discovery_result["type"] is FlowResultType.FORM
|
|
assert discovery_result["step_id"] == "discovery_confirm"
|
|
assert discovery_result["description_placeholders"] == {
|
|
"name": "test8266",
|
|
}
|
|
|
|
discovery_result = await hass.config_entries.flow.async_configure(
|
|
discovery_result["flow_id"],
|
|
{},
|
|
)
|
|
assert discovery_result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert discovery_result["data"] == {
|
|
CONF_HOST: "192.168.43.183",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data=None,
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_updates"
|
|
assert result["description_placeholders"] == {
|
|
"title": "test",
|
|
"name": "test",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_resolve_error(hass: HomeAssistant, mock_client: APIClient) -> None:
|
|
"""Test user step with IP resolve error."""
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.config_flow.APIConnectionError",
|
|
new_callable=lambda: ResolveAPIError,
|
|
) as exc:
|
|
mock_client.device_info.side_effect = exc
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
assert result["errors"] == {"base": "resolve_error"}
|
|
|
|
assert len(mock_client.connect.mock_calls) == 1
|
|
assert len(mock_client.device_info.mock_calls) == 1
|
|
assert len(mock_client.disconnect.mock_calls) == 1
|
|
|
|
# Now simulate the user retrying with the same host and a successful connection
|
|
mock_client.device_info.side_effect = None
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result2["type"] == FlowResultType.CREATE_ENTRY
|
|
assert result2["title"] == "test"
|
|
assert result2["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_DEVICE_NAME: "test",
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_causes_zeroconf_to_abort(hass: HomeAssistant) -> None:
|
|
"""Test that the user flow sets the unique id and aborts the zeroconf flow."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={
|
|
"mac": "1122334455aa",
|
|
},
|
|
type="mock_type",
|
|
)
|
|
discovery_result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
|
|
assert discovery_result["type"] is FlowResultType.FORM
|
|
assert discovery_result["step_id"] == "discovery_confirm"
|
|
assert discovery_result["description_placeholders"] == {
|
|
"name": "test8266",
|
|
}
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data=None,
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
|
|
assert not hass.config_entries.flow.async_progress_by_handler(DOMAIN)
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_connection_error(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test user step with connection error."""
|
|
mock_client.device_info.side_effect = APIConnectionError
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
assert result["errors"] == {"base": "connection_error"}
|
|
|
|
assert len(mock_client.connect.mock_calls) == 1
|
|
assert len(mock_client.device_info.mock_calls) == 1
|
|
assert len(mock_client.disconnect.mock_calls) == 1
|
|
|
|
# Now simulate the user retrying with the same host and a successful connection
|
|
mock_client.device_info.side_effect = None
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result2["type"] == FlowResultType.CREATE_ENTRY
|
|
assert result2["title"] == "test"
|
|
assert result2["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_DEVICE_NAME: "test",
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_with_password(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test user step with password."""
|
|
mock_client.device_info.return_value = DeviceInfo(uses_password=True, name="test")
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "authenticate"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_PASSWORD: "password1"}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "password1",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.password == "password1"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf")
|
|
async def test_user_invalid_password(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test user step with invalid password."""
|
|
mock_client.device_info.return_value = DeviceInfo(uses_password=True, name="test")
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "authenticate"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
mock_client.connect.side_effect = InvalidAuthAPIError
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_PASSWORD: "invalid"}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "authenticate"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
assert result["errors"] == {"base": "invalid_auth"}
|
|
|
|
mock_client.connect.side_effect = None
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_PASSWORD: "good"}
|
|
)
|
|
|
|
assert result2["type"] == FlowResultType.CREATE_ENTRY
|
|
assert result2["title"] == "test"
|
|
assert result2["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_DEVICE_NAME: "test",
|
|
CONF_PASSWORD: "good",
|
|
CONF_NOISE_PSK: "",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_dashboard", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_dashboard_has_wrong_key(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test user step with key from dashboard that is incorrect."""
|
|
mock_client.device_info.side_effect = [
|
|
RequiresEncryptionAPIError,
|
|
InvalidEncryptionKeyAPIError("Wrong key", "test"),
|
|
DeviceInfo(
|
|
uses_password=False,
|
|
name="test",
|
|
mac_address="11:22:33:44:55:AA",
|
|
),
|
|
]
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
return_value=WRONG_NOISE_PSK,
|
|
):
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_discovers_name_and_gets_key_from_dashboard(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
) -> None:
|
|
"""Test user step can discover the name and get the key from the dashboard."""
|
|
mock_client.device_info.side_effect = [
|
|
RequiresEncryptionAPIError,
|
|
InvalidEncryptionKeyAPIError("Wrong key", "test"),
|
|
DeviceInfo(
|
|
uses_password=False,
|
|
name="test",
|
|
mac_address="11:22:33:44:55:AA",
|
|
),
|
|
]
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test",
|
|
"configuration": "test.yaml",
|
|
}
|
|
)
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
return_value=VALID_NOISE_PSK,
|
|
):
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"dashboard_exception",
|
|
[aiohttp.ClientError(), json.JSONDecodeError("test", "test", 0)],
|
|
)
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_discovers_name_and_gets_key_from_dashboard_fails(
|
|
hass: HomeAssistant,
|
|
dashboard_exception: Exception,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
) -> None:
|
|
"""Test user step can discover the name and get the key from the dashboard."""
|
|
mock_client.device_info.side_effect = [
|
|
RequiresEncryptionAPIError,
|
|
InvalidEncryptionKeyAPIError("Wrong key", "test"),
|
|
DeviceInfo(
|
|
uses_password=False,
|
|
name="test",
|
|
mac_address="11:22:33:44:55:AA",
|
|
),
|
|
]
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test",
|
|
"configuration": "test.yaml",
|
|
}
|
|
)
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
side_effect=dashboard_exception,
|
|
):
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_discovers_name_and_dashboard_is_unavailable(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
) -> None:
|
|
"""Test user step can discover the name but the dashboard is unavailable."""
|
|
mock_client.device_info.side_effect = [
|
|
RequiresEncryptionAPIError,
|
|
InvalidEncryptionKeyAPIError("Wrong key", "test"),
|
|
DeviceInfo(
|
|
uses_password=False,
|
|
name="test",
|
|
mac_address="11:22:33:44:55:AA",
|
|
),
|
|
]
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test",
|
|
"configuration": "test.yaml",
|
|
}
|
|
)
|
|
|
|
with patch(
|
|
"esphome_dashboard_api.ESPHomeDashboardAPI.get_devices",
|
|
side_effect=TimeoutError,
|
|
):
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_login_connection_error(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test user step with connection error on login attempt."""
|
|
mock_client.device_info.return_value = DeviceInfo(uses_password=True, name="test")
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "authenticate"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
mock_client.connect.side_effect = APIConnectionError
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_PASSWORD: "valid"}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "authenticate"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
assert result["errors"] == {"base": "connection_error"}
|
|
|
|
mock_client.connect.side_effect = None
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_PASSWORD: "good"}
|
|
)
|
|
|
|
assert result2["type"] == FlowResultType.CREATE_ENTRY
|
|
assert result2["title"] == "test"
|
|
assert result2["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_DEVICE_NAME: "test",
|
|
CONF_PASSWORD: "good",
|
|
CONF_NOISE_PSK: "",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_initiation(hass: HomeAssistant) -> None:
|
|
"""Test discovery importing works."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={
|
|
"mac": "1122334455aa",
|
|
"friendly_name": "The Test",
|
|
},
|
|
type="mock_type",
|
|
)
|
|
flow = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
assert get_flow_context(hass, flow) == {
|
|
"source": config_entries.SOURCE_ZEROCONF,
|
|
"title_placeholders": {"name": "The Test (test)"},
|
|
"unique_id": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
flow["flow_id"], user_input={}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == "test"
|
|
assert result["data"][CONF_HOST] == "192.168.43.183"
|
|
assert result["data"][CONF_PORT] == 6053
|
|
|
|
assert result["result"]
|
|
assert result["result"].unique_id == "11:22:33:44:55:aa"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_no_mac(hass: HomeAssistant) -> None:
|
|
"""Test discovery aborted if old ESPHome without mac in zeroconf."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={},
|
|
type="mock_type",
|
|
)
|
|
flow = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
assert flow["type"] is FlowResultType.ABORT
|
|
assert flow["reason"] == "mdns_missing_mac"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_already_configured(hass: HomeAssistant) -> None:
|
|
"""Test discovery aborts if already configured via hostname."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "test8266.local", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={"mac": "1122334455aa"},
|
|
type="mock_type",
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_updates"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "unknown",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_duplicate_data(hass: HomeAssistant) -> None:
|
|
"""Test discovery aborts if same mDNS packet arrives."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={"address": "test.local", "mac": "1122334455aa"},
|
|
type="mock_type",
|
|
)
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", data=service_info, context={"source": config_entries.SOURCE_ZEROCONF}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "discovery_confirm"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", data=service_info, context={"source": config_entries.SOURCE_ZEROCONF}
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_in_progress"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_updates_unique_id(hass: HomeAssistant) -> None:
|
|
"""Test a duplicate discovery host aborts and updates existing entry."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={"address": "test8266.local", "mac": "1122334455aa"},
|
|
type="mock_type",
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_updates"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "unknown",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
assert entry.unique_id == "11:22:33:44:55:aa"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_requires_psk(hass: HomeAssistant, mock_client: APIClient) -> None:
|
|
"""Test user step with requiring encryption key."""
|
|
mock_client.device_info.side_effect = RequiresEncryptionAPIError
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["errors"] == {}
|
|
assert result["description_placeholders"] == {"name": "ESPHome"}
|
|
|
|
assert len(mock_client.connect.mock_calls) == 2
|
|
assert len(mock_client.device_info.mock_calls) == 2
|
|
assert len(mock_client.disconnect.mock_calls) == 2
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["errors"] == {"base": "requires_encryption_key"}
|
|
assert result["description_placeholders"] == {"name": "ESPHome"}
|
|
|
|
mock_client.device_info.side_effect = None
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_encryption_key_valid_psk(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test encryption key step with valid key."""
|
|
|
|
mock_client.device_info.side_effect = RequiresEncryptionAPIError
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "ESPHome"}
|
|
|
|
mock_client.device_info = AsyncMock(
|
|
return_value=DeviceInfo(uses_password=False, name="test")
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_encryption_key_invalid_psk(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test encryption key step with invalid key."""
|
|
|
|
mock_client.device_info.side_effect = RequiresEncryptionAPIError
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "ESPHome"}
|
|
|
|
mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["errors"] == {"base": "invalid_psk"}
|
|
assert result["description_placeholders"] == {"name": "ESPHome"}
|
|
assert mock_client.noise_psk == INVALID_NOISE_PSK
|
|
|
|
mock_client.device_info.side_effect = None
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_confirm_valid(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reauth initiation with valid PSK."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reauth_flow(hass)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reauth_confirm"
|
|
assert result["description_placeholders"] == {
|
|
"name": "Mock Title (test)",
|
|
}
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reauth_successful"
|
|
assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reauth_attempt_to_change_mac_aborts(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reauth initiation with valid PSK attempting to change mac.
|
|
|
|
This can happen if reauth starts, but they don't finish it before
|
|
a new device takes the place of the old one at the same IP.
|
|
"""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reauth_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reauth_unique_id_changed"
|
|
assert CONF_NOISE_PSK not in entry.data
|
|
assert result["description_placeholders"] == {
|
|
"expected_mac": "11:22:33:44:55:aa",
|
|
"host": "127.0.0.1",
|
|
"name": "test",
|
|
"unexpected_device_name": "test",
|
|
"unexpected_mac": "11:22:33:44:55:bb",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_fixed_via_dashboard(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
) -> None:
|
|
"""Test reauth fixed automatically via dashboard."""
|
|
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test",
|
|
"configuration": "test.yaml",
|
|
}
|
|
)
|
|
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
return_value=VALID_NOISE_PSK,
|
|
) as mock_get_encryption_key:
|
|
result = await entry.start_reauth_flow(hass)
|
|
|
|
assert result["type"] is FlowResultType.ABORT, result
|
|
assert result["reason"] == "reauth_successful"
|
|
assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
assert len(mock_get_encryption_key.mock_calls) == 1
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_fixed_via_dashboard_add_encryption_remove_password(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test reauth fixed automatically via dashboard with password removed."""
|
|
mock_client.device_info.side_effect = (
|
|
InvalidAuthAPIError,
|
|
DeviceInfo(uses_password=False, name="test", mac_address="11:22:33:44:55:aa"),
|
|
)
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test",
|
|
"configuration": "test.yaml",
|
|
}
|
|
)
|
|
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
return_value=VALID_NOISE_PSK,
|
|
) as mock_get_encryption_key:
|
|
result = await mock_config_entry.start_reauth_flow(hass)
|
|
|
|
assert result["type"] is FlowResultType.ABORT, result
|
|
assert result["reason"] == "reauth_successful"
|
|
assert mock_config_entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
assert mock_config_entry.data[CONF_PASSWORD] == ""
|
|
|
|
assert len(mock_get_encryption_key.mock_calls) == 1
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_dashboard", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_fixed_via_remove_password(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test reauth fixed automatically by seeing password removed."""
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
|
|
result = await mock_config_entry.start_reauth_flow(hass)
|
|
|
|
assert result["type"] is FlowResultType.ABORT, result
|
|
assert result["reason"] == "reauth_successful"
|
|
assert mock_config_entry.data[CONF_PASSWORD] == ""
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_fixed_via_dashboard_at_confirm(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
) -> None:
|
|
"""Test reauth fixed automatically via dashboard at confirm step."""
|
|
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
|
|
result = await entry.start_reauth_flow(hass)
|
|
|
|
assert result["type"] is FlowResultType.FORM, result
|
|
assert result["step_id"] == "reauth_confirm"
|
|
assert result["description_placeholders"] == {
|
|
"name": "Mock Title (test)",
|
|
}
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test",
|
|
"configuration": "test.yaml",
|
|
}
|
|
)
|
|
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
return_value=VALID_NOISE_PSK,
|
|
) as mock_get_encryption_key:
|
|
# We just fetch the form
|
|
result = await hass.config_entries.flow.async_configure(result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.ABORT, result
|
|
assert result["reason"] == "reauth_successful"
|
|
assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
assert len(mock_get_encryption_key.mock_calls) == 1
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_confirm_invalid(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reauth initiation with invalid PSK."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reauth_flow(hass)
|
|
|
|
mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reauth_confirm"
|
|
assert result["description_placeholders"] == {
|
|
"name": "Mock Title (test)",
|
|
}
|
|
assert result["errors"]
|
|
assert result["errors"]["base"] == "invalid_psk"
|
|
|
|
mock_client.device_info = AsyncMock(
|
|
return_value=DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reauth_successful"
|
|
assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_confirm_invalid_with_unique_id(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reauth initiation with invalid PSK."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reauth_flow(hass)
|
|
|
|
mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: INVALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reauth_confirm"
|
|
assert result["description_placeholders"] == {
|
|
"name": "Mock Title (test)",
|
|
}
|
|
assert result["errors"]
|
|
assert result["errors"]["base"] == "invalid_psk"
|
|
|
|
mock_client.device_info = AsyncMock(
|
|
return_value=DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reauth_successful"
|
|
assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_reauth_encryption_key_removed(hass: HomeAssistant) -> None:
|
|
"""Test reauth when the encryption key was removed."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reauth_flow(hass)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reauth_encryption_removed_confirm"
|
|
assert result["description_placeholders"] == {
|
|
"name": "Mock Title (test)",
|
|
}
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reauth_successful"
|
|
assert entry.data[CONF_NOISE_PSK] == ""
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_dhcp_updates_host(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test dhcp discovery updates host and aborts."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
mock_client.device_info = AsyncMock(
|
|
return_value=DeviceInfo(name="test8266", mac_address="1122334455aa")
|
|
)
|
|
|
|
service_info = DhcpServiceInfo(
|
|
ip="192.168.43.184",
|
|
hostname="test8266",
|
|
macaddress="1122334455aa",
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_DHCP}, data=service_info
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_updates"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "unknown",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
assert entry.data[CONF_HOST] == "192.168.43.184"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry")
|
|
async def test_discovery_dhcp_does_not_update_host_wrong_mac(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test dhcp discovery does not update the host if the mac is wrong."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
mock_client.device_info = AsyncMock(
|
|
return_value=DeviceInfo(name="test8266", mac_address="1122334455ff")
|
|
)
|
|
|
|
service_info = DhcpServiceInfo(
|
|
ip="192.168.43.184",
|
|
hostname="test8266",
|
|
macaddress="1122334455aa",
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_DHCP}, data=service_info
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_detailed"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "unknown",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
# Mac was wrong, should not update
|
|
assert entry.data[CONF_HOST] == "192.168.43.183"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry")
|
|
async def test_discovery_dhcp_does_not_update_host_wrong_mac_bad_key(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test dhcp discovery does not update the host if the mac is wrong."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError(
|
|
"Wrong key", "test8266", "1122334455cc"
|
|
)
|
|
service_info = DhcpServiceInfo(
|
|
ip="192.168.43.184",
|
|
hostname="test8266",
|
|
macaddress="1122334455aa",
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_DHCP}, data=service_info
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_detailed"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "unknown",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
# Mac was wrong, should not update
|
|
assert entry.data[CONF_HOST] == "192.168.43.183"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry")
|
|
async def test_discovery_dhcp_does_not_update_host_missing_mac_bad_key(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test dhcp discovery does not update the host if the mac is missing."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
mock_client.device_info.side_effect = InvalidEncryptionKeyAPIError(
|
|
"Wrong key", "test8266", None
|
|
)
|
|
service_info = DhcpServiceInfo(
|
|
ip="192.168.43.184",
|
|
hostname="test8266",
|
|
macaddress="1122334455aa",
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_DHCP}, data=service_info
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured_detailed"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "unknown",
|
|
"mac": "11:22:33:44:55:aa",
|
|
}
|
|
|
|
# Mac was missing, should not update
|
|
assert entry.data[CONF_HOST] == "192.168.43.183"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry")
|
|
async def test_discovery_dhcp_no_changes(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test dhcp discovery updates host and aborts."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_HOST: "192.168.43.183", CONF_PORT: 6053, CONF_PASSWORD: ""},
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_client.device_info = AsyncMock(return_value=DeviceInfo(name="test8266"))
|
|
|
|
service_info = DhcpServiceInfo(
|
|
ip="192.168.43.183",
|
|
hostname="test8266",
|
|
macaddress="000000000000",
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_DHCP}, data=service_info
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
assert entry.data[CONF_HOST] == "192.168.43.183"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_dashboard")
|
|
async def test_discovery_hassio(hass: HomeAssistant) -> None:
|
|
"""Test dashboard discovery."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
data=HassioServiceInfo(
|
|
config={
|
|
"host": "mock-esphome",
|
|
"port": 6052,
|
|
},
|
|
name="ESPHome",
|
|
slug="mock-slug",
|
|
uuid="1234",
|
|
),
|
|
context={"source": config_entries.SOURCE_HASSIO},
|
|
)
|
|
assert result
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "service_received"
|
|
|
|
dash = dashboard.async_get_dashboard(hass)
|
|
assert dash is not None
|
|
assert dash.addon_slug == "mock-slug"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_zeroconf_encryption_key_via_dashboard(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
) -> None:
|
|
"""Test encryption key retrieved from dashboard."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={
|
|
"mac": "1122334455aa",
|
|
},
|
|
type="mock_type",
|
|
)
|
|
flow = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
|
|
assert flow["type"] is FlowResultType.FORM
|
|
assert flow["step_id"] == "discovery_confirm"
|
|
assert flow["description_placeholders"] == {"name": "test8266"}
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test8266",
|
|
"configuration": "test8266.yaml",
|
|
}
|
|
)
|
|
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
mock_client.device_info.side_effect = [
|
|
RequiresEncryptionAPIError,
|
|
DeviceInfo(
|
|
uses_password=False,
|
|
name="test8266",
|
|
mac_address="11:22:33:44:55:AA",
|
|
),
|
|
]
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
return_value=VALID_NOISE_PSK,
|
|
) as mock_get_encryption_key:
|
|
result = await hass.config_entries.flow.async_configure(
|
|
flow["flow_id"], user_input={}
|
|
)
|
|
|
|
assert len(mock_get_encryption_key.mock_calls) == 1
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == "test8266"
|
|
assert result["data"][CONF_HOST] == "192.168.43.183"
|
|
assert result["data"][CONF_PORT] == 6053
|
|
assert result["data"][CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
assert result["result"]
|
|
assert result["result"].unique_id == "11:22:33:44:55:aa"
|
|
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_zeroconf_encryption_key_via_dashboard_with_api_encryption_prop(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_dashboard: dict[str, Any],
|
|
) -> None:
|
|
"""Test encryption key retrieved from dashboard with api_encryption property set."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={
|
|
"mac": "1122334455aa",
|
|
"api_encryption": "any",
|
|
},
|
|
type="mock_type",
|
|
)
|
|
flow = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
|
|
assert flow["type"] is FlowResultType.FORM
|
|
assert flow["step_id"] == "discovery_confirm"
|
|
assert flow["description_placeholders"] == {"name": "test8266"}
|
|
|
|
mock_dashboard["configured"].append(
|
|
{
|
|
"name": "test8266",
|
|
"configuration": "test8266.yaml",
|
|
}
|
|
)
|
|
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
mock_client.device_info.side_effect = [
|
|
DeviceInfo(
|
|
uses_password=False,
|
|
name="test8266",
|
|
mac_address="11:22:33:44:55:AA",
|
|
),
|
|
]
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.coordinator.ESPHomeDashboardAPI.get_encryption_key",
|
|
return_value=VALID_NOISE_PSK,
|
|
) as mock_get_encryption_key:
|
|
result = await hass.config_entries.flow.async_configure(
|
|
flow["flow_id"], user_input={}
|
|
)
|
|
|
|
assert len(mock_get_encryption_key.mock_calls) == 1
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == "test8266"
|
|
assert result["data"][CONF_HOST] == "192.168.43.183"
|
|
assert result["data"][CONF_PORT] == 6053
|
|
assert result["data"][CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
assert result["result"]
|
|
assert result["result"].unique_id == "11:22:33:44:55:aa"
|
|
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_dashboard", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_zeroconf_no_encryption_key_via_dashboard(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test encryption key not retrieved from dashboard."""
|
|
service_info = ZeroconfServiceInfo(
|
|
ip_address=ip_address("192.168.43.183"),
|
|
ip_addresses=[ip_address("192.168.43.183")],
|
|
hostname="test8266.local.",
|
|
name="mock_name",
|
|
port=6053,
|
|
properties={
|
|
"mac": "1122334455aa",
|
|
},
|
|
type="mock_type",
|
|
)
|
|
flow = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_ZEROCONF}, data=service_info
|
|
)
|
|
|
|
assert flow["type"] is FlowResultType.FORM
|
|
assert flow["step_id"] == "discovery_confirm"
|
|
assert flow["description_placeholders"] == {"name": "test8266"}
|
|
|
|
await dashboard.async_get_dashboard(hass).async_refresh()
|
|
|
|
mock_client.device_info.side_effect = RequiresEncryptionAPIError
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
flow["flow_id"], user_input={}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "test8266"}
|
|
|
|
mock_client.device_info.side_effect = None
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "192.168.43.183",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
async def test_option_flow_allow_service_calls(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_generic_device_entry: Callable[
|
|
[APIClient, list[EntityInfo], list[UserService], list[EntityState]],
|
|
Awaitable[MockConfigEntry],
|
|
],
|
|
) -> None:
|
|
"""Test config flow options for allow service calls."""
|
|
entry = await mock_generic_device_entry(
|
|
mock_client=mock_client,
|
|
entity_info=[],
|
|
user_service=[],
|
|
states=[],
|
|
)
|
|
|
|
result = await hass.config_entries.options.async_init(entry.entry_id)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "init"
|
|
assert result["data_schema"]({}) == {
|
|
CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
|
|
CONF_SUBSCRIBE_LOGS: False,
|
|
}
|
|
|
|
result = await hass.config_entries.options.async_init(entry.entry_id)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "init"
|
|
assert result["data_schema"]({}) == {
|
|
CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
|
|
CONF_SUBSCRIBE_LOGS: False,
|
|
}
|
|
with patch(
|
|
"homeassistant.components.esphome.async_setup_entry", return_value=True
|
|
) as mock_reload:
|
|
result = await hass.config_entries.options.async_configure(
|
|
result["flow_id"],
|
|
user_input={CONF_ALLOW_SERVICE_CALLS: True, CONF_SUBSCRIBE_LOGS: False},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_ALLOW_SERVICE_CALLS: True,
|
|
CONF_SUBSCRIBE_LOGS: False,
|
|
}
|
|
assert len(mock_reload.mock_calls) == 1
|
|
|
|
|
|
async def test_option_flow_subscribe_logs(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
mock_generic_device_entry: Callable[
|
|
[APIClient, list[EntityInfo], list[UserService], list[EntityState]],
|
|
Awaitable[MockConfigEntry],
|
|
],
|
|
) -> None:
|
|
"""Test config flow options with subscribe logs."""
|
|
entry = await mock_generic_device_entry(
|
|
mock_client=mock_client,
|
|
entity_info=[],
|
|
user_service=[],
|
|
states=[],
|
|
)
|
|
|
|
result = await hass.config_entries.options.async_init(entry.entry_id)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "init"
|
|
assert result["data_schema"]({}) == {
|
|
CONF_ALLOW_SERVICE_CALLS: DEFAULT_NEW_CONFIG_ALLOW_ALLOW_SERVICE_CALLS,
|
|
CONF_SUBSCRIBE_LOGS: False,
|
|
}
|
|
|
|
with patch(
|
|
"homeassistant.components.esphome.async_setup_entry", return_value=True
|
|
) as mock_reload:
|
|
result = await hass.config_entries.options.async_configure(
|
|
result["flow_id"],
|
|
user_input={CONF_ALLOW_SERVICE_CALLS: False, CONF_SUBSCRIBE_LOGS: True},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_ALLOW_SERVICE_CALLS: False,
|
|
CONF_SUBSCRIBE_LOGS: True,
|
|
}
|
|
assert len(mock_reload.mock_calls) == 1
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_discovers_name_no_dashboard(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test user step can discover the name and the there is not dashboard."""
|
|
mock_client.device_info.side_effect = [
|
|
RequiresEncryptionAPIError,
|
|
InvalidEncryptionKeyAPIError("Wrong key", "test"),
|
|
DeviceInfo(
|
|
uses_password=False,
|
|
name="test",
|
|
mac_address="11:22:33:44:55:AA",
|
|
),
|
|
]
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "test"}
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert mock_client.noise_psk == VALID_NOISE_PSK
|
|
|
|
|
|
async def mqtt_discovery_test_abort(
|
|
hass: HomeAssistant, payload: str, reason: str
|
|
) -> None:
|
|
"""Test discovery aborted."""
|
|
service_info = MqttServiceInfo(
|
|
topic="esphome/discover/test",
|
|
payload=payload,
|
|
qos=0,
|
|
retain=False,
|
|
subscribed_topic="esphome/discover/#",
|
|
timestamp=None,
|
|
)
|
|
flow = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_MQTT}, data=service_info
|
|
)
|
|
assert flow["type"] is FlowResultType.ABORT
|
|
assert flow["reason"] == reason
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_mqtt_no_mac(hass: HomeAssistant) -> None:
|
|
"""Test discovery aborted if mac is missing in MQTT payload."""
|
|
await mqtt_discovery_test_abort(hass, "{}", "mqtt_missing_mac")
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_mqtt_empty_payload(hass: HomeAssistant) -> None:
|
|
"""Test discovery aborted if MQTT payload is empty."""
|
|
await mqtt_discovery_test_abort(hass, "", "mqtt_missing_payload")
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_mqtt_no_api(hass: HomeAssistant) -> None:
|
|
"""Test discovery aborted if api/port is missing in MQTT payload."""
|
|
await mqtt_discovery_test_abort(hass, '{"mac":"abcdef123456"}', "mqtt_missing_api")
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_mqtt_no_ip(hass: HomeAssistant) -> None:
|
|
"""Test discovery aborted if ip is missing in MQTT payload."""
|
|
await mqtt_discovery_test_abort(
|
|
hass, '{"mac":"abcdef123456","port":6053}', "mqtt_missing_ip"
|
|
)
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_client", "mock_setup_entry", "mock_zeroconf")
|
|
async def test_discovery_mqtt_initiation(hass: HomeAssistant) -> None:
|
|
"""Test discovery importing works."""
|
|
service_info = MqttServiceInfo(
|
|
topic="esphome/discover/test",
|
|
payload='{"name":"mock_name","mac":"1122334455aa","port":6053,"ip":"192.168.43.183"}',
|
|
qos=0,
|
|
retain=False,
|
|
subscribed_topic="esphome/discover/#",
|
|
timestamp=None,
|
|
)
|
|
flow = await hass.config_entries.flow.async_init(
|
|
"esphome", context={"source": config_entries.SOURCE_MQTT}, data=service_info
|
|
)
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
flow["flow_id"], user_input={}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == "test"
|
|
assert result["data"][CONF_HOST] == "192.168.43.183"
|
|
assert result["data"][CONF_PORT] == 6053
|
|
|
|
assert result["result"]
|
|
assert result["result"].unique_id == "11:22:33:44:55:aa"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_flow_name_conflict_migrate(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test handle migration on name conflict."""
|
|
existing_entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_DEVICE_NAME: "test"},
|
|
unique_id="11:22:33:44:55:cc",
|
|
)
|
|
existing_entry.add_to_hass(hass)
|
|
mock_client.device_info = AsyncMock(
|
|
return_value=DeviceInfo(
|
|
uses_password=False,
|
|
name="test",
|
|
mac_address="11:22:33:44:55:AA",
|
|
)
|
|
)
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.MENU
|
|
assert result["step_id"] == "name_conflict"
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={"next_step_id": "name_conflict_migrate"}
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "name_conflict_migrated"
|
|
assert result["description_placeholders"] == {
|
|
"existing_mac": "11:22:33:44:55:cc",
|
|
"mac": "11:22:33:44:55:aa",
|
|
"name": "test",
|
|
}
|
|
assert existing_entry.data == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert existing_entry.unique_id == "11:22:33:44:55:aa"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_setup_entry", "mock_zeroconf")
|
|
async def test_user_flow_name_conflict_overwrite(
|
|
hass: HomeAssistant,
|
|
mock_client: APIClient,
|
|
) -> None:
|
|
"""Test handle overwrite on name conflict."""
|
|
existing_entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={CONF_DEVICE_NAME: "test"},
|
|
unique_id="11:22:33:44:55:cc",
|
|
)
|
|
existing_entry.add_to_hass(hass)
|
|
mock_client.device_info = AsyncMock(
|
|
return_value=DeviceInfo(
|
|
uses_password=False,
|
|
name="test",
|
|
mac_address="11:22:33:44:55:AA",
|
|
)
|
|
)
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
"esphome",
|
|
context={"source": config_entries.SOURCE_USER},
|
|
data={CONF_HOST: "127.0.0.1", CONF_PORT: 6053},
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] is FlowResultType.MENU
|
|
assert result["step_id"] == "name_conflict"
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={"next_step_id": "name_conflict_overwrite"}
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert result["context"]["unique_id"] == "11:22:33:44:55:aa"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_success_with_same_ip_new_name(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig initiation with same ip and new name."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="other", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_successful"
|
|
assert entry.data[CONF_HOST] == "127.0.0.1"
|
|
assert entry.data[CONF_DEVICE_NAME] == "other"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_success_with_new_ip_new_name(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig initiation with new ip and new name."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="other", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_successful"
|
|
assert entry.data[CONF_HOST] == "127.0.0.2"
|
|
assert entry.data[CONF_DEVICE_NAME] == "other"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_success_with_new_ip_same_name(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig initiation with new ip and same name."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_successful"
|
|
assert entry.data[CONF_HOST] == "127.0.0.1"
|
|
assert entry.data[CONF_DEVICE_NAME] == "test"
|
|
assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_success_noise_psk_changes(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig initiation with new ip and new noise psk."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
CONF_NOISE_PSK: VALID_NOISE_PSK,
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
mock_client.device_info.side_effect = [
|
|
RequiresEncryptionAPIError,
|
|
InvalidEncryptionKeyAPIError("Wrong key", "test"),
|
|
DeviceInfo(uses_password=False, name="test", mac_address="11:22:33:44:55:aa"),
|
|
]
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.1", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "Mock Title (test)"}
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "encryption_key"
|
|
assert result["description_placeholders"] == {"name": "Mock Title (test)"}
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_NOISE_PSK: VALID_NOISE_PSK}
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_successful"
|
|
assert entry.data[CONF_HOST] == "127.0.0.1"
|
|
assert entry.data[CONF_DEVICE_NAME] == "test"
|
|
assert entry.data[CONF_NOISE_PSK] == VALID_NOISE_PSK
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_name_conflict_with_existing_entry(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig with a name conflict with an existing entry."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
entry2 = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.2",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "other",
|
|
},
|
|
unique_id="11:22:33:44:55:bb",
|
|
)
|
|
entry2.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="other", mac_address="11:22:33:44:55:aa"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.3", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_name_conflict"
|
|
assert result["description_placeholders"] == {
|
|
"existing_title": "Mock Title",
|
|
"expected_mac": "11:22:33:44:55:aa",
|
|
"host": "127.0.0.3",
|
|
"name": "test",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_attempt_to_change_mac_aborts(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig initiation with valid PSK attempting to change mac."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="other", mac_address="11:22:33:44:55:bb"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_unique_id_changed"
|
|
assert CONF_NOISE_PSK not in entry.data
|
|
assert result["description_placeholders"] == {
|
|
"expected_mac": "11:22:33:44:55:aa",
|
|
"host": "127.0.0.2",
|
|
"name": "test",
|
|
"unexpected_device_name": "other",
|
|
"unexpected_mac": "11:22:33:44:55:bb",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_mac_used_by_other_entry(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig when there is another entry for the mac."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
entry2 = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.2",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test4",
|
|
},
|
|
unique_id="11:22:33:44:55:bb",
|
|
)
|
|
entry2.add_to_hass(hass)
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_already_configured"
|
|
assert result["description_placeholders"] == {
|
|
"title": "Mock Title",
|
|
"name": "test4",
|
|
"mac": "11:22:33:44:55:bb",
|
|
}
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_name_conflict_migrate(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig initiation when device has been replaced."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.MENU
|
|
assert result["step_id"] == "name_conflict"
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={"next_step_id": "name_conflict_migrate"}
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "name_conflict_migrated"
|
|
|
|
assert entry.data == {
|
|
CONF_HOST: "127.0.0.2",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert entry.unique_id == "11:22:33:44:55:bb"
|
|
|
|
|
|
@pytest.mark.usefixtures("mock_zeroconf", "mock_setup_entry")
|
|
async def test_reconfig_name_conflict_overwrite(
|
|
hass: HomeAssistant, mock_client: APIClient
|
|
) -> None:
|
|
"""Test reconfig initiation when device has been replaced."""
|
|
entry = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={
|
|
CONF_HOST: "127.0.0.1",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
},
|
|
unique_id="11:22:33:44:55:aa",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
result = await entry.start_reconfigure_flow(hass)
|
|
|
|
mock_client.device_info.return_value = DeviceInfo(
|
|
uses_password=False, name="test", mac_address="11:22:33:44:55:bb"
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={CONF_HOST: "127.0.0.2", CONF_PORT: 6053}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.MENU
|
|
assert result["step_id"] == "name_conflict"
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input={"next_step_id": "name_conflict_overwrite"}
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
|
|
assert result["data"] == {
|
|
CONF_HOST: "127.0.0.2",
|
|
CONF_PORT: 6053,
|
|
CONF_PASSWORD: "",
|
|
CONF_NOISE_PSK: "",
|
|
CONF_DEVICE_NAME: "test",
|
|
}
|
|
assert result["context"]["unique_id"] == "11:22:33:44:55:bb"
|
|
assert (
|
|
hass.config_entries.async_entry_for_domain_unique_id(
|
|
DOMAIN, "11:22:33:44:55:aa"
|
|
)
|
|
is None
|
|
)
|