Handle an unsupported device in the Broadlink config flow (#40242)

pull/40310/head
Felipe Martins Diel 2020-09-18 10:29:26 -03:00 committed by Paulus Schoutsen
parent 57b7ed6a07
commit 1eb8035122
5 changed files with 62 additions and 1 deletions

View File

@ -11,7 +11,7 @@ from broadlink.exceptions import (
) )
import voluptuous as vol import voluptuous as vol
from homeassistant import config_entries from homeassistant import config_entries, data_entry_flow
from homeassistant.const import CONF_HOST, CONF_MAC, CONF_NAME, CONF_TIMEOUT, CONF_TYPE from homeassistant.const import CONF_HOST, CONF_MAC, CONF_NAME, CONF_TIMEOUT, CONF_TYPE
from homeassistant.helpers import config_validation as cv from homeassistant.helpers import config_validation as cv
@ -20,6 +20,7 @@ from .const import ( # pylint: disable=unused-import
DEFAULT_PORT, DEFAULT_PORT,
DEFAULT_TIMEOUT, DEFAULT_TIMEOUT,
DOMAIN, DOMAIN,
DOMAINS_AND_TYPES,
) )
from .helpers import format_mac from .helpers import format_mac
@ -36,6 +37,19 @@ class BroadlinkFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
async def async_set_device(self, device, raise_on_progress=True): async def async_set_device(self, device, raise_on_progress=True):
"""Define a device for the config flow.""" """Define a device for the config flow."""
supported_types = {
device_type
for _, device_types in DOMAINS_AND_TYPES
for device_type in device_types
}
if device.type not in supported_types:
LOGGER.error(
"Unsupported device: %s. If it worked before, please open "
"an issue at https://github.com/home-assistant/core/issues",
hex(device.devtype),
)
raise data_entry_flow.AbortFlow("not_supported")
await self.async_set_unique_id( await self.async_set_unique_id(
device.mac.hex(), raise_on_progress=raise_on_progress device.mac.hex(), raise_on_progress=raise_on_progress
) )

View File

@ -35,6 +35,7 @@
"already_in_progress": "There is already a configuration flow in progress for this device", "already_in_progress": "There is already a configuration flow in progress for this device",
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]", "cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
"invalid_host": "Invalid hostname or IP address", "invalid_host": "Invalid hostname or IP address",
"not_supported": "Device not supported",
"unknown": "[%key:common::config_flow::error::unknown%]" "unknown": "[%key:common::config_flow::error::unknown%]"
}, },
"error": { "error": {

View File

@ -5,6 +5,7 @@
"already_in_progress": "There is already a configuration flow in progress for this device", "already_in_progress": "There is already a configuration flow in progress for this device",
"cannot_connect": "Failed to connect", "cannot_connect": "Failed to connect",
"invalid_host": "Invalid hostname or IP address", "invalid_host": "Invalid hostname or IP address",
"not_supported": "Device not supported",
"unknown": "Unexpected error" "unknown": "Unexpected error"
}, },
"error": { "error": {

View File

@ -56,6 +56,16 @@ BROADLINK_DEVICES = {
20025, 20025,
5, 5,
), ),
"Kitchen": ( # Not supported.
"192.168.0.64",
"34ea34b61d2c",
"LB1",
"Broadlink",
"SmartBulb",
0x504E,
57,
5,
),
} }

View File

@ -172,6 +172,25 @@ async def test_flow_user_device_not_found(hass):
assert result["errors"] == {"base": "cannot_connect"} assert result["errors"] == {"base": "cannot_connect"}
async def test_flow_user_device_not_supported(hass):
"""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}
)
with patch(DEVICE_DISCOVERY, return_value=[mock_api]):
result = await hass.config_entries.flow.async_configure(
result["flow_id"],
{"host": device.host},
)
assert result["type"] == "abort"
assert result["reason"] == "not_supported"
async def test_flow_user_network_unreachable(hass): async def test_flow_user_network_unreachable(hass):
"""Test we handle a network unreachable in the user step.""" """Test we handle a network unreachable in the user step."""
result = await hass.config_entries.flow.async_init( result = await hass.config_entries.flow.async_init(
@ -631,6 +650,22 @@ async def test_flow_import_device_not_found(hass):
assert result["reason"] == "cannot_connect" assert result["reason"] == "cannot_connect"
async def test_flow_import_device_not_supported(hass):
"""Test we handle a device not supported in the import step."""
device = get_device("Kitchen")
mock_api = device.get_mock_api()
with patch(DEVICE_DISCOVERY, return_value=[mock_api]):
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"
async def test_flow_import_invalid_ip_address(hass): async def test_flow_import_invalid_ip_address(hass):
"""Test we handle an invalid IP address in the import step.""" """Test we handle an invalid IP address in the import step."""
with patch("broadlink.discover", side_effect=OSError(errno.EINVAL, None)): with patch("broadlink.discover", side_effect=OSError(errno.EINVAL, None)):