mirror of
https://github.com/home-assistant/core.git
synced 2025-07-15 17:27:10 +00:00
Add DHCP discovery to Obihai (#88984)
* Add DHCP discovery to Obihai * Unique ID is MAC * Move try blocks, cleanup * Migrate existing unique_ids * Use PyObihai to update Unique ID * Auth then use get_device_mac * Config flow changes * Reworking flow according to feedback * Cleanup
This commit is contained in:
parent
fa332668d6
commit
7c6a32ebb5
@ -1,9 +1,11 @@
|
|||||||
"""The Obihai integration."""
|
"""The Obihai integration."""
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
|
|
||||||
from .const import PLATFORMS
|
from .connectivity import ObihaiConnection
|
||||||
|
from .const import LOGGER, PLATFORMS
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
@ -13,6 +15,32 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
async def async_migrate_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
|
"""Migrate old entry."""
|
||||||
|
|
||||||
|
version = entry.version
|
||||||
|
|
||||||
|
LOGGER.debug("Migrating from version %s", version)
|
||||||
|
if version != 2:
|
||||||
|
requester = ObihaiConnection(
|
||||||
|
entry.data[CONF_HOST],
|
||||||
|
username=entry.data[CONF_USERNAME],
|
||||||
|
password=entry.data[CONF_PASSWORD],
|
||||||
|
)
|
||||||
|
await hass.async_add_executor_job(requester.update)
|
||||||
|
|
||||||
|
new_unique_id = await hass.async_add_executor_job(
|
||||||
|
requester.pyobihai.get_device_mac
|
||||||
|
)
|
||||||
|
hass.config_entries.async_update_entry(entry, unique_id=new_unique_id)
|
||||||
|
|
||||||
|
entry.version = 2
|
||||||
|
|
||||||
|
LOGGER.info("Migration to version %s successful", entry.version)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||||
"""Unload a config entry."""
|
"""Unload a config entry."""
|
||||||
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
||||||
|
@ -1,10 +1,14 @@
|
|||||||
"""Config flow to configure the Obihai integration."""
|
"""Config flow to configure the Obihai integration."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from socket import gaierror, gethostbyname
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
|
from pyobihai import PyObihai
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
|
from homeassistant.components import dhcp
|
||||||
from homeassistant.config_entries import ConfigFlow
|
from homeassistant.config_entries import ConfigFlow
|
||||||
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, CONF_USERNAME
|
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, CONF_USERNAME
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
@ -16,11 +20,11 @@ from .const import DEFAULT_PASSWORD, DEFAULT_USERNAME, DOMAIN
|
|||||||
DATA_SCHEMA = vol.Schema(
|
DATA_SCHEMA = vol.Schema(
|
||||||
{
|
{
|
||||||
vol.Required(CONF_HOST): str,
|
vol.Required(CONF_HOST): str,
|
||||||
vol.Optional(
|
vol.Required(
|
||||||
CONF_USERNAME,
|
CONF_USERNAME,
|
||||||
default=DEFAULT_USERNAME,
|
default=DEFAULT_USERNAME,
|
||||||
): str,
|
): str,
|
||||||
vol.Optional(
|
vol.Required(
|
||||||
CONF_PASSWORD,
|
CONF_PASSWORD,
|
||||||
default=DEFAULT_PASSWORD,
|
default=DEFAULT_PASSWORD,
|
||||||
): str,
|
): str,
|
||||||
@ -28,48 +32,122 @@ DATA_SCHEMA = vol.Schema(
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def async_validate_creds(hass: HomeAssistant, user_input: dict[str, Any]) -> bool:
|
async def async_validate_creds(
|
||||||
|
hass: HomeAssistant, user_input: dict[str, Any]
|
||||||
|
) -> PyObihai | None:
|
||||||
"""Manage Obihai options."""
|
"""Manage Obihai options."""
|
||||||
return await hass.async_add_executor_job(
|
|
||||||
validate_auth,
|
if user_input[CONF_USERNAME] and user_input[CONF_PASSWORD]:
|
||||||
user_input[CONF_HOST],
|
return await hass.async_add_executor_job(
|
||||||
user_input[CONF_USERNAME],
|
validate_auth,
|
||||||
user_input[CONF_PASSWORD],
|
user_input[CONF_HOST],
|
||||||
)
|
user_input[CONF_USERNAME],
|
||||||
|
user_input[CONF_PASSWORD],
|
||||||
|
)
|
||||||
|
|
||||||
|
# Don't bother authenticating if we've already determined the credentials are invalid
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
class ObihaiFlowHandler(ConfigFlow, domain=DOMAIN):
|
class ObihaiFlowHandler(ConfigFlow, domain=DOMAIN):
|
||||||
"""Config flow for Obihai."""
|
"""Config flow for Obihai."""
|
||||||
|
|
||||||
VERSION = 1
|
VERSION = 2
|
||||||
|
discovery_schema: vol.Schema | None = None
|
||||||
|
_dhcp_discovery_info: dhcp.DhcpServiceInfo | None = None
|
||||||
|
|
||||||
async def async_step_user(
|
async def async_step_user(
|
||||||
self, user_input: dict[str, Any] | None = None
|
self, user_input: dict[str, Any] | None = None
|
||||||
) -> FlowResult:
|
) -> FlowResult:
|
||||||
"""Handle a flow initialized by the user."""
|
"""Handle a flow initialized by the user."""
|
||||||
|
|
||||||
errors: dict[str, str] = {}
|
errors: dict[str, str] = {}
|
||||||
|
ip: str | None = None
|
||||||
|
|
||||||
if user_input is not None:
|
if user_input is not None:
|
||||||
self._async_abort_entries_match({CONF_HOST: user_input[CONF_HOST]})
|
try:
|
||||||
if await async_validate_creds(self.hass, user_input):
|
ip = gethostbyname(user_input[CONF_HOST])
|
||||||
return self.async_create_entry(
|
except gaierror:
|
||||||
title=user_input[CONF_HOST],
|
errors["base"] = "cannot_connect"
|
||||||
data=user_input,
|
|
||||||
)
|
|
||||||
errors["base"] = "cannot_connect"
|
|
||||||
|
|
||||||
data_schema = self.add_suggested_values_to_schema(DATA_SCHEMA, user_input)
|
if ip:
|
||||||
|
if pyobihai := await async_validate_creds(self.hass, user_input):
|
||||||
|
device_mac = await self.hass.async_add_executor_job(
|
||||||
|
pyobihai.get_device_mac
|
||||||
|
)
|
||||||
|
await self.async_set_unique_id(device_mac)
|
||||||
|
self._abort_if_unique_id_configured()
|
||||||
|
|
||||||
|
return self.async_create_entry(
|
||||||
|
title=user_input[CONF_HOST],
|
||||||
|
data=user_input,
|
||||||
|
)
|
||||||
|
errors["base"] = "invalid_auth"
|
||||||
|
|
||||||
|
data_schema = self.discovery_schema or DATA_SCHEMA
|
||||||
return self.async_show_form(
|
return self.async_show_form(
|
||||||
step_id="user",
|
step_id="user",
|
||||||
errors=errors,
|
errors=errors,
|
||||||
data_schema=data_schema,
|
data_schema=self.add_suggested_values_to_schema(data_schema, user_input),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
async def async_step_dhcp(self, discovery_info: dhcp.DhcpServiceInfo) -> FlowResult:
|
||||||
|
"""Prepare configuration for a DHCP discovered Obihai."""
|
||||||
|
|
||||||
|
self._dhcp_discovery_info = discovery_info
|
||||||
|
return await self.async_step_dhcp_confirm()
|
||||||
|
|
||||||
|
async def async_step_dhcp_confirm(
|
||||||
|
self, user_input: dict[str, Any] | None = None
|
||||||
|
) -> FlowResult:
|
||||||
|
"""Attempt to confirm."""
|
||||||
|
assert self._dhcp_discovery_info
|
||||||
|
await self.async_set_unique_id(self._dhcp_discovery_info.macaddress)
|
||||||
|
self._abort_if_unique_id_configured()
|
||||||
|
|
||||||
|
if user_input is None:
|
||||||
|
credentials = {
|
||||||
|
CONF_HOST: self._dhcp_discovery_info.ip,
|
||||||
|
CONF_PASSWORD: DEFAULT_PASSWORD,
|
||||||
|
CONF_USERNAME: DEFAULT_USERNAME,
|
||||||
|
}
|
||||||
|
if await async_validate_creds(self.hass, credentials):
|
||||||
|
self.discovery_schema = self.add_suggested_values_to_schema(
|
||||||
|
DATA_SCHEMA, credentials
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
self.discovery_schema = self.add_suggested_values_to_schema(
|
||||||
|
DATA_SCHEMA,
|
||||||
|
{
|
||||||
|
CONF_HOST: self._dhcp_discovery_info.ip,
|
||||||
|
CONF_USERNAME: "",
|
||||||
|
CONF_PASSWORD: "",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
# Show the confirmation dialog
|
||||||
|
return self.async_show_form(
|
||||||
|
step_id="dhcp_confirm",
|
||||||
|
data_schema=self.discovery_schema,
|
||||||
|
description_placeholders={CONF_HOST: self._dhcp_discovery_info.ip},
|
||||||
|
)
|
||||||
|
|
||||||
|
return await self.async_step_user(user_input=user_input)
|
||||||
|
|
||||||
# DEPRECATED
|
# DEPRECATED
|
||||||
async def async_step_import(self, config: dict[str, Any]) -> FlowResult:
|
async def async_step_import(self, config: dict[str, Any]) -> FlowResult:
|
||||||
"""Handle a flow initialized by importing a config."""
|
"""Handle a flow initialized by importing a config."""
|
||||||
self._async_abort_entries_match({CONF_HOST: config[CONF_HOST]})
|
|
||||||
if await async_validate_creds(self.hass, config):
|
try:
|
||||||
|
_ = gethostbyname(config[CONF_HOST])
|
||||||
|
except gaierror:
|
||||||
|
return self.async_abort(reason="cannot_connect")
|
||||||
|
|
||||||
|
if pyobihai := await async_validate_creds(self.hass, config):
|
||||||
|
device_mac = await self.hass.async_add_executor_job(pyobihai.get_device_mac)
|
||||||
|
await self.async_set_unique_id(device_mac)
|
||||||
|
self._abort_if_unique_id_configured()
|
||||||
|
|
||||||
return self.async_create_entry(
|
return self.async_create_entry(
|
||||||
title=config.get(CONF_NAME, config[CONF_HOST]),
|
title=config.get(CONF_NAME, config[CONF_HOST]),
|
||||||
data={
|
data={
|
||||||
@ -79,4 +157,4 @@ class ObihaiFlowHandler(ConfigFlow, domain=DOMAIN):
|
|||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
return self.async_abort(reason="cannot_connect")
|
return self.async_abort(reason="invalid_auth")
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
"""Support for Obihai Connectivity."""
|
"""Support for Obihai Connectivity."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from pyobihai import PyObihai
|
from pyobihai import PyObihai
|
||||||
@ -12,6 +13,7 @@ def get_pyobihai(
|
|||||||
password: str,
|
password: str,
|
||||||
) -> PyObihai:
|
) -> PyObihai:
|
||||||
"""Retrieve an authenticated PyObihai."""
|
"""Retrieve an authenticated PyObihai."""
|
||||||
|
|
||||||
return PyObihai(host, username, password)
|
return PyObihai(host, username, password)
|
||||||
|
|
||||||
|
|
||||||
@ -19,16 +21,17 @@ def validate_auth(
|
|||||||
host: str,
|
host: str,
|
||||||
username: str,
|
username: str,
|
||||||
password: str,
|
password: str,
|
||||||
) -> bool:
|
) -> PyObihai | None:
|
||||||
"""Test if the given setting works as expected."""
|
"""Test if the given setting works as expected."""
|
||||||
|
|
||||||
obi = get_pyobihai(host, username, password)
|
obi = get_pyobihai(host, username, password)
|
||||||
|
|
||||||
login = obi.check_account()
|
login = obi.check_account()
|
||||||
if not login:
|
if not login:
|
||||||
LOGGER.debug("Invalid credentials")
|
LOGGER.debug("Invalid credentials")
|
||||||
return False
|
return None
|
||||||
|
|
||||||
return True
|
return obi
|
||||||
|
|
||||||
|
|
||||||
class ObihaiConnection:
|
class ObihaiConnection:
|
||||||
@ -53,6 +56,7 @@ class ObihaiConnection:
|
|||||||
|
|
||||||
def update(self) -> bool:
|
def update(self) -> bool:
|
||||||
"""Validate connection and retrieve a list of sensors."""
|
"""Validate connection and retrieve a list of sensors."""
|
||||||
|
|
||||||
if not self.pyobihai:
|
if not self.pyobihai:
|
||||||
self.pyobihai = get_pyobihai(self.host, self.username, self.password)
|
self.pyobihai = get_pyobihai(self.host, self.username, self.password)
|
||||||
|
|
||||||
|
@ -3,6 +3,11 @@
|
|||||||
"name": "Obihai",
|
"name": "Obihai",
|
||||||
"codeowners": ["@dshokouhi", "@ejpenney"],
|
"codeowners": ["@dshokouhi", "@ejpenney"],
|
||||||
"config_flow": true,
|
"config_flow": true,
|
||||||
|
"dhcp": [
|
||||||
|
{
|
||||||
|
"macaddress": "9CADEF*"
|
||||||
|
}
|
||||||
|
],
|
||||||
"documentation": "https://www.home-assistant.io/integrations/obihai",
|
"documentation": "https://www.home-assistant.io/integrations/obihai",
|
||||||
"iot_class": "local_polling",
|
"iot_class": "local_polling",
|
||||||
"loggers": ["pyobihai"],
|
"loggers": ["pyobihai"],
|
||||||
|
@ -7,10 +7,19 @@
|
|||||||
"password": "[%key:common::config_flow::data::password%]",
|
"password": "[%key:common::config_flow::data::password%]",
|
||||||
"username": "[%key:common::config_flow::data::username%]"
|
"username": "[%key:common::config_flow::data::username%]"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"dhcp_confirm": {
|
||||||
|
"description": "Do you want to set up {host}?",
|
||||||
|
"data": {
|
||||||
|
"host": "[%key:common::config_flow::data::host%]",
|
||||||
|
"password": "[%key:common::config_flow::data::password%]",
|
||||||
|
"username": "[%key:common::config_flow::data::username%]"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"error": {
|
"error": {
|
||||||
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]"
|
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
|
||||||
|
"invalid_auth": "[%key:common::config_flow::error::invalid_auth%]"
|
||||||
},
|
},
|
||||||
"abort": {
|
"abort": {
|
||||||
"already_configured": "[%key:common::config_flow::abort::already_configured_device%]"
|
"already_configured": "[%key:common::config_flow::abort::already_configured_device%]"
|
||||||
|
@ -330,6 +330,10 @@ DHCP: list[dict[str, str | bool]] = [
|
|||||||
"domain": "nuki",
|
"domain": "nuki",
|
||||||
"hostname": "nuki_bridge_*",
|
"hostname": "nuki_bridge_*",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"domain": "obihai",
|
||||||
|
"macaddress": "9CADEF*",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"domain": "oncue",
|
"domain": "oncue",
|
||||||
"hostname": "kohlergen*",
|
"hostname": "kohlergen*",
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
"""Tests for the Obihai Integration."""
|
"""Tests for the Obihai Integration."""
|
||||||
|
|
||||||
|
from homeassistant.components import dhcp
|
||||||
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
|
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
|
||||||
|
|
||||||
USER_INPUT = {
|
USER_INPUT = {
|
||||||
@ -8,3 +8,27 @@ USER_INPUT = {
|
|||||||
CONF_PASSWORD: "admin",
|
CONF_PASSWORD: "admin",
|
||||||
CONF_USERNAME: "admin",
|
CONF_USERNAME: "admin",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DHCP_SERVICE_INFO = dhcp.DhcpServiceInfo(
|
||||||
|
hostname="obi200",
|
||||||
|
ip="192.168.1.100",
|
||||||
|
macaddress="9CADEF000000",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class MockPyObihai:
|
||||||
|
"""Mock PyObihai: Returns simulated PyObihai data."""
|
||||||
|
|
||||||
|
def get_device_mac(self):
|
||||||
|
"""Mock PyObihai.get_device_mac, return simulated MAC address."""
|
||||||
|
|
||||||
|
return DHCP_SERVICE_INFO.macaddress
|
||||||
|
|
||||||
|
|
||||||
|
def get_schema_suggestion(schema, key):
|
||||||
|
"""Get suggested value for key in voluptuous schema."""
|
||||||
|
for k in schema:
|
||||||
|
if k == key:
|
||||||
|
if k.description is None or "suggested_value" not in k.description:
|
||||||
|
return None
|
||||||
|
return k.description["suggested_value"]
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
"""Define test fixtures for Obihai."""
|
"""Define test fixtures for Obihai."""
|
||||||
|
|
||||||
from collections.abc import Generator
|
from collections.abc import Generator
|
||||||
|
from socket import gaierror
|
||||||
from unittest.mock import AsyncMock, patch
|
from unittest.mock import AsyncMock, patch
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
@ -9,7 +10,19 @@ import pytest
|
|||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def mock_setup_entry() -> Generator[AsyncMock, None, None]:
|
def mock_setup_entry() -> Generator[AsyncMock, None, None]:
|
||||||
"""Override async_setup_entry."""
|
"""Override async_setup_entry."""
|
||||||
|
|
||||||
with patch(
|
with patch(
|
||||||
"homeassistant.components.obihai.async_setup_entry", return_value=True
|
"homeassistant.components.obihai.async_setup_entry", return_value=True
|
||||||
) as mock_setup_entry:
|
) as mock_setup_entry:
|
||||||
yield mock_setup_entry
|
yield mock_setup_entry
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def mock_gaierror() -> Generator[AsyncMock, None, None]:
|
||||||
|
"""Override async_setup_entry."""
|
||||||
|
|
||||||
|
with patch(
|
||||||
|
"homeassistant.components.obihai.config_flow.gethostbyname",
|
||||||
|
side_effect=gaierror(),
|
||||||
|
) as mock_setup_entry:
|
||||||
|
yield mock_setup_entry
|
||||||
|
@ -1,14 +1,17 @@
|
|||||||
"""Test the Obihai config flow."""
|
"""Test the Obihai config flow."""
|
||||||
|
|
||||||
|
from collections.abc import Generator
|
||||||
from unittest.mock import AsyncMock, patch
|
from unittest.mock import AsyncMock, patch
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from homeassistant import config_entries
|
from homeassistant import config_entries
|
||||||
from homeassistant.components.obihai.const import DOMAIN
|
from homeassistant.components.obihai.const import DOMAIN
|
||||||
|
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.data_entry_flow import FlowResultType
|
from homeassistant.data_entry_flow import FlowResultType
|
||||||
|
|
||||||
from . import USER_INPUT
|
from . import DHCP_SERVICE_INFO, USER_INPUT, MockPyObihai, get_schema_suggestion
|
||||||
|
|
||||||
VALIDATE_AUTH_PATCH = "homeassistant.components.obihai.config_flow.validate_auth"
|
VALIDATE_AUTH_PATCH = "homeassistant.components.obihai.config_flow.validate_auth"
|
||||||
|
|
||||||
@ -25,7 +28,7 @@ async def test_user_form(hass: HomeAssistant, mock_setup_entry: AsyncMock) -> No
|
|||||||
assert result["step_id"] == "user"
|
assert result["step_id"] == "user"
|
||||||
assert result["errors"] == {}
|
assert result["errors"] == {}
|
||||||
|
|
||||||
with patch("pyobihai.PyObihai.check_account"):
|
with patch(VALIDATE_AUTH_PATCH, return_value=MockPyObihai()):
|
||||||
result = await hass.config_entries.flow.async_configure(
|
result = await hass.config_entries.flow.async_configure(
|
||||||
result["flow_id"],
|
result["flow_id"],
|
||||||
USER_INPUT,
|
USER_INPUT,
|
||||||
@ -40,7 +43,7 @@ async def test_user_form(hass: HomeAssistant, mock_setup_entry: AsyncMock) -> No
|
|||||||
|
|
||||||
|
|
||||||
async def test_auth_failure(hass: HomeAssistant) -> None:
|
async def test_auth_failure(hass: HomeAssistant) -> None:
|
||||||
"""Test we get the authentication error."""
|
"""Test we get the authentication error for user flow."""
|
||||||
result = await hass.config_entries.flow.async_init(
|
result = await hass.config_entries.flow.async_init(
|
||||||
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
||||||
)
|
)
|
||||||
@ -52,6 +55,24 @@ async def test_auth_failure(hass: HomeAssistant) -> None:
|
|||||||
)
|
)
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result["type"] == FlowResultType.FORM
|
||||||
|
assert result["step_id"] == "user"
|
||||||
|
assert result["errors"]["base"] == "invalid_auth"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_connect_failure(hass: HomeAssistant, mock_gaierror: Generator) -> None:
|
||||||
|
"""Test we get the connection error for user flow."""
|
||||||
|
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
||||||
|
)
|
||||||
|
|
||||||
|
result = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
USER_INPUT,
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert result["type"] == FlowResultType.FORM
|
assert result["type"] == FlowResultType.FORM
|
||||||
assert result["step_id"] == "user"
|
assert result["step_id"] == "user"
|
||||||
assert result["errors"]["base"] == "cannot_connect"
|
assert result["errors"]["base"] == "cannot_connect"
|
||||||
@ -59,7 +80,8 @@ async def test_auth_failure(hass: HomeAssistant) -> None:
|
|||||||
|
|
||||||
async def test_yaml_import(hass: HomeAssistant) -> None:
|
async def test_yaml_import(hass: HomeAssistant) -> None:
|
||||||
"""Test we get the YAML imported."""
|
"""Test we get the YAML imported."""
|
||||||
with patch(VALIDATE_AUTH_PATCH, return_value=True):
|
|
||||||
|
with patch(VALIDATE_AUTH_PATCH, return_value=MockPyObihai()):
|
||||||
result = await hass.config_entries.flow.async_init(
|
result = await hass.config_entries.flow.async_init(
|
||||||
DOMAIN,
|
DOMAIN,
|
||||||
context={"source": config_entries.SOURCE_IMPORT},
|
context={"source": config_entries.SOURCE_IMPORT},
|
||||||
@ -71,8 +93,9 @@ async def test_yaml_import(hass: HomeAssistant) -> None:
|
|||||||
assert "errors" not in result
|
assert "errors" not in result
|
||||||
|
|
||||||
|
|
||||||
async def test_yaml_import_fail(hass: HomeAssistant) -> None:
|
async def test_yaml_import_auth_fail(hass: HomeAssistant) -> None:
|
||||||
"""Test the YAML import fails."""
|
"""Test the YAML import fails."""
|
||||||
|
|
||||||
with patch(VALIDATE_AUTH_PATCH, return_value=False):
|
with patch(VALIDATE_AUTH_PATCH, return_value=False):
|
||||||
result = await hass.config_entries.flow.async_init(
|
result = await hass.config_entries.flow.async_init(
|
||||||
DOMAIN,
|
DOMAIN,
|
||||||
@ -81,6 +104,97 @@ async def test_yaml_import_fail(hass: HomeAssistant) -> None:
|
|||||||
)
|
)
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result["type"] == FlowResultType.ABORT
|
||||||
|
assert result["reason"] == "invalid_auth"
|
||||||
|
assert "errors" not in result
|
||||||
|
|
||||||
|
|
||||||
|
async def test_yaml_import_connect_fail(
|
||||||
|
hass: HomeAssistant, mock_gaierror: Generator
|
||||||
|
) -> None:
|
||||||
|
"""Test the YAML import fails with invalid host."""
|
||||||
|
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_IMPORT},
|
||||||
|
data=USER_INPUT,
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert result["type"] == FlowResultType.ABORT
|
assert result["type"] == FlowResultType.ABORT
|
||||||
assert result["reason"] == "cannot_connect"
|
assert result["reason"] == "cannot_connect"
|
||||||
assert "errors" not in result
|
assert "errors" not in result
|
||||||
|
|
||||||
|
|
||||||
|
async def test_dhcp_flow(hass: HomeAssistant) -> None:
|
||||||
|
"""Test that DHCP discovery works."""
|
||||||
|
|
||||||
|
with patch(
|
||||||
|
VALIDATE_AUTH_PATCH,
|
||||||
|
return_value=MockPyObihai(),
|
||||||
|
):
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
data=DHCP_SERVICE_INFO,
|
||||||
|
context={"source": config_entries.SOURCE_DHCP},
|
||||||
|
)
|
||||||
|
|
||||||
|
flows = hass.config_entries.flow.async_progress()
|
||||||
|
assert result["type"] == FlowResultType.FORM
|
||||||
|
assert len(flows) == 1
|
||||||
|
assert (
|
||||||
|
get_schema_suggestion(result["data_schema"].schema, CONF_USERNAME)
|
||||||
|
== USER_INPUT[CONF_USERNAME]
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
get_schema_suggestion(result["data_schema"].schema, CONF_PASSWORD)
|
||||||
|
== USER_INPUT[CONF_PASSWORD]
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
get_schema_suggestion(result["data_schema"].schema, CONF_HOST)
|
||||||
|
== DHCP_SERVICE_INFO.ip
|
||||||
|
)
|
||||||
|
assert flows[0].get("context", {}).get("source") == config_entries.SOURCE_DHCP
|
||||||
|
|
||||||
|
# Verify we get dropped into the normal user flow with non-default credentials
|
||||||
|
result = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"], user_input=USER_INPUT
|
||||||
|
)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert result["type"] == FlowResultType.CREATE_ENTRY
|
||||||
|
|
||||||
|
|
||||||
|
async def test_dhcp_flow_auth_failure(hass: HomeAssistant) -> None:
|
||||||
|
"""Test that DHCP fails if creds aren't default."""
|
||||||
|
|
||||||
|
with patch(
|
||||||
|
VALIDATE_AUTH_PATCH,
|
||||||
|
return_value=False,
|
||||||
|
):
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
data=DHCP_SERVICE_INFO,
|
||||||
|
context={"source": config_entries.SOURCE_DHCP},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["step_id"] == "dhcp_confirm"
|
||||||
|
assert get_schema_suggestion(result["data_schema"].schema, CONF_USERNAME) == ""
|
||||||
|
assert get_schema_suggestion(result["data_schema"].schema, CONF_PASSWORD) == ""
|
||||||
|
assert (
|
||||||
|
get_schema_suggestion(result["data_schema"].schema, CONF_HOST)
|
||||||
|
== DHCP_SERVICE_INFO.ip
|
||||||
|
)
|
||||||
|
|
||||||
|
# Verify we get dropped into the normal user flow with non-default credentials
|
||||||
|
result = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
user_input={
|
||||||
|
CONF_HOST: DHCP_SERVICE_INFO.ip,
|
||||||
|
CONF_USERNAME: "",
|
||||||
|
CONF_PASSWORD: "",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["errors"]["base"] == "invalid_auth"
|
||||||
|
assert result["step_id"] == "user"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user