2020-07-06 01:17:53 +00:00
|
|
|
"""Test the Bond config flow."""
|
2021-03-18 14:13:22 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-12-08 01:20:38 +00:00
|
|
|
from http import HTTPStatus
|
2021-03-18 14:13:22 +00:00
|
|
|
from typing import Any
|
2021-10-13 04:39:46 +00:00
|
|
|
from unittest.mock import MagicMock, Mock, patch
|
2020-07-08 01:36:22 +00:00
|
|
|
|
2020-07-23 01:22:25 +00:00
|
|
|
from aiohttp import ClientConnectionError, ClientResponseError
|
2020-07-06 01:17:53 +00:00
|
|
|
|
2021-10-07 10:58:00 +00:00
|
|
|
from homeassistant import config_entries, core
|
2021-11-02 17:27:31 +00:00
|
|
|
from homeassistant.components import zeroconf
|
2020-07-06 01:17:53 +00:00
|
|
|
from homeassistant.components.bond.const import DOMAIN
|
2021-10-13 04:39:46 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntryState
|
2020-07-06 01:17:53 +00:00
|
|
|
from homeassistant.const import CONF_ACCESS_TOKEN, CONF_HOST
|
|
|
|
|
2021-02-20 07:06:43 +00:00
|
|
|
from .common import (
|
|
|
|
patch_bond_bridge,
|
|
|
|
patch_bond_device,
|
|
|
|
patch_bond_device_ids,
|
|
|
|
patch_bond_device_properties,
|
2021-02-23 21:42:56 +00:00
|
|
|
patch_bond_token,
|
2021-02-20 07:06:43 +00:00
|
|
|
patch_bond_version,
|
|
|
|
)
|
2020-07-26 17:15:21 +00:00
|
|
|
|
2020-07-31 12:08:25 +00:00
|
|
|
from tests.common import MockConfigEntry
|
2020-07-06 01:17:53 +00:00
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_user_form(hass: core.HomeAssistant):
|
|
|
|
"""Test we get the user initiated form."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2020-07-06 01:17:53 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
2020-07-30 23:00:58 +00:00
|
|
|
with patch_bond_version(
|
|
|
|
return_value={"bondid": "test-bond-id"}
|
2021-02-20 07:06:43 +00:00
|
|
|
), patch_bond_device_ids(
|
|
|
|
return_value=["f6776c11", "f6776c12"]
|
2021-03-29 23:23:44 +00:00
|
|
|
), patch_bond_bridge(), patch_bond_device_properties(), patch_bond_device(), _patch_async_setup_entry() as mock_setup_entry:
|
2020-07-06 01:17:53 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
2020-07-26 17:15:21 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
2020-07-06 01:17:53 +00:00
|
|
|
)
|
2020-10-24 14:20:56 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-07-06 01:17:53 +00:00
|
|
|
|
|
|
|
assert result2["type"] == "create_entry"
|
2021-02-20 07:06:43 +00:00
|
|
|
assert result2["title"] == "bond-name"
|
|
|
|
assert result2["data"] == {
|
|
|
|
CONF_HOST: "some host",
|
|
|
|
CONF_ACCESS_TOKEN: "test-token",
|
|
|
|
}
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
|
|
|
|
|
|
async def test_user_form_with_non_bridge(hass: core.HomeAssistant):
|
|
|
|
"""Test setup a smart by bond fan."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-02-20 07:06:43 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
|
|
|
with patch_bond_version(
|
|
|
|
return_value={"bondid": "test-bond-id"}
|
|
|
|
), patch_bond_device_ids(
|
|
|
|
return_value=["f6776c11"]
|
|
|
|
), patch_bond_device_properties(), patch_bond_device(
|
|
|
|
return_value={
|
|
|
|
"name": "New Fan",
|
|
|
|
}
|
|
|
|
), patch_bond_bridge(
|
|
|
|
return_value={}
|
2021-03-29 23:23:44 +00:00
|
|
|
), _patch_async_setup_entry() as mock_setup_entry:
|
2021-02-20 07:06:43 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
{CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result2["type"] == "create_entry"
|
|
|
|
assert result2["title"] == "New Fan"
|
2020-07-06 01:17:53 +00:00
|
|
|
assert result2["data"] == {
|
2020-07-26 17:15:21 +00:00
|
|
|
CONF_HOST: "some host",
|
2020-07-06 01:17:53 +00:00
|
|
|
CONF_ACCESS_TOKEN: "test-token",
|
|
|
|
}
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_user_form_invalid_auth(hass: core.HomeAssistant):
|
2020-07-06 01:17:53 +00:00
|
|
|
"""Test we handle invalid auth."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2020-07-30 23:00:58 +00:00
|
|
|
with patch_bond_version(
|
|
|
|
return_value={"bond_id": "test-bond-id"}
|
2021-02-20 07:06:43 +00:00
|
|
|
), patch_bond_bridge(), patch_bond_device_ids(
|
2020-07-23 01:22:25 +00:00
|
|
|
side_effect=ClientResponseError(Mock(), Mock(), status=401),
|
2020-07-06 01:17:53 +00:00
|
|
|
):
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
2020-07-26 17:15:21 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
2020-07-06 01:17:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result2["type"] == "form"
|
|
|
|
assert result2["errors"] == {"base": "invalid_auth"}
|
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_user_form_cannot_connect(hass: core.HomeAssistant):
|
2020-07-06 01:17:53 +00:00
|
|
|
"""Test we handle cannot connect error."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2020-07-31 12:08:25 +00:00
|
|
|
with patch_bond_version(
|
|
|
|
side_effect=ClientConnectionError()
|
2021-02-20 07:06:43 +00:00
|
|
|
), patch_bond_bridge(), patch_bond_device_ids():
|
2020-07-06 01:17:53 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
2020-07-26 17:15:21 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
2020-07-06 01:17:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result2["type"] == "form"
|
|
|
|
assert result2["errors"] == {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
|
2020-08-08 02:22:13 +00:00
|
|
|
async def test_user_form_old_firmware(hass: core.HomeAssistant):
|
|
|
|
"""Test we handle unsupported old firmware."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
|
|
|
with patch_bond_version(
|
|
|
|
return_value={"no_bond_id": "present"}
|
2021-02-20 07:06:43 +00:00
|
|
|
), patch_bond_bridge(), patch_bond_device_ids():
|
2020-08-08 02:22:13 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
{CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result2["type"] == "form"
|
|
|
|
assert result2["errors"] == {"base": "old_firmware"}
|
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_user_form_unexpected_client_error(hass: core.HomeAssistant):
|
|
|
|
"""Test we handle unexpected client error gracefully."""
|
|
|
|
await _help_test_form_unexpected_error(
|
|
|
|
hass,
|
|
|
|
source=config_entries.SOURCE_USER,
|
|
|
|
user_input={CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
error=ClientResponseError(Mock(), Mock(), status=500),
|
2020-07-06 01:17:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_user_form_unexpected_error(hass: core.HomeAssistant):
|
|
|
|
"""Test we handle unexpected error gracefully."""
|
|
|
|
await _help_test_form_unexpected_error(
|
|
|
|
hass,
|
|
|
|
source=config_entries.SOURCE_USER,
|
|
|
|
user_input={CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
error=Exception(),
|
|
|
|
)
|
2020-07-31 12:08:25 +00:00
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_user_form_one_entry_per_device_allowed(hass: core.HomeAssistant):
|
2020-07-31 12:08:25 +00:00
|
|
|
"""Test that only one entry allowed per unique ID reported by Bond hub device."""
|
|
|
|
MockConfigEntry(
|
|
|
|
domain=DOMAIN,
|
|
|
|
unique_id="already-registered-bond-id",
|
|
|
|
data={CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
).add_to_hass(hass)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
|
|
|
with patch_bond_version(
|
|
|
|
return_value={"bondid": "already-registered-bond-id"}
|
2021-03-29 23:23:44 +00:00
|
|
|
), patch_bond_bridge(), patch_bond_device_ids(), _patch_async_setup_entry() as mock_setup_entry:
|
2020-07-31 12:08:25 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
{CONF_HOST: "some host", CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result2["type"] == "abort"
|
|
|
|
assert result2["reason"] == "already_configured"
|
|
|
|
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
2020-08-01 16:18:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_form(hass: core.HomeAssistant):
|
|
|
|
"""Test we get the discovery form."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-11-25 18:32:26 +00:00
|
|
|
with patch_bond_version(), patch_bond_token():
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
|
|
|
data=zeroconf.ZeroconfServiceInfo(
|
|
|
|
host="test-host",
|
|
|
|
hostname="mock_hostname",
|
|
|
|
name="test-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
|
|
|
),
|
|
|
|
)
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["errors"] == {}
|
2020-08-01 16:18:40 +00:00
|
|
|
|
|
|
|
with patch_bond_version(
|
|
|
|
return_value={"bondid": "test-bond-id"}
|
2021-03-29 23:23:44 +00:00
|
|
|
), patch_bond_bridge(), patch_bond_device_ids(), _patch_async_setup_entry() as mock_setup_entry:
|
2020-08-01 16:18:40 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{CONF_ACCESS_TOKEN: "test-token"},
|
2020-08-01 16:18:40 +00:00
|
|
|
)
|
2020-10-24 14:20:56 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-08-01 16:18:40 +00:00
|
|
|
|
|
|
|
assert result2["type"] == "create_entry"
|
2021-02-20 07:06:43 +00:00
|
|
|
assert result2["title"] == "bond-name"
|
2020-08-01 16:18:40 +00:00
|
|
|
assert result2["data"] == {
|
|
|
|
CONF_HOST: "test-host",
|
|
|
|
CONF_ACCESS_TOKEN: "test-token",
|
|
|
|
}
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
|
|
|
2021-02-23 21:42:56 +00:00
|
|
|
async def test_zeroconf_form_token_unavailable(hass: core.HomeAssistant):
|
|
|
|
"""Test we get the discovery form and we handle the token being unavailable."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-02-23 21:42:56 +00:00
|
|
|
with patch_bond_version(), patch_bond_token():
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
2021-11-10 14:26:47 +00:00
|
|
|
data=zeroconf.ZeroconfServiceInfo(
|
2021-11-23 15:32:58 +00:00
|
|
|
host="test-host",
|
|
|
|
hostname="mock_hostname",
|
|
|
|
name="test-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
2021-11-02 17:27:31 +00:00
|
|
|
),
|
2021-02-23 21:42:56 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
2021-03-29 23:23:44 +00:00
|
|
|
with patch_bond_version(), patch_bond_bridge(), patch_bond_device_ids(), _patch_async_setup_entry() as mock_setup_entry:
|
2021-02-23 21:42:56 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
{CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result2["type"] == "create_entry"
|
|
|
|
assert result2["title"] == "bond-name"
|
|
|
|
assert result2["data"] == {
|
|
|
|
CONF_HOST: "test-host",
|
|
|
|
CONF_ACCESS_TOKEN: "test-token",
|
|
|
|
}
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_form_with_token_available(hass: core.HomeAssistant):
|
|
|
|
"""Test we get the discovery form when we can get the token."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-02-23 21:42:56 +00:00
|
|
|
with patch_bond_version(return_value={"bondid": "test-bond-id"}), patch_bond_token(
|
|
|
|
return_value={"token": "discovered-token"}
|
|
|
|
), patch_bond_bridge(
|
|
|
|
return_value={"name": "discovered-name"}
|
|
|
|
), patch_bond_device_ids():
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
2021-11-10 14:26:47 +00:00
|
|
|
data=zeroconf.ZeroconfServiceInfo(
|
2021-11-23 15:32:58 +00:00
|
|
|
host="test-host",
|
|
|
|
hostname="mock_hostname",
|
|
|
|
name="test-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
2021-11-02 17:27:31 +00:00
|
|
|
),
|
2021-02-23 21:42:56 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
2021-03-29 23:23:44 +00:00
|
|
|
with _patch_async_setup_entry() as mock_setup_entry:
|
2021-02-23 21:42:56 +00:00
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
{},
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result2["type"] == "create_entry"
|
|
|
|
assert result2["title"] == "discovered-name"
|
|
|
|
assert result2["data"] == {
|
|
|
|
CONF_HOST: "test-host",
|
|
|
|
CONF_ACCESS_TOKEN: "discovered-token",
|
|
|
|
}
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
|
|
|
2021-12-08 01:20:38 +00:00
|
|
|
async def test_zeroconf_form_with_token_available_name_unavailable(
|
|
|
|
hass: core.HomeAssistant,
|
|
|
|
):
|
|
|
|
"""Test we get the discovery form when we can get the token but the name is unavailable."""
|
|
|
|
|
|
|
|
with patch_bond_version(
|
|
|
|
side_effect=ClientResponseError(Mock(), (), status=HTTPStatus.BAD_REQUEST)
|
|
|
|
), patch_bond_token(return_value={"token": "discovered-token"}):
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
|
|
|
data=zeroconf.ZeroconfServiceInfo(
|
|
|
|
host="test-host",
|
|
|
|
hostname="mock_hostname",
|
|
|
|
name="test-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
|
|
|
),
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
|
|
|
with _patch_async_setup_entry() as mock_setup_entry:
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
{},
|
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result2["type"] == "create_entry"
|
|
|
|
assert result2["title"] == "test-bond-id"
|
|
|
|
assert result2["data"] == {
|
|
|
|
CONF_HOST: "test-host",
|
|
|
|
CONF_ACCESS_TOKEN: "discovered-token",
|
|
|
|
}
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_zeroconf_already_configured(hass: core.HomeAssistant):
|
|
|
|
"""Test starting a flow from discovery when already configured."""
|
|
|
|
|
|
|
|
entry = MockConfigEntry(
|
|
|
|
domain=DOMAIN,
|
|
|
|
unique_id="already-registered-bond-id",
|
|
|
|
data={CONF_HOST: "stored-host", CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
2021-03-29 23:23:44 +00:00
|
|
|
with _patch_async_setup_entry() as mock_setup_entry:
|
2020-08-01 16:18:40 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
2021-11-10 14:26:47 +00:00
|
|
|
data=zeroconf.ZeroconfServiceInfo(
|
2021-11-02 17:27:31 +00:00
|
|
|
host="updated-host",
|
2021-11-23 15:32:58 +00:00
|
|
|
hostname="mock_hostname",
|
|
|
|
name="already-registered-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
2021-11-02 17:27:31 +00:00
|
|
|
),
|
2020-08-01 16:18:40 +00:00
|
|
|
)
|
2021-11-25 18:32:26 +00:00
|
|
|
await hass.async_block_till_done()
|
2020-08-01 16:18:40 +00:00
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
assert entry.data["host"] == "updated-host"
|
2021-11-25 18:32:26 +00:00
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
2020-08-01 16:18:40 +00:00
|
|
|
|
|
|
|
|
2021-10-13 04:39:46 +00:00
|
|
|
async def test_zeroconf_already_configured_refresh_token(hass: core.HomeAssistant):
|
|
|
|
"""Test starting a flow from zeroconf when already configured and the token is out of date."""
|
|
|
|
entry2 = MockConfigEntry(
|
|
|
|
domain=DOMAIN,
|
|
|
|
unique_id="not-the-same-bond-id",
|
|
|
|
data={CONF_HOST: "stored-host", CONF_ACCESS_TOKEN: "correct-token"},
|
|
|
|
)
|
|
|
|
entry2.add_to_hass(hass)
|
|
|
|
entry = MockConfigEntry(
|
|
|
|
domain=DOMAIN,
|
|
|
|
unique_id="already-registered-bond-id",
|
|
|
|
data={CONF_HOST: "stored-host", CONF_ACCESS_TOKEN: "incorrect-token"},
|
|
|
|
)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
with patch_bond_version(
|
|
|
|
side_effect=ClientResponseError(MagicMock(), MagicMock(), status=401)
|
|
|
|
):
|
|
|
|
await hass.config_entries.async_setup(entry.entry_id)
|
|
|
|
assert entry.state is ConfigEntryState.SETUP_ERROR
|
|
|
|
|
|
|
|
with _patch_async_setup_entry() as mock_setup_entry, patch_bond_token(
|
|
|
|
return_value={"token": "discovered-token"}
|
|
|
|
):
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
2021-11-10 14:26:47 +00:00
|
|
|
data=zeroconf.ZeroconfServiceInfo(
|
2021-11-02 17:27:31 +00:00
|
|
|
host="updated-host",
|
2021-11-23 15:32:58 +00:00
|
|
|
hostname="mock_hostname",
|
|
|
|
name="already-registered-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
2021-11-02 17:27:31 +00:00
|
|
|
),
|
2021-10-13 04:39:46 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
assert entry.data["host"] == "updated-host"
|
|
|
|
assert entry.data[CONF_ACCESS_TOKEN] == "discovered-token"
|
|
|
|
# entry2 should not get changed
|
|
|
|
assert entry2.data[CONF_ACCESS_TOKEN] == "correct-token"
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
|
|
|
|
|
|
|
2021-10-17 17:50:13 +00:00
|
|
|
async def test_zeroconf_already_configured_no_reload_same_host(
|
|
|
|
hass: core.HomeAssistant,
|
|
|
|
):
|
|
|
|
"""Test starting a flow from zeroconf when already configured does not reload if the host is the same."""
|
|
|
|
entry = MockConfigEntry(
|
|
|
|
domain=DOMAIN,
|
|
|
|
unique_id="already-registered-bond-id",
|
|
|
|
data={CONF_HOST: "stored-host", CONF_ACCESS_TOKEN: "correct-token"},
|
|
|
|
)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
with _patch_async_setup_entry() as mock_setup_entry, patch_bond_token(
|
|
|
|
return_value={"token": "correct-token"}
|
|
|
|
):
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_ZEROCONF},
|
2021-11-10 14:26:47 +00:00
|
|
|
data=zeroconf.ZeroconfServiceInfo(
|
2021-11-02 17:27:31 +00:00
|
|
|
host="stored-host",
|
2021-11-23 15:32:58 +00:00
|
|
|
hostname="mock_hostname",
|
|
|
|
name="already-registered-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
2021-11-02 17:27:31 +00:00
|
|
|
),
|
2021-10-17 17:50:13 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
|
|
|
|
|
|
|
2020-08-01 16:18:40 +00:00
|
|
|
async def test_zeroconf_form_unexpected_error(hass: core.HomeAssistant):
|
|
|
|
"""Test we handle unexpected error gracefully."""
|
|
|
|
await _help_test_form_unexpected_error(
|
|
|
|
hass,
|
|
|
|
source=config_entries.SOURCE_ZEROCONF,
|
2021-11-10 14:26:47 +00:00
|
|
|
initial_input=zeroconf.ZeroconfServiceInfo(
|
2021-11-02 17:27:31 +00:00
|
|
|
host="test-host",
|
2021-11-23 15:32:58 +00:00
|
|
|
hostname="mock_hostname",
|
|
|
|
name="test-bond-id.some-other-tail-info",
|
|
|
|
port=None,
|
|
|
|
properties={},
|
|
|
|
type="mock_type",
|
2021-11-02 17:27:31 +00:00
|
|
|
),
|
2020-08-01 16:18:40 +00:00
|
|
|
user_input={CONF_ACCESS_TOKEN: "test-token"},
|
|
|
|
error=Exception(),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def _help_test_form_unexpected_error(
|
|
|
|
hass: core.HomeAssistant,
|
|
|
|
*,
|
|
|
|
source: str,
|
2021-03-18 14:13:22 +00:00
|
|
|
initial_input: dict[str, Any] = None,
|
|
|
|
user_input: dict[str, Any],
|
2020-08-01 16:18:40 +00:00
|
|
|
error: Exception,
|
|
|
|
):
|
|
|
|
"""Test we handle unexpected error gracefully."""
|
2021-11-25 18:32:26 +00:00
|
|
|
with patch_bond_token():
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": source}, data=initial_input
|
|
|
|
)
|
2020-08-01 16:18:40 +00:00
|
|
|
|
|
|
|
with patch_bond_version(
|
|
|
|
return_value={"bond_id": "test-bond-id"}
|
|
|
|
), patch_bond_device_ids(side_effect=error):
|
|
|
|
result2 = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result2["type"] == "form"
|
|
|
|
assert result2["errors"] == {"base": "unknown"}
|
|
|
|
|
|
|
|
|
|
|
|
def _patch_async_setup_entry():
|
2020-08-27 11:56:20 +00:00
|
|
|
return patch(
|
|
|
|
"homeassistant.components.bond.async_setup_entry",
|
|
|
|
return_value=True,
|
|
|
|
)
|