Add dhcp discovery to D-Link (#85661)
Co-authored-by: J. Nick Koston <nick@koston.org>pull/85895/head
parent
77f9548e51
commit
3627a98602
|
@ -8,6 +8,7 @@ from pyW215.pyW215 import SmartPlug
|
|||
import voluptuous as vol
|
||||
|
||||
from homeassistant import config_entries
|
||||
from homeassistant.components import dhcp
|
||||
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, CONF_USERNAME
|
||||
from homeassistant.data_entry_flow import FlowResult
|
||||
|
||||
|
@ -19,6 +20,58 @@ _LOGGER = logging.getLogger(__name__)
|
|||
class DLinkFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
||||
"""Handle a config flow for D-Link Power Plug."""
|
||||
|
||||
def __init__(self) -> None:
|
||||
"""Initialize a D-Link Power Plug flow."""
|
||||
self.ip_address: str | None = None
|
||||
|
||||
async def async_step_dhcp(self, discovery_info: dhcp.DhcpServiceInfo) -> FlowResult:
|
||||
"""Handle dhcp discovery."""
|
||||
await self.async_set_unique_id(discovery_info.macaddress)
|
||||
self._abort_if_unique_id_configured(updates={CONF_HOST: discovery_info.ip})
|
||||
for entry in self.hass.config_entries.async_entries(DOMAIN):
|
||||
if not entry.unique_id and entry.data[CONF_HOST] == discovery_info.ip:
|
||||
# Add mac address as the unique id, can be removed with import
|
||||
self.hass.config_entries.async_update_entry(
|
||||
entry, unique_id=discovery_info.macaddress
|
||||
)
|
||||
return self.async_abort(reason="already_configured")
|
||||
|
||||
self.ip_address = discovery_info.ip
|
||||
return await self.async_step_confirm_discovery()
|
||||
|
||||
async def async_step_confirm_discovery(
|
||||
self, user_input: dict[str, Any] | None = None
|
||||
) -> FlowResult:
|
||||
"""Allow the user to confirm adding the device."""
|
||||
errors = {}
|
||||
if user_input is not None:
|
||||
if (
|
||||
error := await self.hass.async_add_executor_job(
|
||||
self._try_connect, user_input
|
||||
)
|
||||
) is None:
|
||||
return self.async_create_entry(
|
||||
title=DEFAULT_NAME,
|
||||
data=user_input | {CONF_HOST: self.ip_address},
|
||||
)
|
||||
errors["base"] = error
|
||||
|
||||
user_input = user_input or {}
|
||||
return self.async_show_form(
|
||||
step_id="confirm_discovery",
|
||||
data_schema=vol.Schema(
|
||||
{
|
||||
vol.Optional(
|
||||
CONF_USERNAME,
|
||||
default=user_input.get(CONF_USERNAME, DEFAULT_USERNAME),
|
||||
): str,
|
||||
vol.Required(CONF_PASSWORD): str,
|
||||
vol.Required(CONF_USE_LEGACY_PROTOCOL): bool,
|
||||
}
|
||||
),
|
||||
errors=errors,
|
||||
)
|
||||
|
||||
async def async_step_import(self, config: dict[str, Any]) -> FlowResult:
|
||||
"""Import a config entry."""
|
||||
self._async_abort_entries_match({CONF_HOST: config[CONF_HOST]})
|
||||
|
@ -36,10 +89,11 @@ class DLinkFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
|||
if user_input is not None:
|
||||
self._async_abort_entries_match({CONF_HOST: user_input[CONF_HOST]})
|
||||
|
||||
error = await self.hass.async_add_executor_job(
|
||||
if (
|
||||
error := await self.hass.async_add_executor_job(
|
||||
self._try_connect, user_input
|
||||
)
|
||||
if error is None:
|
||||
) is None:
|
||||
return self.async_create_entry(
|
||||
title=DEFAULT_NAME,
|
||||
data=user_input,
|
||||
|
@ -51,7 +105,9 @@ class DLinkFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
|||
step_id="user",
|
||||
data_schema=vol.Schema(
|
||||
{
|
||||
vol.Required(CONF_HOST, default=user_input.get(CONF_HOST, "")): str,
|
||||
vol.Required(
|
||||
CONF_HOST, default=user_input.get(CONF_HOST, self.ip_address)
|
||||
): str,
|
||||
vol.Optional(
|
||||
CONF_USERNAME,
|
||||
default=user_input.get(CONF_USERNAME, DEFAULT_USERNAME),
|
||||
|
@ -67,7 +123,7 @@ class DLinkFlowHandler(config_entries.ConfigFlow, domain=DOMAIN):
|
|||
"""Try connecting to D-Link Power Plug."""
|
||||
try:
|
||||
smartplug = SmartPlug(
|
||||
user_input[CONF_HOST],
|
||||
user_input.get(CONF_HOST, self.ip_address),
|
||||
user_input[CONF_PASSWORD],
|
||||
user_input[CONF_USERNAME],
|
||||
user_input[CONF_USE_LEGACY_PROTOCOL],
|
||||
|
|
|
@ -4,6 +4,7 @@
|
|||
"config_flow": true,
|
||||
"documentation": "https://www.home-assistant.io/integrations/dlink",
|
||||
"requirements": ["pyW215==0.7.0"],
|
||||
"dhcp": [{ "hostname": "dsp-w215" }],
|
||||
"codeowners": ["@tkdrob"],
|
||||
"iot_class": "local_polling",
|
||||
"loggers": ["pyW215"],
|
||||
|
|
|
@ -8,6 +8,13 @@
|
|||
"username": "[%key:common::config_flow::data::username%]",
|
||||
"use_legacy_protocol": "Use legacy protocol"
|
||||
}
|
||||
},
|
||||
"confirm_discovery": {
|
||||
"data": {
|
||||
"password": "[%key:component::dlink::config::step::user::data::password%]",
|
||||
"username": "[%key:common::config_flow::data::username%]",
|
||||
"use_legacy_protocol": "[%key:component::dlink::config::step::user::data::use_legacy_protocol%]"
|
||||
}
|
||||
}
|
||||
},
|
||||
"error": {
|
||||
|
|
|
@ -15,6 +15,13 @@
|
|||
"use_legacy_protocol": "Use legacy protocol",
|
||||
"username": "Username"
|
||||
}
|
||||
},
|
||||
"confirm_discovery": {
|
||||
"data": {
|
||||
"password": "Password (default: PIN code on the back)",
|
||||
"use_legacy_protocol": "Use legacy protocol",
|
||||
"username": "Username"
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
|
|
|
@ -97,6 +97,10 @@ DHCP: list[dict[str, str | bool]] = [
|
|||
"domain": "broadlink",
|
||||
"macaddress": "C8F742*",
|
||||
},
|
||||
{
|
||||
"domain": "dlink",
|
||||
"hostname": "dsp-w215",
|
||||
},
|
||||
{
|
||||
"domain": "elkm1",
|
||||
"registered_devices": True,
|
||||
|
|
|
@ -5,25 +5,41 @@ from unittest.mock import MagicMock, patch
|
|||
|
||||
import pytest
|
||||
|
||||
from homeassistant.components import dhcp
|
||||
from homeassistant.components.dlink.const import CONF_USE_LEGACY_PROTOCOL, DOMAIN
|
||||
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, CONF_USERNAME
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.helpers.device_registry import format_mac
|
||||
|
||||
from tests.common import MockConfigEntry
|
||||
|
||||
HOST = "1.2.3.4"
|
||||
PASSWORD = "123456"
|
||||
MAC = format_mac("AA:BB:CC:DD:EE:FF")
|
||||
USERNAME = "admin"
|
||||
|
||||
CONF_DATA = {
|
||||
CONF_HOST: HOST,
|
||||
CONF_DHCP_DATA = {
|
||||
CONF_USERNAME: USERNAME,
|
||||
CONF_PASSWORD: PASSWORD,
|
||||
CONF_USE_LEGACY_PROTOCOL: True,
|
||||
}
|
||||
|
||||
CONF_DATA = CONF_DHCP_DATA | {CONF_HOST: HOST}
|
||||
|
||||
CONF_IMPORT_DATA = CONF_DATA | {CONF_NAME: "Smart Plug"}
|
||||
|
||||
CONF_DHCP_FLOW = dhcp.DhcpServiceInfo(
|
||||
ip=HOST,
|
||||
macaddress=MAC,
|
||||
hostname="dsp-w215",
|
||||
)
|
||||
|
||||
CONF_DHCP_FLOW_NEW_IP = dhcp.DhcpServiceInfo(
|
||||
ip="5.6.7.8",
|
||||
macaddress=MAC,
|
||||
hostname="dsp-w215",
|
||||
)
|
||||
|
||||
|
||||
def create_entry(hass: HomeAssistant) -> MockConfigEntry:
|
||||
"""Create fixture for adding config entry in Home Assistant."""
|
||||
|
@ -38,6 +54,14 @@ def config_entry(hass: HomeAssistant) -> MockConfigEntry:
|
|||
return create_entry(hass)
|
||||
|
||||
|
||||
@pytest.fixture()
|
||||
def config_entry_with_uid(hass: HomeAssistant) -> MockConfigEntry:
|
||||
"""Add config entry with unique ID in Home Assistant."""
|
||||
config_entry = create_entry(hass)
|
||||
config_entry.unique_id = "aa:bb:cc:dd:ee:ff"
|
||||
return config_entry
|
||||
|
||||
|
||||
@pytest.fixture()
|
||||
def mocked_plug() -> MagicMock:
|
||||
"""Create mocked plug device."""
|
||||
|
|
|
@ -2,11 +2,20 @@
|
|||
from unittest.mock import MagicMock, patch
|
||||
|
||||
from homeassistant import data_entry_flow
|
||||
from homeassistant.components import dhcp
|
||||
from homeassistant.components.dlink.const import DEFAULT_NAME, DOMAIN
|
||||
from homeassistant.config_entries import SOURCE_IMPORT, SOURCE_USER
|
||||
from homeassistant.config_entries import SOURCE_DHCP, SOURCE_IMPORT, SOURCE_USER
|
||||
from homeassistant.const import CONF_HOST
|
||||
from homeassistant.core import HomeAssistant
|
||||
|
||||
from .conftest import CONF_DATA, CONF_IMPORT_DATA, patch_config_flow
|
||||
from .conftest import (
|
||||
CONF_DATA,
|
||||
CONF_DHCP_DATA,
|
||||
CONF_DHCP_FLOW,
|
||||
CONF_DHCP_FLOW_NEW_IP,
|
||||
CONF_IMPORT_DATA,
|
||||
patch_config_flow,
|
||||
)
|
||||
|
||||
from tests.common import MockConfigEntry
|
||||
|
||||
|
@ -99,3 +108,97 @@ async def test_import(hass: HomeAssistant, mocked_plug: MagicMock) -> None:
|
|||
assert result["type"] == data_entry_flow.FlowResultType.CREATE_ENTRY
|
||||
assert result["title"] == "Smart Plug"
|
||||
assert result["data"] == CONF_DATA
|
||||
|
||||
|
||||
async def test_dhcp(hass: HomeAssistant, mocked_plug: MagicMock) -> None:
|
||||
"""Test we can process the discovery from dhcp."""
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_DHCP}, data=CONF_DHCP_FLOW
|
||||
)
|
||||
assert result["type"] == data_entry_flow.FlowResultType.FORM
|
||||
assert result["step_id"] == "confirm_discovery"
|
||||
with patch_config_flow(mocked_plug), _patch_setup_entry():
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
user_input=CONF_DHCP_DATA,
|
||||
)
|
||||
assert result["type"] == data_entry_flow.FlowResultType.CREATE_ENTRY
|
||||
assert result["title"] == DEFAULT_NAME
|
||||
assert result["data"] == CONF_DATA
|
||||
|
||||
|
||||
async def test_dhcp_failed_auth(
|
||||
hass: HomeAssistant, mocked_plug: MagicMock, mocked_plug_no_auth: MagicMock
|
||||
) -> None:
|
||||
"""Test we can recovery from failed authentication during dhcp flow."""
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_DHCP}, data=CONF_DHCP_FLOW
|
||||
)
|
||||
assert result["type"] == data_entry_flow.FlowResultType.FORM
|
||||
assert result["step_id"] == "confirm_discovery"
|
||||
with patch_config_flow(mocked_plug_no_auth):
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
user_input=CONF_DHCP_DATA,
|
||||
)
|
||||
assert result["type"] == data_entry_flow.FlowResultType.FORM
|
||||
assert result["errors"]["base"] == "cannot_connect"
|
||||
|
||||
with patch_config_flow(mocked_plug), _patch_setup_entry():
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
user_input=CONF_DHCP_DATA,
|
||||
)
|
||||
assert result["type"] == data_entry_flow.FlowResultType.CREATE_ENTRY
|
||||
assert result["title"] == DEFAULT_NAME
|
||||
assert result["data"] == CONF_DATA
|
||||
|
||||
|
||||
async def test_dhcp_already_configured(
|
||||
hass: HomeAssistant, config_entry: MockConfigEntry
|
||||
) -> None:
|
||||
"""Test dhcp initialized flow with duplicate server."""
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_DHCP}, data=CONF_DHCP_FLOW
|
||||
)
|
||||
|
||||
assert result["type"] == data_entry_flow.FlowResultType.ABORT
|
||||
assert result["reason"] == "already_configured"
|
||||
assert config_entry.unique_id == "aa:bb:cc:dd:ee:ff"
|
||||
|
||||
|
||||
async def test_dhcp_unique_id_assignment(
|
||||
hass: HomeAssistant, mocked_plug: MagicMock
|
||||
) -> None:
|
||||
"""Test dhcp initialized flow with no unique id for matching entry."""
|
||||
dhcp_data = dhcp.DhcpServiceInfo(
|
||||
ip="2.3.4.5",
|
||||
macaddress="11:22:33:44:55:66",
|
||||
hostname="dsp-w215",
|
||||
)
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_DHCP}, data=dhcp_data
|
||||
)
|
||||
assert result["type"] == data_entry_flow.FlowResultType.FORM
|
||||
assert result["step_id"] == "confirm_discovery"
|
||||
with patch_config_flow(mocked_plug), _patch_setup_entry():
|
||||
result = await hass.config_entries.flow.async_configure(
|
||||
result["flow_id"],
|
||||
user_input=CONF_DHCP_DATA,
|
||||
)
|
||||
assert result["type"] == data_entry_flow.FlowResultType.CREATE_ENTRY
|
||||
assert result["data"] == CONF_DATA | {CONF_HOST: "2.3.4.5"}
|
||||
assert result["result"].unique_id == "11:22:33:44:55:66"
|
||||
|
||||
|
||||
async def test_dhcp_changed_ip(
|
||||
hass: HomeAssistant, config_entry_with_uid: MockConfigEntry
|
||||
) -> None:
|
||||
"""Test that we successfully change IP address for device with known mac address."""
|
||||
result = await hass.config_entries.flow.async_init(
|
||||
DOMAIN, context={"source": SOURCE_DHCP}, data=CONF_DHCP_FLOW_NEW_IP
|
||||
)
|
||||
|
||||
assert result["type"] == data_entry_flow.FlowResultType.ABORT
|
||||
assert result["reason"] == "already_configured"
|
||||
assert config_entry_with_uid.data[CONF_HOST] == "5.6.7.8"
|
||||
|
|
Loading…
Reference in New Issue