mirror of
https://github.com/home-assistant/core.git
synced 2025-08-07 12:38:21 +00:00
Add zeroconf flow to Homee (#149820)
Co-authored-by: Joostlek <joostlek@outlook.com>
This commit is contained in:
parent
afffe0b08b
commit
cbf4130bff
@ -11,10 +11,16 @@ from pyHomee import (
|
|||||||
)
|
)
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigFlow, ConfigFlowResult
|
from homeassistant.config_entries import SOURCE_USER, ConfigFlow, ConfigFlowResult
|
||||||
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
|
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
|
||||||
|
from homeassistant.helpers.service_info.zeroconf import ZeroconfServiceInfo
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import (
|
||||||
|
DOMAIN,
|
||||||
|
RESULT_CANNOT_CONNECT,
|
||||||
|
RESULT_INVALID_AUTH,
|
||||||
|
RESULT_UNKNOWN_ERROR,
|
||||||
|
)
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -33,60 +39,137 @@ class HomeeConfigFlow(ConfigFlow, domain=DOMAIN):
|
|||||||
VERSION = 1
|
VERSION = 1
|
||||||
|
|
||||||
homee: Homee
|
homee: Homee
|
||||||
|
_host: str
|
||||||
|
_name: str
|
||||||
_reauth_host: str
|
_reauth_host: str
|
||||||
_reauth_username: str
|
_reauth_username: str
|
||||||
|
|
||||||
|
async def _connect_homee(self) -> dict[str, str]:
|
||||||
|
errors: dict[str, str] = {}
|
||||||
|
try:
|
||||||
|
await self.homee.get_access_token()
|
||||||
|
except HomeeConnectionFailedException:
|
||||||
|
errors["base"] = RESULT_CANNOT_CONNECT
|
||||||
|
except HomeeAuthenticationFailedException:
|
||||||
|
errors["base"] = RESULT_INVALID_AUTH
|
||||||
|
except Exception:
|
||||||
|
_LOGGER.exception("Unexpected exception")
|
||||||
|
errors["base"] = RESULT_UNKNOWN_ERROR
|
||||||
|
else:
|
||||||
|
_LOGGER.info("Got access token for homee")
|
||||||
|
self.hass.loop.create_task(self.homee.run())
|
||||||
|
_LOGGER.debug("Homee task created")
|
||||||
|
await self.homee.wait_until_connected()
|
||||||
|
_LOGGER.info("Homee connected")
|
||||||
|
self.homee.disconnect()
|
||||||
|
_LOGGER.debug("Homee disconnecting")
|
||||||
|
await self.homee.wait_until_disconnected()
|
||||||
|
_LOGGER.info("Homee config successfully tested")
|
||||||
|
|
||||||
|
await self.async_set_unique_id(
|
||||||
|
self.homee.settings.uid, raise_on_progress=self.source != SOURCE_USER
|
||||||
|
)
|
||||||
|
|
||||||
|
self._abort_if_unique_id_configured()
|
||||||
|
|
||||||
|
_LOGGER.info("Created new homee entry with ID %s", self.homee.settings.uid)
|
||||||
|
|
||||||
|
return errors
|
||||||
|
|
||||||
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
|
||||||
) -> ConfigFlowResult:
|
) -> ConfigFlowResult:
|
||||||
"""Handle the initial user step."""
|
"""Handle the initial user step."""
|
||||||
|
errors: dict[str, str] = {}
|
||||||
|
|
||||||
errors = {}
|
|
||||||
if user_input is not None:
|
if user_input is not None:
|
||||||
self.homee = Homee(
|
self.homee = Homee(
|
||||||
user_input[CONF_HOST],
|
user_input[CONF_HOST],
|
||||||
user_input[CONF_USERNAME],
|
user_input[CONF_USERNAME],
|
||||||
user_input[CONF_PASSWORD],
|
user_input[CONF_PASSWORD],
|
||||||
)
|
)
|
||||||
|
errors = await self._connect_homee()
|
||||||
|
|
||||||
try:
|
if not errors:
|
||||||
await self.homee.get_access_token()
|
|
||||||
except HomeeConnectionFailedException:
|
|
||||||
errors["base"] = "cannot_connect"
|
|
||||||
except HomeeAuthenticationFailedException:
|
|
||||||
errors["base"] = "invalid_auth"
|
|
||||||
except Exception:
|
|
||||||
_LOGGER.exception("Unexpected exception")
|
|
||||||
errors["base"] = "unknown"
|
|
||||||
else:
|
|
||||||
_LOGGER.info("Got access token for homee")
|
|
||||||
self.hass.loop.create_task(self.homee.run())
|
|
||||||
_LOGGER.debug("Homee task created")
|
|
||||||
await self.homee.wait_until_connected()
|
|
||||||
_LOGGER.info("Homee connected")
|
|
||||||
self.homee.disconnect()
|
|
||||||
_LOGGER.debug("Homee disconnecting")
|
|
||||||
await self.homee.wait_until_disconnected()
|
|
||||||
_LOGGER.info("Homee config successfully tested")
|
|
||||||
|
|
||||||
await self.async_set_unique_id(self.homee.settings.uid)
|
|
||||||
|
|
||||||
self._abort_if_unique_id_configured()
|
|
||||||
|
|
||||||
_LOGGER.info(
|
|
||||||
"Created new homee entry with ID %s", self.homee.settings.uid
|
|
||||||
)
|
|
||||||
|
|
||||||
return self.async_create_entry(
|
return self.async_create_entry(
|
||||||
title=f"{self.homee.settings.homee_name} ({self.homee.host})",
|
title=f"{self.homee.settings.homee_name} ({self.homee.host})",
|
||||||
data=user_input,
|
data=user_input,
|
||||||
)
|
)
|
||||||
|
|
||||||
return self.async_show_form(
|
return self.async_show_form(
|
||||||
step_id="user",
|
step_id="user",
|
||||||
data_schema=AUTH_SCHEMA,
|
data_schema=AUTH_SCHEMA,
|
||||||
errors=errors,
|
errors=errors,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
async def async_step_zeroconf(
|
||||||
|
self, discovery_info: ZeroconfServiceInfo
|
||||||
|
) -> ConfigFlowResult:
|
||||||
|
"""Handle zeroconf discovery."""
|
||||||
|
|
||||||
|
# Ensure that an IPv4 address is received
|
||||||
|
self._host = discovery_info.host
|
||||||
|
self._name = discovery_info.hostname[6:18]
|
||||||
|
if discovery_info.ip_address.version == 6:
|
||||||
|
return self.async_abort(reason="ipv6_address")
|
||||||
|
|
||||||
|
await self.async_set_unique_id(self._name)
|
||||||
|
self._abort_if_unique_id_configured(updates={CONF_HOST: self._host})
|
||||||
|
|
||||||
|
# Cause an auth-error to see if homee is reachable.
|
||||||
|
self.homee = Homee(
|
||||||
|
self._host,
|
||||||
|
"dummy_username",
|
||||||
|
"dummy_password",
|
||||||
|
)
|
||||||
|
errors = await self._connect_homee()
|
||||||
|
if errors["base"] != RESULT_INVALID_AUTH:
|
||||||
|
return self.async_abort(reason=RESULT_CANNOT_CONNECT)
|
||||||
|
|
||||||
|
self.context["title_placeholders"] = {"name": self._name, "host": self._host}
|
||||||
|
|
||||||
|
return await self.async_step_zeroconf_confirm()
|
||||||
|
|
||||||
|
async def async_step_zeroconf_confirm(
|
||||||
|
self, user_input: dict[str, Any] | None = None
|
||||||
|
) -> ConfigFlowResult:
|
||||||
|
"""Confirm the configuration of the device."""
|
||||||
|
|
||||||
|
errors: dict[str, str] = {}
|
||||||
|
if user_input is not None:
|
||||||
|
self.homee = Homee(
|
||||||
|
self._host,
|
||||||
|
user_input[CONF_USERNAME],
|
||||||
|
user_input[CONF_PASSWORD],
|
||||||
|
)
|
||||||
|
|
||||||
|
errors = await self._connect_homee()
|
||||||
|
|
||||||
|
if not errors:
|
||||||
|
return self.async_create_entry(
|
||||||
|
title=f"{self.homee.settings.homee_name} ({self.homee.host})",
|
||||||
|
data={
|
||||||
|
CONF_HOST: self._host,
|
||||||
|
CONF_USERNAME: user_input[CONF_USERNAME],
|
||||||
|
CONF_PASSWORD: user_input[CONF_PASSWORD],
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.async_show_form(
|
||||||
|
step_id="zeroconf_confirm",
|
||||||
|
data_schema=vol.Schema(
|
||||||
|
{
|
||||||
|
vol.Required(CONF_USERNAME): str,
|
||||||
|
vol.Required(CONF_PASSWORD): str,
|
||||||
|
}
|
||||||
|
),
|
||||||
|
errors=errors,
|
||||||
|
description_placeholders={
|
||||||
|
CONF_HOST: self._name,
|
||||||
|
},
|
||||||
|
last_step=True,
|
||||||
|
)
|
||||||
|
|
||||||
async def async_step_reauth(
|
async def async_step_reauth(
|
||||||
self, entry_data: Mapping[str, Any]
|
self, entry_data: Mapping[str, Any]
|
||||||
) -> ConfigFlowResult:
|
) -> ConfigFlowResult:
|
||||||
@ -108,12 +191,12 @@ class HomeeConfigFlow(ConfigFlow, domain=DOMAIN):
|
|||||||
try:
|
try:
|
||||||
await self.homee.get_access_token()
|
await self.homee.get_access_token()
|
||||||
except HomeeConnectionFailedException:
|
except HomeeConnectionFailedException:
|
||||||
errors["base"] = "cannot_connect"
|
errors["base"] = RESULT_CANNOT_CONNECT
|
||||||
except HomeeAuthenticationFailedException:
|
except HomeeAuthenticationFailedException:
|
||||||
errors["base"] = "invalid_auth"
|
errors["base"] = RESULT_INVALID_AUTH
|
||||||
except Exception:
|
except Exception:
|
||||||
_LOGGER.exception("Unexpected exception")
|
_LOGGER.exception("Unexpected exception")
|
||||||
errors["base"] = "unknown"
|
errors["base"] = RESULT_UNKNOWN_ERROR
|
||||||
else:
|
else:
|
||||||
self.hass.loop.create_task(self.homee.run())
|
self.hass.loop.create_task(self.homee.run())
|
||||||
await self.homee.wait_until_connected()
|
await self.homee.wait_until_connected()
|
||||||
@ -161,12 +244,12 @@ class HomeeConfigFlow(ConfigFlow, domain=DOMAIN):
|
|||||||
try:
|
try:
|
||||||
await self.homee.get_access_token()
|
await self.homee.get_access_token()
|
||||||
except HomeeConnectionFailedException:
|
except HomeeConnectionFailedException:
|
||||||
errors["base"] = "cannot_connect"
|
errors["base"] = RESULT_CANNOT_CONNECT
|
||||||
except HomeeAuthenticationFailedException:
|
except HomeeAuthenticationFailedException:
|
||||||
errors["base"] = "invalid_auth"
|
errors["base"] = RESULT_INVALID_AUTH
|
||||||
except Exception:
|
except Exception:
|
||||||
_LOGGER.exception("Unexpected exception")
|
_LOGGER.exception("Unexpected exception")
|
||||||
errors["base"] = "unknown"
|
errors["base"] = RESULT_UNKNOWN_ERROR
|
||||||
else:
|
else:
|
||||||
self.hass.loop.create_task(self.homee.run())
|
self.hass.loop.create_task(self.homee.run())
|
||||||
await self.homee.wait_until_connected()
|
await self.homee.wait_until_connected()
|
||||||
|
@ -20,6 +20,11 @@ from homeassistant.const import (
|
|||||||
# General
|
# General
|
||||||
DOMAIN = "homee"
|
DOMAIN = "homee"
|
||||||
|
|
||||||
|
# Error strings
|
||||||
|
RESULT_CANNOT_CONNECT = "cannot_connect"
|
||||||
|
RESULT_INVALID_AUTH = "invalid_auth"
|
||||||
|
RESULT_UNKNOWN_ERROR = "unknown"
|
||||||
|
|
||||||
# Sensor mappings
|
# Sensor mappings
|
||||||
HOMEE_UNIT_TO_HA_UNIT = {
|
HOMEE_UNIT_TO_HA_UNIT = {
|
||||||
"": None,
|
"": None,
|
||||||
|
@ -8,5 +8,11 @@
|
|||||||
"iot_class": "local_push",
|
"iot_class": "local_push",
|
||||||
"loggers": ["homee"],
|
"loggers": ["homee"],
|
||||||
"quality_scale": "silver",
|
"quality_scale": "silver",
|
||||||
"requirements": ["pyHomee==1.2.10"]
|
"requirements": ["pyHomee==1.2.10"],
|
||||||
|
"zeroconf": [
|
||||||
|
{
|
||||||
|
"type": "_ssh._tcp.local.",
|
||||||
|
"name": "homee-*"
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
@ -46,6 +46,17 @@
|
|||||||
"data_description": {
|
"data_description": {
|
||||||
"host": "[%key:component::homee::config::step::user::data_description::host%]"
|
"host": "[%key:component::homee::config::step::user::data_description::host%]"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"zeroconf_confirm": {
|
||||||
|
"title": "Configure discovered homee {host}",
|
||||||
|
"data": {
|
||||||
|
"username": "[%key:common::config_flow::data::username%]",
|
||||||
|
"password": "[%key:common::config_flow::data::password%]"
|
||||||
|
},
|
||||||
|
"data_description": {
|
||||||
|
"username": "[%key:component::homee::config::step::user::data_description::username%]",
|
||||||
|
"password": "[%key:component::homee::config::step::user::data_description::password%]"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
4
homeassistant/generated/zeroconf.py
generated
4
homeassistant/generated/zeroconf.py
generated
@ -890,6 +890,10 @@ ZEROCONF = {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
"_ssh._tcp.local.": [
|
"_ssh._tcp.local.": [
|
||||||
|
{
|
||||||
|
"domain": "homee",
|
||||||
|
"name": "homee-*",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"domain": "smappee",
|
"domain": "smappee",
|
||||||
"name": "smappee1*",
|
"name": "smappee1*",
|
||||||
|
@ -1,15 +1,23 @@
|
|||||||
"""Test the Homee config flow."""
|
"""Test the Homee config flow."""
|
||||||
|
|
||||||
|
from ipaddress import ip_address
|
||||||
from unittest.mock import AsyncMock
|
from unittest.mock import AsyncMock
|
||||||
|
|
||||||
from pyHomee import HomeeAuthFailedException, HomeeConnectionFailedException
|
from pyHomee import HomeeAuthFailedException, HomeeConnectionFailedException
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from homeassistant.components.homee.const import DOMAIN
|
from homeassistant import config_entries
|
||||||
|
from homeassistant.components.homee.const import (
|
||||||
|
DOMAIN,
|
||||||
|
RESULT_CANNOT_CONNECT,
|
||||||
|
RESULT_INVALID_AUTH,
|
||||||
|
RESULT_UNKNOWN_ERROR,
|
||||||
|
)
|
||||||
from homeassistant.config_entries import SOURCE_USER
|
from homeassistant.config_entries import SOURCE_USER
|
||||||
from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME
|
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 homeassistant.helpers.service_info.zeroconf import ZeroconfServiceInfo
|
||||||
|
|
||||||
from .conftest import (
|
from .conftest import (
|
||||||
HOMEE_ID,
|
HOMEE_ID,
|
||||||
@ -24,6 +32,24 @@ from .conftest import (
|
|||||||
|
|
||||||
from tests.common import MockConfigEntry
|
from tests.common import MockConfigEntry
|
||||||
|
|
||||||
|
PARAMETRIZED_ERRORS = (
|
||||||
|
("side_eff", "error"),
|
||||||
|
[
|
||||||
|
(
|
||||||
|
HomeeConnectionFailedException("connection timed out"),
|
||||||
|
{"base": RESULT_CANNOT_CONNECT},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
HomeeAuthFailedException("wrong username or password"),
|
||||||
|
{"base": RESULT_INVALID_AUTH},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
Exception,
|
||||||
|
{"base": RESULT_UNKNOWN_ERROR},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.usefixtures("mock_homee", "mock_config_entry", "mock_setup_entry")
|
@pytest.mark.usefixtures("mock_homee", "mock_config_entry", "mock_setup_entry")
|
||||||
async def test_config_flow(
|
async def test_config_flow(
|
||||||
@ -58,23 +84,7 @@ async def test_config_flow(
|
|||||||
assert result["result"].unique_id == HOMEE_ID
|
assert result["result"].unique_id == HOMEE_ID
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(*PARAMETRIZED_ERRORS)
|
||||||
("side_eff", "error"),
|
|
||||||
[
|
|
||||||
(
|
|
||||||
HomeeConnectionFailedException("connection timed out"),
|
|
||||||
{"base": "cannot_connect"},
|
|
||||||
),
|
|
||||||
(
|
|
||||||
HomeeAuthFailedException("wrong username or password"),
|
|
||||||
{"base": "invalid_auth"},
|
|
||||||
),
|
|
||||||
(
|
|
||||||
Exception,
|
|
||||||
{"base": "unknown"},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
async def test_config_flow_errors(
|
async def test_config_flow_errors(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
mock_homee: AsyncMock,
|
mock_homee: AsyncMock,
|
||||||
@ -140,6 +150,172 @@ async def test_flow_already_configured(
|
|||||||
assert result["reason"] == "already_configured"
|
assert result["reason"] == "already_configured"
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.usefixtures("mock_homee", "mock_config_entry")
|
||||||
|
async def test_zeroconf_success(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
mock_setup_entry: AsyncMock,
|
||||||
|
mock_homee: AsyncMock,
|
||||||
|
) -> None:
|
||||||
|
"""Test zeroconf discovery flow."""
|
||||||
|
mock_homee.get_access_token.side_effect = HomeeAuthFailedException(
|
||||||
|
"wrong username or password"
|
||||||
|
)
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
||||||
|
data=ZeroconfServiceInfo(
|
||||||
|
name=f"homee-{HOMEE_ID}._ssh._tcp.local.",
|
||||||
|
type="_ssh._tcp.local.",
|
||||||
|
hostname=f"homee-{HOMEE_ID}.local.",
|
||||||
|
ip_address=ip_address(HOMEE_IP),
|
||||||
|
ip_addresses=[ip_address(HOMEE_IP)],
|
||||||
|
port=22,
|
||||||
|
properties={},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["type"] is FlowResultType.FORM
|
||||||
|
assert result["step_id"] == "zeroconf_confirm"
|
||||||
|
assert result["handler"] == DOMAIN
|
||||||
|
mock_setup_entry.assert_not_called()
|
||||||
|
|
||||||
|
mock_homee.get_access_token.side_effect = None
|
||||||
|
result = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
user_input={
|
||||||
|
CONF_USERNAME: TESTUSER,
|
||||||
|
CONF_PASSWORD: TESTPASS,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["data"] == {
|
||||||
|
CONF_HOST: HOMEE_IP,
|
||||||
|
CONF_USERNAME: TESTUSER,
|
||||||
|
CONF_PASSWORD: TESTPASS,
|
||||||
|
}
|
||||||
|
|
||||||
|
mock_setup_entry.assert_called_once()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(*PARAMETRIZED_ERRORS)
|
||||||
|
async def test_zeroconf_confirm_errors(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
mock_homee: AsyncMock,
|
||||||
|
side_eff: Exception,
|
||||||
|
error: dict[str, str],
|
||||||
|
) -> None:
|
||||||
|
"""Test zeroconf discovery flow errors."""
|
||||||
|
mock_homee.get_access_token.side_effect = HomeeAuthFailedException(
|
||||||
|
"wrong username or password"
|
||||||
|
)
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
||||||
|
data=ZeroconfServiceInfo(
|
||||||
|
name=f"homee-{HOMEE_ID}._ssh._tcp.local.",
|
||||||
|
type="_ssh._tcp.local.",
|
||||||
|
hostname=f"homee-{HOMEE_ID}.local.",
|
||||||
|
ip_address=ip_address(HOMEE_IP),
|
||||||
|
ip_addresses=[ip_address(HOMEE_IP)],
|
||||||
|
port=22,
|
||||||
|
properties={},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
flow_id = result["flow_id"]
|
||||||
|
|
||||||
|
assert result["type"] is FlowResultType.FORM
|
||||||
|
assert result["step_id"] == "zeroconf_confirm"
|
||||||
|
assert result["handler"] == DOMAIN
|
||||||
|
|
||||||
|
mock_homee.get_access_token.side_effect = side_eff
|
||||||
|
result = await hass.config_entries.flow.async_configure(
|
||||||
|
result["flow_id"],
|
||||||
|
user_input={
|
||||||
|
CONF_USERNAME: TESTUSER,
|
||||||
|
CONF_PASSWORD: TESTPASS,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["type"] == FlowResultType.FORM
|
||||||
|
assert result["errors"] == error
|
||||||
|
|
||||||
|
mock_homee.get_access_token.side_effect = None
|
||||||
|
result = await hass.config_entries.flow.async_configure(
|
||||||
|
flow_id,
|
||||||
|
user_input={
|
||||||
|
CONF_USERNAME: TESTUSER,
|
||||||
|
CONF_PASSWORD: TESTPASS,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["type"] == FlowResultType.CREATE_ENTRY
|
||||||
|
|
||||||
|
|
||||||
|
async def test_zeroconf_already_configured(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
mock_config_entry: MockConfigEntry,
|
||||||
|
) -> None:
|
||||||
|
"""Test zeroconf discovery flow when already configured."""
|
||||||
|
mock_config_entry.add_to_hass(hass)
|
||||||
|
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
||||||
|
data=ZeroconfServiceInfo(
|
||||||
|
name=f"homee-{HOMEE_ID}._ssh._tcp.local.",
|
||||||
|
type="_ssh._tcp.local.",
|
||||||
|
hostname=f"homee-{HOMEE_ID}.local.",
|
||||||
|
ip_address=ip_address(HOMEE_IP),
|
||||||
|
ip_addresses=[ip_address(HOMEE_IP)],
|
||||||
|
port=22,
|
||||||
|
properties={},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["type"] is FlowResultType.ABORT
|
||||||
|
assert result["reason"] == "already_configured"
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
("side_eff", "ip", "reason"),
|
||||||
|
[
|
||||||
|
(
|
||||||
|
HomeeConnectionFailedException("connection timed out"),
|
||||||
|
HOMEE_IP,
|
||||||
|
RESULT_CANNOT_CONNECT,
|
||||||
|
),
|
||||||
|
(Exception, HOMEE_IP, RESULT_CANNOT_CONNECT),
|
||||||
|
(None, "2001:db8::1", "ipv6_address"),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
async def test_zeroconf_errors(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
mock_homee: AsyncMock,
|
||||||
|
side_eff: Exception,
|
||||||
|
ip: str,
|
||||||
|
reason: str,
|
||||||
|
) -> None:
|
||||||
|
"""Test zeroconf discovery flow with an IPv6 address."""
|
||||||
|
mock_homee.get_access_token.side_effect = side_eff
|
||||||
|
result = await hass.config_entries.flow.async_init(
|
||||||
|
DOMAIN,
|
||||||
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
||||||
|
data=ZeroconfServiceInfo(
|
||||||
|
name=f"homee-{HOMEE_ID}._ssh._tcp.local.",
|
||||||
|
type="_ssh._tcp.local.",
|
||||||
|
hostname=f"homee-{HOMEE_ID}.local.",
|
||||||
|
ip_address=ip_address(ip),
|
||||||
|
ip_addresses=[ip_address(ip)],
|
||||||
|
port=22,
|
||||||
|
properties={},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
assert result["type"] is FlowResultType.ABORT
|
||||||
|
assert result["reason"] == reason
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.usefixtures("mock_homee", "mock_setup_entry")
|
@pytest.mark.usefixtures("mock_homee", "mock_setup_entry")
|
||||||
async def test_reauth_success(
|
async def test_reauth_success(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
@ -171,23 +347,7 @@ async def test_reauth_success(
|
|||||||
assert mock_config_entry.data[CONF_PASSWORD] == NEW_TESTPASS
|
assert mock_config_entry.data[CONF_PASSWORD] == NEW_TESTPASS
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(*PARAMETRIZED_ERRORS)
|
||||||
("side_eff", "error"),
|
|
||||||
[
|
|
||||||
(
|
|
||||||
HomeeConnectionFailedException("connection timed out"),
|
|
||||||
{"base": "cannot_connect"},
|
|
||||||
),
|
|
||||||
(
|
|
||||||
HomeeAuthFailedException("wrong username or password"),
|
|
||||||
{"base": "invalid_auth"},
|
|
||||||
),
|
|
||||||
(
|
|
||||||
Exception,
|
|
||||||
{"base": "unknown"},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
async def test_reauth_errors(
|
async def test_reauth_errors(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
mock_config_entry: MockConfigEntry,
|
mock_config_entry: MockConfigEntry,
|
||||||
@ -296,23 +456,7 @@ async def test_reconfigure_success(
|
|||||||
assert mock_config_entry.data[CONF_PASSWORD] == TESTPASS
|
assert mock_config_entry.data[CONF_PASSWORD] == TESTPASS
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(*PARAMETRIZED_ERRORS)
|
||||||
("side_eff", "error"),
|
|
||||||
[
|
|
||||||
(
|
|
||||||
HomeeConnectionFailedException("connection timed out"),
|
|
||||||
{"base": "cannot_connect"},
|
|
||||||
),
|
|
||||||
(
|
|
||||||
HomeeAuthFailedException("wrong username or password"),
|
|
||||||
{"base": "invalid_auth"},
|
|
||||||
),
|
|
||||||
(
|
|
||||||
Exception,
|
|
||||||
{"base": "unknown"},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
async def test_reconfigure_errors(
|
async def test_reconfigure_errors(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
mock_config_entry: MockConfigEntry,
|
mock_config_entry: MockConfigEntry,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user