Add config flow to Switchbot (#50653)
Co-authored-by: Daniel Hjelseth Høyer <mail@dahoiv.net> Co-authored-by: J. Nick Koston <nick@koston.org>
This commit is contained in:
parent
f31b9eae61
commit
3ce8109e5e
16 changed files with 978 additions and 48 deletions
64
tests/components/switchbot/__init__.py
Normal file
64
tests/components/switchbot/__init__.py
Normal file
|
@ -0,0 +1,64 @@
|
|||
"""Tests for the switchbot integration."""
|
||||
from unittest.mock import patch
|
||||
|
||||
from homeassistant.const import CONF_MAC, CONF_NAME, CONF_PASSWORD, CONF_SENSOR_TYPE
|
||||
from homeassistant.core import HomeAssistant
|
||||
|
||||
from tests.common import MockConfigEntry
|
||||
|
||||
DOMAIN = "switchbot"
|
||||
|
||||
ENTRY_CONFIG = {
|
||||
CONF_NAME: "test-name",
|
||||
CONF_PASSWORD: "test-password",
|
||||
CONF_MAC: "e7:89:43:99:99:99",
|
||||
}
|
||||
|
||||
USER_INPUT = {
|
||||
CONF_NAME: "test-name",
|
||||
CONF_PASSWORD: "test-password",
|
||||
CONF_MAC: "e7:89:43:99:99:99",
|
||||
}
|
||||
|
||||
USER_INPUT_UNSUPPORTED_DEVICE = {
|
||||
CONF_NAME: "test-name",
|
||||
CONF_PASSWORD: "test-password",
|
||||
CONF_MAC: "test",
|
||||
}
|
||||
|
||||
USER_INPUT_INVALID = {
|
||||
CONF_NAME: "test-name",
|
||||
CONF_PASSWORD: "test-password",
|
||||
CONF_MAC: "invalid-mac",
|
||||
}
|
||||
|
||||
YAML_CONFIG = {
|
||||
CONF_NAME: "test-name",
|
||||
CONF_PASSWORD: "test-password",
|
||||
CONF_MAC: "e7:89:43:99:99:99",
|
||||
CONF_SENSOR_TYPE: "bot",
|
||||
}
|
||||
|
||||
|
||||
def _patch_async_setup_entry(return_value=True):
|
||||
return patch(
|
||||
"homeassistant.components.switchbot.async_setup_entry",
|
||||
return_value=return_value,
|
||||
)
|
||||
|
||||
|
||||
async def init_integration(
|
||||
hass: HomeAssistant,
|
||||
*,
|
||||
data: dict = ENTRY_CONFIG,
|
||||
skip_entry_setup: bool = False,
|
||||
) -> MockConfigEntry:
|
||||
"""Set up the Switchbot integration in Home Assistant."""
|
||||
entry = MockConfigEntry(domain=DOMAIN, data=data)
|
||||
entry.add_to_hass(hass)
|
||||
|
||||
if not skip_entry_setup:
|
||||
await hass.config_entries.async_setup(entry.entry_id)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
return entry
|
78
tests/components/switchbot/conftest.py
Normal file
78
tests/components/switchbot/conftest.py
Normal file
|
@ -0,0 +1,78 @@
|
|||
"""Define fixtures available for all tests."""
|
||||
import sys
|
||||
from unittest.mock import MagicMock, patch
|
||||
|
||||
from pytest import fixture
|
||||
|
||||
|
||||
class MocGetSwitchbotDevices:
|
||||
"""Scan for all Switchbot devices and return by type."""
|
||||
|
||||
def __init__(self, interface=None) -> None:
|
||||
"""Get switchbot devices class constructor."""
|
||||
self._interface = interface
|
||||
self._all_services_data = {
|
||||
"mac_address": "e7:89:43:99:99:99",
|
||||
"Flags": "06",
|
||||
"Manufacturer": "5900e78943d9fe7c",
|
||||
"Complete 128b Services": "cba20d00-224d-11e6-9fb8-0002a5d5c51b",
|
||||
"data": {
|
||||
"switchMode": "true",
|
||||
"isOn": "true",
|
||||
"battery": 91,
|
||||
"rssi": -71,
|
||||
},
|
||||
"model": "H",
|
||||
"modelName": "WoHand",
|
||||
}
|
||||
self._unsupported_device = {
|
||||
"mac_address": "test",
|
||||
"Flags": "06",
|
||||
"Manufacturer": "5900e78943d9fe7c",
|
||||
"Complete 128b Services": "cba20d00-224d-11e6-9fb8-0002a5d5c51b",
|
||||
"data": {
|
||||
"switchMode": "true",
|
||||
"isOn": "true",
|
||||
"battery": 91,
|
||||
"rssi": -71,
|
||||
},
|
||||
"model": "HoN",
|
||||
"modelName": "WoOther",
|
||||
}
|
||||
|
||||
def discover(self, retry=0, scan_timeout=0):
|
||||
"""Mock discover."""
|
||||
return self._all_services_data
|
||||
|
||||
def get_device_data(self, mac=None):
|
||||
"""Return data for specific device."""
|
||||
if mac == "e7:89:43:99:99:99":
|
||||
return self._all_services_data
|
||||
if mac == "test":
|
||||
return self._unsupported_device
|
||||
|
||||
return None
|
||||
|
||||
|
||||
class MocNotConnectedError(Exception):
|
||||
"""Mock exception."""
|
||||
|
||||
|
||||
module = type(sys)("switchbot")
|
||||
module.GetSwitchbotDevices = MocGetSwitchbotDevices
|
||||
module.NotConnectedError = MocNotConnectedError
|
||||
sys.modules["switchbot"] = module
|
||||
|
||||
|
||||
@fixture
|
||||
def switchbot_config_flow(hass):
|
||||
"""Mock the bluepy api for easier config flow testing."""
|
||||
with patch.object(MocGetSwitchbotDevices, "discover", return_value=True), patch(
|
||||
"homeassistant.components.switchbot.config_flow.GetSwitchbotDevices"
|
||||
) as mock_switchbot:
|
||||
instance = mock_switchbot.return_value
|
||||
|
||||
instance.discover = MagicMock(return_value=True)
|
||||
instance.get_device_data = MagicMock(return_value=True)
|
||||
|
||||
yield mock_switchbot
|
226
tests/components/switchbot/test_config_flow.py
Normal file
226
tests/components/switchbot/test_config_flow.py
Normal file
|
@ -0,0 +1,226 @@
|
|||
"""Test the switchbot config flow."""
|
||||
|
||||
from unittest.mock import patch
|
||||
|
||||
from homeassistant.components.switchbot.config_flow import NotConnectedError
|
||||
from homeassistant.config_entries import SOURCE_IMPORT, SOURCE_USER
|
||||
from homeassistant.const import CONF_MAC, CONF_NAME, CONF_PASSWORD, CONF_SENSOR_TYPE
|
||||
from homeassistant.data_entry_flow import (
|
||||
RESULT_TYPE_ABORT,
|
||||
RESULT_TYPE_CREATE_ENTRY,
|
||||
RESULT_TYPE_FORM,
|
||||
)
|
||||
from homeassistant.setup import async_setup_component
|
||||
|
||||
from . import (
|
||||
USER_INPUT,
|
||||
USER_INPUT_INVALID,
|
||||
USER_INPUT_UNSUPPORTED_DEVICE,
|
||||
YAML_CONFIG,
|
||||
_patch_async_setup_entry,
|
||||
init_integration,
|
||||
)
|
||||
|
||||
DOMAIN = "switchbot"
|
||||
|
||||
|
||||
async def test_user_form_valid_mac(hass):
|
||||
"""Test the user initiated form with password and valid mac."""
|
||||
await async_setup_component(hass, "persistent_notification", {})
|
||||
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_USER}
|
||||
)
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["step_id"] == "user"
|
||||
assert result["errors"] == {}
|
||||
|
||||
with _patch_async_setup_entry() as mock_setup_entry:
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
USER_INPUT,
|
||||
)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert result["type"] == RESULT_TYPE_CREATE_ENTRY
|
||||
assert result["title"] == "test-name"
|
||||
assert result["data"] == {
|
||||
CONF_MAC: "e7:89:43:99:99:99",
|
||||
CONF_NAME: "test-name",
|
||||
CONF_PASSWORD: "test-password",
|
||||
CONF_SENSOR_TYPE: "bot",
|
||||
}
|
||||
|
||||
assert len(mock_setup_entry.mock_calls) == 1
|
||||
|
||||
# test duplicate device creation fails.
|
||||
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_USER}
|
||||
)
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["step_id"] == "user"
|
||||
assert result["errors"] == {}
|
||||
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
USER_INPUT,
|
||||
)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert result["type"] == RESULT_TYPE_ABORT
|
||||
assert result["reason"] == "already_configured_device"
|
||||
|
||||
|
||||
async def test_user_form_unsupported_device(hass):
|
||||
"""Test the user initiated form for unsupported device type."""
|
||||
await async_setup_component(hass, "persistent_notification", {})
|
||||
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_USER}
|
||||
)
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["step_id"] == "user"
|
||||
assert result["errors"] == {}
|
||||
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
USER_INPUT_UNSUPPORTED_DEVICE,
|
||||
)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert result["type"] == RESULT_TYPE_ABORT
|
||||
assert result["reason"] == "switchbot_unsupported_type"
|
||||
|
||||
|
||||
async def test_user_form_invalid_device(hass):
|
||||
"""Test the user initiated form for invalid device type."""
|
||||
await async_setup_component(hass, "persistent_notification", {})
|
||||
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_USER}
|
||||
)
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["step_id"] == "user"
|
||||
assert result["errors"] == {}
|
||||
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
USER_INPUT_INVALID,
|
||||
)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["errors"] == {"base": "cannot_connect"}
|
||||
|
||||
|
||||
async def test_async_step_import(hass):
|
||||
"""Test the config import flow."""
|
||||
await async_setup_component(hass, "persistent_notification", {})
|
||||
|
||||
with _patch_async_setup_entry() as mock_setup_entry:
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_IMPORT}, data=YAML_CONFIG
|
||||
)
|
||||
assert result["type"] == RESULT_TYPE_CREATE_ENTRY
|
||||
assert result["data"] == {
|
||||
CONF_MAC: "e7:89:43:99:99:99",
|
||||
CONF_NAME: "test-name",
|
||||
CONF_PASSWORD: "test-password",
|
||||
CONF_SENSOR_TYPE: "bot",
|
||||
}
|
||||
|
||||
assert len(mock_setup_entry.mock_calls) == 1
|
||||
|
||||
|
||||
async def test_user_form_exception(hass, switchbot_config_flow):
|
||||
"""Test we handle exception on user form."""
|
||||
await async_setup_component(hass, "persistent_notification", {})
|
||||
|
||||
switchbot_config_flow.side_effect = NotConnectedError
|
||||
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_USER}
|
||||
)
|
||||
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
USER_INPUT,
|
||||
)
|
||||
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["step_id"] == "user"
|
||||
assert result["errors"] == {"base": "cannot_connect"}
|
||||
|
||||
switchbot_config_flow.side_effect = Exception
|
||||
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
USER_INPUT,
|
||||
)
|
||||
|
||||
assert result["type"] == RESULT_TYPE_ABORT
|
||||
assert result["reason"] == "unknown"
|
||||
|
||||
|
||||
async def test_options_flow(hass):
|
||||
"""Test updating options."""
|
||||
with patch("homeassistant.components.switchbot.PLATFORMS", []):
|
||||
entry = await init_integration(hass)
|
||||
|
||||
assert entry.options["update_time"] == 60
|
||||
assert entry.options["retry_count"] == 3
|
||||
assert entry.options["retry_timeout"] == 5
|
||||
assert entry.options["scan_timeout"] == 5
|
||||
|
||||
result = await hass.config_entries.options.async_init(entry.entry_id)
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["step_id"] == "init"
|
||||
assert result["errors"] is None
|
||||
|
||||
with _patch_async_setup_entry() as mock_setup_entry:
|
||||
result = await hass.config_entries.options.async_configure(
|
||||
result["flow_id"],
|
||||
user_input={
|
||||
"update_time": 60,
|
||||
"retry_count": 3,
|
||||
"retry_timeout": 5,
|
||||
"scan_timeout": 5,
|
||||
},
|
||||
)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert result["type"] == RESULT_TYPE_CREATE_ENTRY
|
||||
assert result["data"]["update_time"] == 60
|
||||
assert result["data"]["retry_count"] == 3
|
||||
assert result["data"]["retry_timeout"] == 5
|
||||
assert result["data"]["scan_timeout"] == 5
|
||||
|
||||
assert len(mock_setup_entry.mock_calls) == 0
|
||||
|
||||
# Test changing of entry options.
|
||||
|
||||
result = await hass.config_entries.options.async_init(entry.entry_id)
|
||||
assert result["type"] == RESULT_TYPE_FORM
|
||||
assert result["step_id"] == "init"
|
||||
assert result["errors"] is None
|
||||
|
||||
with _patch_async_setup_entry() as mock_setup_entry:
|
||||
result = await hass.config_entries.options.async_configure(
|
||||
result["flow_id"],
|
||||
user_input={
|
||||
"update_time": 60,
|
||||
"retry_count": 3,
|
||||
"retry_timeout": 5,
|
||||
"scan_timeout": 5,
|
||||
},
|
||||
)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert result["type"] == RESULT_TYPE_CREATE_ENTRY
|
||||
assert result["data"]["update_time"] == 60
|
||||
assert result["data"]["retry_count"] == 3
|
||||
assert result["data"]["retry_timeout"] == 5
|
||||
assert result["data"]["scan_timeout"] == 5
|
||||
|
||||
assert len(mock_setup_entry.mock_calls) == 0
|
Loading…
Add table
Add a link
Reference in a new issue