mirror of https://github.com/home-assistant/core
237 lines
7.8 KiB
Python
237 lines
7.8 KiB
Python
"""Tests for the MadVR config flow."""
|
|
|
|
from collections.abc import AsyncGenerator
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
import pytest
|
|
|
|
from homeassistant.components.madvr.const import DEFAULT_NAME, DOMAIN
|
|
from homeassistant.config_entries import SOURCE_USER
|
|
from homeassistant.const import CONF_HOST, CONF_PORT
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.data_entry_flow import FlowResultType
|
|
|
|
from .const import MOCK_CONFIG, MOCK_MAC, MOCK_MAC_NEW
|
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
async def avoid_wait() -> AsyncGenerator[None]:
|
|
"""Mock sleep."""
|
|
with patch("homeassistant.components.madvr.config_flow.RETRY_INTERVAL", 0):
|
|
yield
|
|
|
|
|
|
async def test_full_flow(
|
|
hass: HomeAssistant,
|
|
mock_madvr_client: AsyncMock,
|
|
mock_setup_entry: AsyncMock,
|
|
) -> None:
|
|
"""Test full config flow."""
|
|
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 hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: MOCK_CONFIG[CONF_HOST], CONF_PORT: MOCK_CONFIG[CONF_PORT]},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"] == {
|
|
CONF_HOST: MOCK_CONFIG[CONF_HOST],
|
|
CONF_PORT: MOCK_CONFIG[CONF_PORT],
|
|
}
|
|
assert result["result"].unique_id == MOCK_MAC
|
|
mock_madvr_client.open_connection.assert_called_once()
|
|
mock_madvr_client.async_add_tasks.assert_called_once()
|
|
mock_madvr_client.async_cancel_tasks.assert_called_once()
|
|
|
|
|
|
async def test_flow_errors(
|
|
hass: HomeAssistant,
|
|
mock_madvr_client: AsyncMock,
|
|
mock_setup_entry: AsyncMock,
|
|
) -> None:
|
|
"""Test error handling in config flow."""
|
|
mock_madvr_client.open_connection.side_effect = TimeoutError
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: MOCK_CONFIG[CONF_HOST], CONF_PORT: MOCK_CONFIG[CONF_PORT]},
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
mock_madvr_client.open_connection.side_effect = None
|
|
mock_madvr_client.connected = False
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: MOCK_CONFIG[CONF_HOST], CONF_PORT: MOCK_CONFIG[CONF_PORT]},
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
mock_madvr_client.connected = True
|
|
mock_madvr_client.mac_address = None
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: MOCK_CONFIG[CONF_HOST], CONF_PORT: MOCK_CONFIG[CONF_PORT]},
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "no_mac"}
|
|
|
|
# ensure an error is recoverable
|
|
mock_madvr_client.mac_address = MOCK_MAC
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: MOCK_CONFIG[CONF_HOST], CONF_PORT: MOCK_CONFIG[CONF_PORT]},
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["title"] == DEFAULT_NAME
|
|
assert result["data"] == {
|
|
CONF_HOST: MOCK_CONFIG[CONF_HOST],
|
|
CONF_PORT: MOCK_CONFIG[CONF_PORT],
|
|
}
|
|
|
|
# Verify method calls
|
|
assert mock_madvr_client.open_connection.call_count == 4
|
|
assert mock_madvr_client.async_add_tasks.call_count == 2
|
|
# the first call will not call this due to timeout as expected
|
|
assert mock_madvr_client.async_cancel_tasks.call_count == 2
|
|
|
|
|
|
async def test_duplicate(
|
|
hass: HomeAssistant,
|
|
mock_madvr_client: AsyncMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test duplicate config entries."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: MOCK_CONFIG[CONF_HOST], CONF_PORT: MOCK_CONFIG[CONF_PORT]},
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
async def test_reconfigure_flow(
|
|
hass: HomeAssistant,
|
|
mock_madvr_client: AsyncMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test reconfigure flow."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
result = await mock_config_entry.start_reconfigure_flow(hass)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reconfigure"
|
|
assert result["errors"] == {}
|
|
|
|
# define new host
|
|
new_host = "192.168.1.100"
|
|
# make sure setting port works
|
|
new_port = 44078
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: new_host, CONF_PORT: new_port},
|
|
)
|
|
|
|
# should get the abort with success result
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_successful"
|
|
|
|
# Verify that the config entry was updated
|
|
assert mock_config_entry.data[CONF_HOST] == new_host
|
|
assert mock_config_entry.data[CONF_PORT] == new_port
|
|
|
|
# Verify that the connection was tested
|
|
mock_madvr_client.open_connection.assert_called()
|
|
mock_madvr_client.async_add_tasks.assert_called()
|
|
mock_madvr_client.async_cancel_tasks.assert_called()
|
|
|
|
|
|
async def test_reconfigure_new_device(
|
|
hass: HomeAssistant,
|
|
mock_madvr_client: AsyncMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test reconfigure flow."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
# test reconfigure with a new device (should fail)
|
|
result = await mock_config_entry.start_reconfigure_flow(hass)
|
|
|
|
# define new host
|
|
new_host = "192.168.1.100"
|
|
# make sure setting port works
|
|
new_port = 44078
|
|
|
|
# modify test_connection so it returns new_mac
|
|
mock_madvr_client.mac_address = MOCK_MAC_NEW
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: new_host, CONF_PORT: new_port},
|
|
)
|
|
|
|
# unique id should remain unchanged with new device, should fail
|
|
assert mock_config_entry.unique_id == MOCK_MAC
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "set_up_new_device"
|
|
|
|
|
|
async def test_reconfigure_flow_errors(
|
|
hass: HomeAssistant,
|
|
mock_madvr_client: AsyncMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test error handling in reconfigure flow."""
|
|
mock_config_entry.add_to_hass(hass)
|
|
|
|
result = await mock_config_entry.start_reconfigure_flow(hass)
|
|
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reconfigure"
|
|
|
|
# Test CannotConnect error
|
|
mock_madvr_client.open_connection.side_effect = TimeoutError
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: "192.168.1.100", CONF_PORT: 44077},
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
# Test no_mac error
|
|
mock_madvr_client.open_connection.side_effect = None
|
|
mock_madvr_client.connected = True
|
|
mock_madvr_client.mac_address = None
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: "192.168.1.100", CONF_PORT: 44077},
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["errors"] == {"base": "no_mac"}
|
|
|
|
# Ensure errors are recoverable
|
|
mock_madvr_client.mac_address = MOCK_MAC
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
{CONF_HOST: "192.168.1.100", CONF_PORT: 44077},
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reconfigure_successful"
|