2020-08-20 15:30:41 +00:00
|
|
|
"""Test the Broadlink config flow."""
|
|
|
|
import errno
|
|
|
|
import socket
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import call, patch
|
2020-08-20 15:30:41 +00:00
|
|
|
|
|
|
|
import broadlink.exceptions as blke
|
|
|
|
import pytest
|
|
|
|
|
2021-10-07 10:58:00 +00:00
|
|
|
from homeassistant import config_entries
|
2021-11-19 11:29:20 +00:00
|
|
|
from homeassistant.components import dhcp
|
2020-08-20 15:30:41 +00:00
|
|
|
from homeassistant.components.broadlink.const import DOMAIN
|
2023-02-08 17:08:43 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2023-03-01 08:11:14 +00:00
|
|
|
from homeassistant.helpers import device_registry as dr
|
2020-08-20 15:30:41 +00:00
|
|
|
|
|
|
|
from . import get_device
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
DEVICE_HELLO = "homeassistant.components.broadlink.config_flow.blk.hello"
|
2020-09-12 01:00:28 +00:00
|
|
|
DEVICE_FACTORY = "homeassistant.components.broadlink.config_flow.blk.gendevice"
|
|
|
|
|
2020-08-20 15:30:41 +00:00
|
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def broadlink_setup_fixture():
|
|
|
|
"""Mock broadlink entry setup."""
|
|
|
|
with patch(
|
2020-09-12 01:00:28 +00:00
|
|
|
"homeassistant.components.broadlink.async_setup", return_value=True
|
|
|
|
), patch("homeassistant.components.broadlink.async_setup_entry", return_value=True):
|
2020-08-20 15:30:41 +00:00
|
|
|
yield
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_works(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test a config flow initiated by the user.
|
|
|
|
|
|
|
|
Best case scenario with no errors or locks.
|
|
|
|
"""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api) as mock_hello:
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "finish"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"name": device.name},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == device.name
|
|
|
|
assert result["data"] == device.get_entry_data()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
assert mock_hello.call_count == 1
|
2020-08-20 15:30:41 +00:00
|
|
|
assert mock_api.auth.call_count == 1
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_already_in_progress(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we do not accept more than one config flow per device."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=device.get_mock_api()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=device.get_mock_api()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_in_progress"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_mac_already_configured(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we do not accept more than one config entry per device.
|
|
|
|
|
|
|
|
We need to abort the flow and update the existing entry.
|
|
|
|
"""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
|
|
|
device.host = "192.168.1.64"
|
|
|
|
device.timeout = 20
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
assert dict(mock_entry.data) == device.get_entry_data()
|
|
|
|
assert mock_api.auth.call_count == 0
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_invalid_ip_address(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an invalid IP address in the user step."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError(errno.EINVAL, None)):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": "0.0.0.1"},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
assert result["errors"] == {"base": "invalid_host"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_invalid_hostname(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an invalid hostname in the user step."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError(socket.EAI_NONAME, None)):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": "pancakemaster.local"},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
assert result["errors"] == {"base": "invalid_host"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_device_not_found(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a device not found in the user step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=blke.NetworkTimeoutError()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_device_not_supported(hass: HomeAssistant) -> None:
|
2020-09-18 13:29:26 +00:00
|
|
|
"""Test we handle a device not supported in the user step."""
|
|
|
|
device = get_device("Kitchen")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-09-18 13:29:26 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "not_supported"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_network_unreachable(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a network unreachable in the user step."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError(errno.ENETUNREACH, None)):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": "192.168.1.32"},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_user_os_error(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an OS error in the user step."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": "192.168.1.32"},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
assert result["errors"] == {"base": "unknown"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_auth_authentication_error(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an authentication error in the auth step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = blke.AuthenticationError()
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "reset"
|
|
|
|
assert result["errors"] == {"base": "invalid_auth"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_auth_network_timeout(hass: HomeAssistant) -> None:
|
2020-09-26 15:46:02 +00:00
|
|
|
"""Test we handle a network timeout in the auth step."""
|
2020-08-20 15:30:41 +00:00
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
2020-09-26 15:46:02 +00:00
|
|
|
mock_api.auth.side_effect = blke.NetworkTimeoutError()
|
2020-08-20 15:30:41 +00:00
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "auth"
|
|
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_auth_firmware_error(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a firmware error in the auth step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = blke.BroadlinkException()
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "auth"
|
|
|
|
assert result["errors"] == {"base": "unknown"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_auth_network_unreachable(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a network unreachable in the auth step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = OSError(errno.ENETUNREACH, None)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "auth"
|
|
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_auth_os_error(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an OS error in the auth step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = OSError()
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "auth"
|
|
|
|
assert result["errors"] == {"base": "unknown"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_reset_works(hass: HomeAssistant) -> None:
|
2020-11-10 09:42:59 +00:00
|
|
|
"""Test we finish a config flow after a manual unlock."""
|
2020-08-20 15:30:41 +00:00
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = blke.AuthenticationError()
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=device.get_mock_api()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"name": device.name},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == device.name
|
|
|
|
assert result["data"] == device.get_entry_data()
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_unlock_works(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we finish a config flow with an unlock request."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.is_locked = True
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "unlock"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"unlock": True},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"name": device.name},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == device.name
|
|
|
|
assert result["data"] == device.get_entry_data()
|
|
|
|
|
|
|
|
assert mock_api.set_lock.call_args == call(False)
|
|
|
|
assert mock_api.set_lock.call_count == 1
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_unlock_network_timeout(hass: HomeAssistant) -> None:
|
2020-09-26 15:46:02 +00:00
|
|
|
"""Test we handle a network timeout in the unlock step."""
|
2020-08-20 15:30:41 +00:00
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.is_locked = True
|
2020-09-26 15:46:02 +00:00
|
|
|
mock_api.set_lock.side_effect = blke.NetworkTimeoutError()
|
2020-08-20 15:30:41 +00:00
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"unlock": True},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "unlock"
|
|
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_unlock_firmware_error(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a firmware error in the unlock step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.is_locked = True
|
|
|
|
mock_api.set_lock.side_effect = blke.BroadlinkException
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"unlock": True},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "unlock"
|
|
|
|
assert result["errors"] == {"base": "unknown"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_unlock_network_unreachable(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a network unreachable in the unlock step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.is_locked = True
|
|
|
|
mock_api.set_lock.side_effect = OSError(errno.ENETUNREACH, None)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"unlock": True},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "unlock"
|
|
|
|
assert result["errors"] == {"base": "cannot_connect"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_unlock_os_error(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an OS error in the unlock step."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.is_locked = True
|
|
|
|
mock_api.set_lock.side_effect = OSError()
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"unlock": True},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "unlock"
|
|
|
|
assert result["errors"] == {"base": "unknown"}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_do_not_unlock(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we do not unlock the device if the user does not want to."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.is_locked = True
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_USER}
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"unlock": False},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"name": device.name},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == device.name
|
|
|
|
assert result["data"] == device.get_entry_data()
|
|
|
|
|
|
|
|
assert mock_api.set_lock.call_count == 0
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_works(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test an import flow."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api) as mock_hello:
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": device.host},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "finish"
|
|
|
|
assert result["errors"] == {}
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"name": device.name},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "create_entry"
|
|
|
|
assert result["title"] == device.name
|
|
|
|
assert result["data"]["host"] == device.host
|
|
|
|
assert result["data"]["mac"] == device.mac
|
|
|
|
assert result["data"]["type"] == device.devtype
|
|
|
|
|
|
|
|
assert mock_api.auth.call_count == 1
|
2021-04-19 08:16:03 +00:00
|
|
|
assert mock_hello.call_count == 1
|
2020-08-20 15:30:41 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_already_in_progress(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we do not import more than one flow per device."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
data = {"host": device.host}
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=device.get_mock_api()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_IMPORT}, data=data
|
|
|
|
)
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=device.get_mock_api()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_IMPORT}, data=data
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_in_progress"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_host_already_configured(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we do not import a host that is already configured."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": device.host},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_mac_already_configured(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we do not import more than one config entry per device.
|
|
|
|
|
|
|
|
We need to abort the flow and update the existing entry.
|
|
|
|
"""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
device.host = "192.168.1.16"
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": device.host},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
assert mock_entry.data["host"] == device.host
|
|
|
|
assert mock_entry.data["mac"] == device.mac
|
|
|
|
assert mock_entry.data["type"] == device.devtype
|
|
|
|
assert mock_api.auth.call_count == 0
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_device_not_found(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a device not found in the import step."""
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=blke.NetworkTimeoutError()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": "192.168.1.32"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "cannot_connect"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_device_not_supported(hass: HomeAssistant) -> None:
|
2020-09-18 13:29:26 +00:00
|
|
|
"""Test we handle a device not supported in the import step."""
|
|
|
|
device = get_device("Kitchen")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2020-09-18 13:29:26 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": device.host},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "not_supported"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_invalid_ip_address(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an invalid IP address in the import step."""
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError(errno.EINVAL, None)):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": "0.0.0.1"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "invalid_host"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_invalid_hostname(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an invalid hostname in the import step."""
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError(socket.EAI_NONAME, None)):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": "hotdog.local"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "invalid_host"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_network_unreachable(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle a network unreachable in the import step."""
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError(errno.ENETUNREACH, None)):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": "192.168.1.64"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "cannot_connect"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_import_os_error(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we handle an OS error in the import step."""
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError()):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
|
|
data={"host": "192.168.1.64"},
|
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "unknown"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_reauth_works(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test a reauthentication flow."""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = blke.AuthenticationError()
|
|
|
|
data = {"name": device.name, **device.get_entry_data()}
|
|
|
|
|
2020-09-12 01:00:28 +00:00
|
|
|
with patch(DEVICE_FACTORY, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2021-04-25 09:27:40 +00:00
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_REAUTH}, data=data
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "reset"
|
|
|
|
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api) as mock_hello:
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
assert dict(mock_entry.data) == device.get_entry_data()
|
|
|
|
assert mock_api.auth.call_count == 1
|
2021-04-19 08:16:03 +00:00
|
|
|
assert mock_hello.call_count == 1
|
2020-08-20 15:30:41 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_reauth_invalid_host(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we do not accept an invalid host for reauthentication.
|
|
|
|
|
|
|
|
The MAC address cannot change.
|
|
|
|
"""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = blke.AuthenticationError()
|
|
|
|
data = {"name": device.name, **device.get_entry_data()}
|
|
|
|
|
2020-09-12 01:00:28 +00:00
|
|
|
with patch(DEVICE_FACTORY, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2021-04-25 09:27:40 +00:00
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_REAUTH}, data=data
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
device.mac = get_device("Office").mac
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api) as mock_hello:
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "user"
|
|
|
|
assert result["errors"] == {"base": "invalid_host"}
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
assert mock_hello.call_count == 1
|
2020-08-20 15:30:41 +00:00
|
|
|
assert mock_api.auth.call_count == 0
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_flow_reauth_valid_host(hass: HomeAssistant) -> None:
|
2020-08-20 15:30:41 +00:00
|
|
|
"""Test we accept a valid host for reauthentication.
|
|
|
|
|
|
|
|
The hostname/IP address may change. We need to update the entry.
|
|
|
|
"""
|
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
mock_api.auth.side_effect = blke.AuthenticationError()
|
|
|
|
data = {"name": device.name, **device.get_entry_data()}
|
|
|
|
|
2020-09-12 01:00:28 +00:00
|
|
|
with patch(DEVICE_FACTORY, return_value=mock_api):
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
2021-04-25 09:27:40 +00:00
|
|
|
DOMAIN, context={"source": config_entries.SOURCE_REAUTH}, data=data
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
device.host = "192.168.1.128"
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api) as mock_hello:
|
2020-08-20 15:30:41 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
2020-08-27 11:56:20 +00:00
|
|
|
result["flow_id"],
|
|
|
|
{"host": device.host, "timeout": device.timeout},
|
2020-08-20 15:30:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
assert mock_entry.data["host"] == device.host
|
2021-04-19 08:16:03 +00:00
|
|
|
assert mock_hello.call_count == 1
|
2020-08-20 15:30:41 +00:00
|
|
|
assert mock_api.auth.call_count == 1
|
2021-03-27 18:40:19 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_dhcp_can_finish(hass: HomeAssistant) -> None:
|
2021-03-27 18:40:19 +00:00
|
|
|
"""Test DHCP discovery flow can finish right away."""
|
|
|
|
|
|
|
|
device = get_device("Living Room")
|
|
|
|
device.host = "1.2.3.4"
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2021-03-27 18:40:19 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
2021-04-25 09:27:40 +00:00
|
|
|
context={"source": config_entries.SOURCE_DHCP},
|
2021-11-19 11:29:20 +00:00
|
|
|
data=dhcp.DhcpServiceInfo(
|
|
|
|
hostname="broadlink",
|
|
|
|
ip="1.2.3.4",
|
2023-03-01 08:11:14 +00:00
|
|
|
macaddress=dr.format_mac(device.mac),
|
2021-11-19 11:29:20 +00:00
|
|
|
),
|
2021-03-27 18:40:19 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "form"
|
|
|
|
assert result["step_id"] == "finish"
|
|
|
|
|
|
|
|
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"] == "Living Room"
|
|
|
|
assert result2["data"] == {
|
|
|
|
"host": "1.2.3.4",
|
|
|
|
"mac": "34ea34b43b5a",
|
|
|
|
"timeout": 10,
|
|
|
|
"type": 24374,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_dhcp_fails_to_connect(hass: HomeAssistant) -> None:
|
2021-03-27 18:40:19 +00:00
|
|
|
"""Test DHCP discovery flow that fails to connect."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=blke.NetworkTimeoutError()):
|
2021-03-27 18:40:19 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
2021-04-25 09:27:40 +00:00
|
|
|
context={"source": config_entries.SOURCE_DHCP},
|
2021-11-19 11:29:20 +00:00
|
|
|
data=dhcp.DhcpServiceInfo(
|
|
|
|
hostname="broadlink",
|
|
|
|
ip="1.2.3.4",
|
|
|
|
macaddress="34:ea:34:b4:3b:5a",
|
|
|
|
),
|
2021-03-27 18:40:19 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "cannot_connect"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_dhcp_unreachable(hass: HomeAssistant) -> None:
|
2021-03-27 18:40:19 +00:00
|
|
|
"""Test DHCP discovery flow that fails to connect."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError(errno.ENETUNREACH, None)):
|
2021-03-27 18:40:19 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
2021-04-25 09:27:40 +00:00
|
|
|
context={"source": config_entries.SOURCE_DHCP},
|
2021-11-19 11:29:20 +00:00
|
|
|
data=dhcp.DhcpServiceInfo(
|
|
|
|
hostname="broadlink",
|
|
|
|
ip="1.2.3.4",
|
|
|
|
macaddress="34:ea:34:b4:3b:5a",
|
|
|
|
),
|
2021-03-27 18:40:19 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "cannot_connect"
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_dhcp_connect_unknown_error(hass: HomeAssistant) -> None:
|
2021-04-19 04:12:27 +00:00
|
|
|
"""Test DHCP discovery flow that fails to connect with an OSError."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, side_effect=OSError()):
|
2021-03-27 18:40:19 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
2021-04-25 09:27:40 +00:00
|
|
|
context={"source": config_entries.SOURCE_DHCP},
|
2021-11-19 11:29:20 +00:00
|
|
|
data=dhcp.DhcpServiceInfo(
|
|
|
|
hostname="broadlink",
|
|
|
|
ip="1.2.3.4",
|
|
|
|
macaddress="34:ea:34:b4:3b:5a",
|
|
|
|
),
|
2021-03-27 18:40:19 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
2021-04-19 04:12:27 +00:00
|
|
|
assert result["reason"] == "unknown"
|
2021-03-27 18:40:19 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_dhcp_device_not_supported(hass: HomeAssistant) -> None:
|
2021-04-19 04:12:27 +00:00
|
|
|
"""Test DHCP discovery flow that fails because the device is not supported."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-04-19 04:12:27 +00:00
|
|
|
device = get_device("Kitchen")
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2021-03-27 18:40:19 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
2021-04-25 09:27:40 +00:00
|
|
|
context={"source": config_entries.SOURCE_DHCP},
|
2021-11-19 11:29:20 +00:00
|
|
|
data=dhcp.DhcpServiceInfo(
|
|
|
|
hostname="broadlink",
|
|
|
|
ip=device.host,
|
2023-03-01 08:11:14 +00:00
|
|
|
macaddress=dr.format_mac(device.mac),
|
2021-11-19 11:29:20 +00:00
|
|
|
),
|
2021-03-27 18:40:19 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
2021-04-19 04:12:27 +00:00
|
|
|
assert result["reason"] == "not_supported"
|
2021-03-27 18:40:19 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_dhcp_already_exists(hass: HomeAssistant) -> None:
|
2021-03-27 18:40:19 +00:00
|
|
|
"""Test DHCP discovery flow that fails to connect."""
|
2021-10-07 10:58:00 +00:00
|
|
|
|
2021-03-27 18:40:19 +00:00
|
|
|
device = get_device("Living Room")
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
device.host = "1.2.3.4"
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2021-03-27 18:40:19 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
2021-04-25 09:27:40 +00:00
|
|
|
context={"source": config_entries.SOURCE_DHCP},
|
2021-11-19 11:29:20 +00:00
|
|
|
data=dhcp.DhcpServiceInfo(
|
|
|
|
hostname="broadlink",
|
|
|
|
ip="1.2.3.4",
|
|
|
|
macaddress="34:ea:34:b4:3b:5a",
|
|
|
|
),
|
2021-03-27 18:40:19 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
2021-03-27 19:47:47 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_dhcp_updates_host(hass: HomeAssistant) -> None:
|
2021-03-27 19:47:47 +00:00
|
|
|
"""Test DHCP updates host."""
|
|
|
|
|
|
|
|
device = get_device("Living Room")
|
|
|
|
device.host = "1.2.3.4"
|
|
|
|
mock_entry = device.get_mock_entry()
|
|
|
|
mock_entry.add_to_hass(hass)
|
|
|
|
mock_api = device.get_mock_api()
|
|
|
|
|
2021-04-19 08:16:03 +00:00
|
|
|
with patch(DEVICE_HELLO, return_value=mock_api):
|
2021-03-27 19:47:47 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
2021-04-25 09:27:40 +00:00
|
|
|
context={"source": config_entries.SOURCE_DHCP},
|
2021-11-19 11:29:20 +00:00
|
|
|
data=dhcp.DhcpServiceInfo(
|
|
|
|
hostname="broadlink",
|
|
|
|
ip="4.5.6.7",
|
|
|
|
macaddress="34:ea:34:b4:3b:5a",
|
|
|
|
),
|
2021-03-27 19:47:47 +00:00
|
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert result["type"] == "abort"
|
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
assert mock_entry.data["host"] == "4.5.6.7"
|