mirror of
https://github.com/home-assistant/core.git
synced 2026-07-03 12:46:09 +01:00
18a259718f
Co-authored-by: Claude <noreply@anthropic.com>
454 lines
15 KiB
Python
454 lines
15 KiB
Python
"""Tests for the Vistapool config flow.
|
|
|
|
These tests run in the Home Assistant Core test environment.
|
|
Run with: pytest tests/components/vistapool/test_config_flow.py
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Generator
|
|
from typing import Any
|
|
from unittest.mock import AsyncMock, MagicMock, patch
|
|
|
|
from aioaquarite import AquariteError, AuthenticationError
|
|
import pytest
|
|
|
|
from homeassistant.components.vistapool.const import DOMAIN
|
|
from homeassistant.config_entries import SOURCE_DHCP, SOURCE_USER
|
|
from homeassistant.const import CONF_PASSWORD, CONF_USERNAME
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.data_entry_flow import FlowResultType
|
|
from homeassistant.helpers.service_info.dhcp import DhcpServiceInfo
|
|
|
|
from .conftest import MOCK_PASSWORD, MOCK_POOLS, MOCK_USERNAME
|
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
_DHCP_INFO = DhcpServiceInfo(
|
|
ip="1.2.3.4",
|
|
hostname="sugarwifi",
|
|
macaddress="aabbccddeeff",
|
|
)
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_setup_entry() -> Generator[AsyncMock]:
|
|
"""Prevent actual setup during config flow tests."""
|
|
with patch(
|
|
"homeassistant.components.vistapool.async_setup_entry", return_value=True
|
|
) as mock:
|
|
yield mock
|
|
|
|
|
|
async def _submit(hass: HomeAssistant, flow_id: str) -> dict:
|
|
"""Submit the user step with the standard credentials."""
|
|
return await hass.config_entries.flow.async_configure(
|
|
flow_id,
|
|
{CONF_USERNAME: MOCK_USERNAME, CONF_PASSWORD: MOCK_PASSWORD},
|
|
)
|
|
|
|
|
|
async def _configure(hass: HomeAssistant) -> dict:
|
|
"""Run the user step from start to finish with the standard credentials."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
return await _submit(hass, result["flow_id"])
|
|
|
|
|
|
# ── User Step ─────────────────────────────────────────────────────
|
|
|
|
|
|
async def test_user_step(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test the user step shows a form and creates an entry on submission."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
assert result["data"] == {
|
|
CONF_USERNAME: MOCK_USERNAME,
|
|
CONF_PASSWORD: MOCK_PASSWORD,
|
|
}
|
|
|
|
|
|
# ── Error Handling (each path also verifies the flow can recover) ─
|
|
|
|
|
|
async def test_invalid_auth(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
mock_vistapool_auth: MagicMock,
|
|
) -> None:
|
|
"""Test the flow surfaces invalid_auth and recovers on retry."""
|
|
mock_vistapool_auth.authenticate.side_effect = AuthenticationError
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "invalid_auth"}
|
|
|
|
# Recover: clear the failure and retry the same flow.
|
|
mock_vistapool_auth.authenticate.side_effect = None
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
|
|
|
|
async def test_cannot_connect(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
mock_vistapool_auth: MagicMock,
|
|
) -> None:
|
|
"""Test cannot_connect on auth failure and recovery on retry."""
|
|
mock_vistapool_auth.authenticate.side_effect = AquariteError("network down")
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
mock_vistapool_auth.authenticate.side_effect = None
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
|
|
|
|
async def test_cannot_connect_during_pool_fetch(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test cannot_connect on get_pools failure and recovery on retry."""
|
|
mock_vistapool_client.get_pools.side_effect = AquariteError("network down")
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
mock_vistapool_client.get_pools.side_effect = None
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
|
|
|
|
async def test_unknown_exception(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
mock_vistapool_auth: MagicMock,
|
|
) -> None:
|
|
"""Test unknown error on auth failure and recovery on retry."""
|
|
mock_vistapool_auth.authenticate.side_effect = RuntimeError("Connection refused")
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "unknown"}
|
|
|
|
mock_vistapool_auth.authenticate.side_effect = None
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
|
|
|
|
async def test_unknown_exception_during_pool_fetch(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test unknown error on get_pools failure and recovery on retry."""
|
|
mock_vistapool_client.get_pools.side_effect = RuntimeError("boom")
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "unknown"}
|
|
|
|
mock_vistapool_client.get_pools.side_effect = None
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
|
|
|
|
async def test_no_pools(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test no_pools on an empty account and recovery once pools appear."""
|
|
mock_vistapool_client.get_pools.return_value = {}
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "no_pools"}
|
|
|
|
# Recover: pools appear on the account; resubmit the same flow.
|
|
mock_vistapool_client.get_pools.return_value = MOCK_POOLS
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
|
|
|
|
async def test_duplicate_account_aborts(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test the flow aborts when an entry for the account already exists."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
|
|
result = await _configure(hass)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
# ── DHCP discovery ────────────────────────────────────────────────
|
|
|
|
|
|
async def test_dhcp_discovery_starts_user_flow(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test DHCP discovery routes the user into the credentials step."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": SOURCE_DHCP},
|
|
data=_DHCP_INFO,
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
result = await _submit(hass, result["flow_id"])
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == MOCK_USERNAME
|
|
|
|
|
|
async def test_dhcp_discovery_aborts_when_configured(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test DHCP discovery aborts when an account is already configured."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": SOURCE_DHCP},
|
|
data=_DHCP_INFO,
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
async def test_dhcp_discovery_aborts_when_in_progress(
|
|
hass: HomeAssistant,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
) -> None:
|
|
"""Test a second DHCP discovery aborts while another flow is in progress."""
|
|
first = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": SOURCE_DHCP},
|
|
data=_DHCP_INFO,
|
|
)
|
|
assert first["type"] is FlowResultType.FORM
|
|
|
|
second = await hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": SOURCE_DHCP},
|
|
data=_DHCP_INFO,
|
|
)
|
|
|
|
assert second["type"] is FlowResultType.ABORT
|
|
assert second["reason"] == "already_in_progress"
|
|
|
|
|
|
_NEW_PASSWORD = "new-password"
|
|
|
|
_FLOW_PARAMS = [
|
|
pytest.param(
|
|
MockConfigEntry.start_reauth_flow,
|
|
"reauth_confirm",
|
|
"reauth_successful",
|
|
id="reauth",
|
|
),
|
|
pytest.param(
|
|
MockConfigEntry.start_reconfigure_flow,
|
|
"reconfigure",
|
|
"reconfigure_successful",
|
|
id="reconfigure",
|
|
),
|
|
]
|
|
|
|
|
|
@pytest.mark.parametrize(("flow_starter", "step_id", "success_reason"), _FLOW_PARAMS)
|
|
async def test_credential_update_flow(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
flow_starter: Any,
|
|
step_id: str,
|
|
success_reason: str,
|
|
) -> None:
|
|
"""Test reauth / reconfigure updates the stored password and reloads the entry."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
|
|
result = await flow_starter(mock_config_entry, hass)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == step_id
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], {CONF_PASSWORD: _NEW_PASSWORD}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == success_reason
|
|
assert mock_config_entry.data[CONF_PASSWORD] == _NEW_PASSWORD
|
|
assert mock_setup_entry.call_count == 1
|
|
|
|
|
|
@pytest.mark.parametrize(("flow_starter", "step_id", "success_reason"), _FLOW_PARAMS)
|
|
async def test_credential_update_invalid_auth(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
mock_vistapool_auth: MagicMock,
|
|
flow_starter: Any,
|
|
step_id: str,
|
|
success_reason: str,
|
|
) -> None:
|
|
"""Test reauth / reconfigure surfaces invalid_auth and recovers on retry."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
mock_vistapool_auth.authenticate.side_effect = AuthenticationError
|
|
|
|
result = await flow_starter(mock_config_entry, hass)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], {CONF_PASSWORD: _NEW_PASSWORD}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "invalid_auth"}
|
|
|
|
mock_vistapool_auth.authenticate.side_effect = None
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], {CONF_PASSWORD: _NEW_PASSWORD}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == success_reason
|
|
assert mock_config_entry.data[CONF_PASSWORD] == _NEW_PASSWORD
|
|
assert mock_setup_entry.call_count == 1
|
|
|
|
|
|
@pytest.mark.parametrize(("flow_starter", "step_id", "success_reason"), _FLOW_PARAMS)
|
|
async def test_credential_update_account_mismatch(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
mock_vistapool_auth: MagicMock,
|
|
flow_starter: Any,
|
|
step_id: str,
|
|
success_reason: str,
|
|
) -> None:
|
|
"""Test reauth / reconfigure aborts when credentials belong to a different account."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
mock_vistapool_auth.user_id = "a-different-firebase-uid"
|
|
|
|
result = await flow_starter(mock_config_entry, hass)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], {CONF_PASSWORD: _NEW_PASSWORD}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "account_mismatch"
|
|
assert mock_setup_entry.call_count == 0
|
|
|
|
|
|
@pytest.mark.parametrize(("flow_starter", "step_id", "success_reason"), _FLOW_PARAMS)
|
|
@pytest.mark.parametrize(
|
|
("auth_exception", "expected_error"),
|
|
[
|
|
pytest.param(AquariteError("network"), "cannot_connect", id="cannot_connect"),
|
|
pytest.param(RuntimeError("boom"), "unknown", id="unknown"),
|
|
],
|
|
)
|
|
async def test_credential_update_error_paths(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_setup_entry: AsyncMock,
|
|
mock_vistapool_client: AsyncMock,
|
|
mock_vistapool_auth: MagicMock,
|
|
flow_starter: Any,
|
|
step_id: str,
|
|
success_reason: str,
|
|
auth_exception: Exception,
|
|
expected_error: str,
|
|
) -> None:
|
|
"""Test reauth / reconfigure surfaces non-auth errors and recovers on retry."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
mock_vistapool_auth.authenticate.side_effect = auth_exception
|
|
|
|
result = await flow_starter(mock_config_entry, hass)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], {CONF_PASSWORD: _NEW_PASSWORD}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == step_id
|
|
assert result["errors"] == {"base": expected_error}
|
|
|
|
mock_vistapool_auth.authenticate.side_effect = None
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], {CONF_PASSWORD: _NEW_PASSWORD}
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == success_reason
|
|
assert mock_setup_entry.call_count == 1
|