mirror of https://github.com/home-assistant/core
279 lines
9.6 KiB
Python
279 lines
9.6 KiB
Python
"""Tests for Tankerkoenig config flow."""
|
|
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
from aiotankerkoenig.exceptions import TankerkoenigInvalidKeyError
|
|
|
|
from homeassistant.components.tankerkoenig.const import (
|
|
CONF_FUEL_TYPES,
|
|
CONF_STATIONS,
|
|
DOMAIN,
|
|
)
|
|
from homeassistant.config_entries import SOURCE_USER
|
|
from homeassistant.const import (
|
|
CONF_API_KEY,
|
|
CONF_LATITUDE,
|
|
CONF_LOCATION,
|
|
CONF_LONGITUDE,
|
|
CONF_NAME,
|
|
CONF_RADIUS,
|
|
CONF_SHOW_ON_MAP,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.data_entry_flow import FlowResultType
|
|
from homeassistant.setup import async_setup_component
|
|
|
|
from .const import NEARBY_STATIONS
|
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
MOCK_USER_DATA = {
|
|
CONF_NAME: "Home",
|
|
CONF_API_KEY: "269534f6-xxxx-xxxx-xxxx-yyyyzzzzxxxx",
|
|
CONF_FUEL_TYPES: ["e5"],
|
|
CONF_LOCATION: {CONF_LATITUDE: 51.0, CONF_LONGITUDE: 13.0},
|
|
CONF_RADIUS: 2.0,
|
|
}
|
|
|
|
MOCK_STATIONS_DATA = {
|
|
CONF_STATIONS: [
|
|
"3bcd61da-xxxx-xxxx-xxxx-19d5523a7ae8",
|
|
"36b4b812-xxxx-xxxx-xxxx-c51735325858",
|
|
],
|
|
}
|
|
|
|
MOCK_OPTIONS_DATA = {
|
|
**MOCK_USER_DATA,
|
|
CONF_STATIONS: [
|
|
"3bcd61da-xxxx-xxxx-xxxx-19d5523a7ae8",
|
|
"36b4b812-xxxx-xxxx-xxxx-c51735325858",
|
|
"54e2b642-xxxx-xxxx-xxxx-87cd4e9867f1",
|
|
],
|
|
}
|
|
|
|
|
|
async def test_user(hass: HomeAssistant) -> None:
|
|
"""Test starting a flow by user."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
with (
|
|
patch(
|
|
"homeassistant.components.tankerkoenig.async_setup_entry", return_value=True
|
|
) as mock_setup_entry,
|
|
patch(
|
|
"homeassistant.components.tankerkoenig.config_flow.Tankerkoenig.nearby_stations",
|
|
return_value=NEARBY_STATIONS,
|
|
),
|
|
):
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input=MOCK_USER_DATA
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "select_station"
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input=MOCK_STATIONS_DATA
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert result["data"][CONF_NAME] == "Home"
|
|
assert result["data"][CONF_API_KEY] == "269534f6-xxxx-xxxx-xxxx-yyyyzzzzxxxx"
|
|
assert result["data"][CONF_FUEL_TYPES] == ["e5"]
|
|
assert result["data"][CONF_LOCATION] == {"latitude": 51.0, "longitude": 13.0}
|
|
assert result["data"][CONF_RADIUS] == 2.0
|
|
assert result["data"][CONF_STATIONS] == [
|
|
"3bcd61da-xxxx-xxxx-xxxx-19d5523a7ae8",
|
|
"36b4b812-xxxx-xxxx-xxxx-c51735325858",
|
|
]
|
|
assert result["options"][CONF_SHOW_ON_MAP]
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert mock_setup_entry.called
|
|
|
|
|
|
async def test_user_already_configured(hass: HomeAssistant) -> None:
|
|
"""Test starting a flow by user with an already configured region."""
|
|
|
|
mock_config = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data={**MOCK_USER_DATA, **MOCK_STATIONS_DATA},
|
|
unique_id=f"{MOCK_USER_DATA[CONF_LOCATION][CONF_LATITUDE]}_{MOCK_USER_DATA[CONF_LOCATION][CONF_LONGITUDE]}",
|
|
)
|
|
mock_config.add_to_hass(hass)
|
|
|
|
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"], user_input=MOCK_USER_DATA
|
|
)
|
|
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
async def test_exception_security(hass: HomeAssistant) -> None:
|
|
"""Test starting a flow by user with invalid api key."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
with patch(
|
|
"homeassistant.components.tankerkoenig.config_flow.Tankerkoenig.nearby_stations",
|
|
side_effect=TankerkoenigInvalidKeyError,
|
|
):
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input=MOCK_USER_DATA
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
assert result["errors"][CONF_API_KEY] == "invalid_auth"
|
|
|
|
|
|
async def test_user_no_stations(hass: HomeAssistant) -> None:
|
|
"""Test starting a flow by user which does not find any station."""
|
|
result = await hass.config_entries.flow.async_init(
|
|
DOMAIN, context={"source": SOURCE_USER}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
|
|
with patch(
|
|
"homeassistant.components.tankerkoenig.config_flow.Tankerkoenig.nearby_stations",
|
|
return_value=[],
|
|
):
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"], user_input=MOCK_USER_DATA
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "user"
|
|
assert result["errors"][CONF_RADIUS] == "no_stations"
|
|
|
|
|
|
async def test_reauth(hass: HomeAssistant, config_entry: MockConfigEntry) -> None:
|
|
"""Test starting a flow by user to re-auth."""
|
|
config_entry.add_to_hass(hass)
|
|
# re-auth initialized
|
|
result = await config_entry.start_reauth_flow(hass)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reauth_confirm"
|
|
|
|
with (
|
|
patch(
|
|
"homeassistant.components.tankerkoenig.async_setup_entry", return_value=True
|
|
) as mock_setup_entry,
|
|
patch(
|
|
"homeassistant.components.tankerkoenig.config_flow.Tankerkoenig.nearby_stations",
|
|
) as mock_nearby_stations,
|
|
):
|
|
# re-auth unsuccessful
|
|
mock_nearby_stations.side_effect = TankerkoenigInvalidKeyError("Booom!")
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
user_input={
|
|
CONF_API_KEY: "269534f6-aaaa-bbbb-cccc-yyyyzzzzxxxx",
|
|
},
|
|
)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "reauth_confirm"
|
|
assert result["errors"] == {CONF_API_KEY: "invalid_auth"}
|
|
|
|
# re-auth successful
|
|
mock_nearby_stations.side_effect = None
|
|
result = await hass.config_entries.flow.async_configure(
|
|
result["flow_id"],
|
|
user_input={
|
|
CONF_API_KEY: "269534f6-aaaa-bbbb-cccc-yyyyzzzzxxxx",
|
|
},
|
|
)
|
|
assert result["type"] is FlowResultType.ABORT
|
|
assert result["reason"] == "reauth_successful"
|
|
|
|
mock_setup_entry.assert_called()
|
|
|
|
entry = hass.config_entries.async_get_entry(config_entry.entry_id)
|
|
assert entry.data[CONF_API_KEY] == "269534f6-aaaa-bbbb-cccc-yyyyzzzzxxxx"
|
|
|
|
|
|
async def test_options_flow(hass: HomeAssistant, tankerkoenig: AsyncMock) -> None:
|
|
"""Test options flow."""
|
|
|
|
mock_config = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data=MOCK_OPTIONS_DATA,
|
|
options={CONF_SHOW_ON_MAP: True},
|
|
unique_id=f"{DOMAIN}_{MOCK_USER_DATA[CONF_LOCATION][CONF_LATITUDE]}_{MOCK_USER_DATA[CONF_LOCATION][CONF_LONGITUDE]}",
|
|
)
|
|
mock_config.add_to_hass(hass)
|
|
assert await async_setup_component(hass, DOMAIN, {})
|
|
await hass.async_block_till_done()
|
|
|
|
with (
|
|
patch(
|
|
"homeassistant.components.tankerkoenig.config_flow.Tankerkoenig.nearby_stations",
|
|
return_value=NEARBY_STATIONS,
|
|
),
|
|
patch(
|
|
"homeassistant.config_entries.ConfigEntries.async_reload"
|
|
) as mock_async_reload,
|
|
):
|
|
result = await hass.config_entries.options.async_init(mock_config.entry_id)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "init"
|
|
|
|
result = await hass.config_entries.options.async_configure(
|
|
result["flow_id"],
|
|
user_input={
|
|
CONF_SHOW_ON_MAP: False,
|
|
CONF_STATIONS: MOCK_OPTIONS_DATA[CONF_STATIONS],
|
|
},
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert not mock_config.options[CONF_SHOW_ON_MAP]
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert mock_async_reload.call_count == 1
|
|
|
|
|
|
async def test_options_flow_error(hass: HomeAssistant) -> None:
|
|
"""Test options flow."""
|
|
|
|
mock_config = MockConfigEntry(
|
|
domain=DOMAIN,
|
|
data=MOCK_OPTIONS_DATA,
|
|
options={CONF_SHOW_ON_MAP: True},
|
|
unique_id=f"{DOMAIN}_{MOCK_USER_DATA[CONF_LOCATION][CONF_LATITUDE]}_{MOCK_USER_DATA[CONF_LOCATION][CONF_LONGITUDE]}",
|
|
)
|
|
mock_config.add_to_hass(hass)
|
|
|
|
with patch(
|
|
"homeassistant.components.tankerkoenig.config_flow.Tankerkoenig.nearby_stations",
|
|
side_effect=TankerkoenigInvalidKeyError("Booom!"),
|
|
) as mock_nearby_stations:
|
|
result = await hass.config_entries.options.async_init(mock_config.entry_id)
|
|
assert result["type"] is FlowResultType.FORM
|
|
assert result["step_id"] == "init"
|
|
assert result["errors"] == {"base": "invalid_auth"}
|
|
|
|
mock_nearby_stations.return_value = NEARBY_STATIONS
|
|
|
|
result = await hass.config_entries.options.async_configure(
|
|
result["flow_id"],
|
|
user_input={
|
|
CONF_SHOW_ON_MAP: False,
|
|
CONF_STATIONS: MOCK_OPTIONS_DATA[CONF_STATIONS],
|
|
},
|
|
)
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
assert not mock_config.options[CONF_SHOW_ON_MAP]
|