Add reboot button to SFRBox (#86514)
parent
44beb350cd
commit
e96cea997e
|
@ -13,7 +13,7 @@ from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
|
|||
from homeassistant.helpers import device_registry as dr
|
||||
from homeassistant.helpers.httpx_client import get_async_client
|
||||
|
||||
from .const import DOMAIN, PLATFORMS
|
||||
from .const import DOMAIN, PLATFORMS, PLATFORMS_WITH_AUTH
|
||||
from .coordinator import SFRDataUpdateCoordinator
|
||||
from .models import DomainData
|
||||
|
||||
|
@ -21,6 +21,7 @@ from .models import DomainData
|
|||
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||
"""Set up SFR box as config entry."""
|
||||
box = SFRBox(ip=entry.data[CONF_HOST], client=get_async_client(hass))
|
||||
platforms = PLATFORMS
|
||||
if (username := entry.data.get(CONF_USERNAME)) and (
|
||||
password := entry.data.get(CONF_PASSWORD)
|
||||
):
|
||||
|
@ -30,8 +31,10 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||
raise ConfigEntryAuthFailed() from err
|
||||
except SFRBoxError as err:
|
||||
raise ConfigEntryNotReady() from err
|
||||
platforms = PLATFORMS_WITH_AUTH
|
||||
|
||||
data = DomainData(
|
||||
box=box,
|
||||
dsl=SFRDataUpdateCoordinator(hass, box, "dsl", lambda b: b.dsl_get_info()),
|
||||
system=SFRDataUpdateCoordinator(
|
||||
hass, box, "system", lambda b: b.system_get_info()
|
||||
|
@ -56,7 +59,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||
configuration_url=f"http://{entry.data[CONF_HOST]}",
|
||||
)
|
||||
|
||||
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
||||
await hass.config_entries.async_forward_entry_setups(entry, platforms)
|
||||
|
||||
return True
|
||||
|
||||
|
|
|
@ -0,0 +1,108 @@
|
|||
"""SFR Box button platform."""
|
||||
from __future__ import annotations
|
||||
|
||||
from collections.abc import Awaitable, Callable, Coroutine
|
||||
from dataclasses import dataclass
|
||||
from functools import wraps
|
||||
from typing import Any, Concatenate, ParamSpec, TypeVar
|
||||
|
||||
from sfrbox_api.bridge import SFRBox
|
||||
from sfrbox_api.exceptions import SFRBoxError
|
||||
from sfrbox_api.models import SystemInfo
|
||||
|
||||
from homeassistant.components.button import (
|
||||
ButtonDeviceClass,
|
||||
ButtonEntity,
|
||||
ButtonEntityDescription,
|
||||
)
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.exceptions import HomeAssistantError
|
||||
from homeassistant.helpers.entity import EntityCategory
|
||||
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||
|
||||
from .const import DOMAIN
|
||||
from .models import DomainData
|
||||
|
||||
_T = TypeVar("_T")
|
||||
_P = ParamSpec("_P")
|
||||
|
||||
|
||||
def with_error_wrapping(
|
||||
func: Callable[Concatenate[SFRBoxButton, _P], Awaitable[_T]]
|
||||
) -> Callable[Concatenate[SFRBoxButton, _P], Coroutine[Any, Any, _T]]:
|
||||
"""Catch SFR errors."""
|
||||
|
||||
@wraps(func)
|
||||
async def wrapper(
|
||||
self: SFRBoxButton,
|
||||
*args: _P.args,
|
||||
**kwargs: _P.kwargs,
|
||||
) -> _T:
|
||||
"""Catch SFRBoxError errors and raise HomeAssistantError."""
|
||||
try:
|
||||
return await func(self, *args, **kwargs)
|
||||
except SFRBoxError as err:
|
||||
raise HomeAssistantError(err) from err
|
||||
|
||||
return wrapper
|
||||
|
||||
|
||||
@dataclass
|
||||
class SFRBoxButtonMixin:
|
||||
"""Mixin for SFR Box buttons."""
|
||||
|
||||
async_press: Callable[[SFRBox], Coroutine[None, None, None]]
|
||||
|
||||
|
||||
@dataclass
|
||||
class SFRBoxButtonEntityDescription(ButtonEntityDescription, SFRBoxButtonMixin):
|
||||
"""Description for SFR Box buttons."""
|
||||
|
||||
|
||||
BUTTON_TYPES: tuple[SFRBoxButtonEntityDescription, ...] = (
|
||||
SFRBoxButtonEntityDescription(
|
||||
async_press=lambda x: x.system_reboot(),
|
||||
device_class=ButtonDeviceClass.RESTART,
|
||||
entity_category=EntityCategory.CONFIG,
|
||||
key="system_reboot",
|
||||
name="Reboot",
|
||||
),
|
||||
)
|
||||
|
||||
|
||||
async def async_setup_entry(
|
||||
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||
) -> None:
|
||||
"""Set up the buttons."""
|
||||
data: DomainData = hass.data[DOMAIN][entry.entry_id]
|
||||
|
||||
entities = [
|
||||
SFRBoxButton(data.box, description, data.system.data)
|
||||
for description in BUTTON_TYPES
|
||||
]
|
||||
async_add_entities(entities)
|
||||
|
||||
|
||||
class SFRBoxButton(ButtonEntity):
|
||||
"""Mixin for button specific attributes."""
|
||||
|
||||
entity_description: SFRBoxButtonEntityDescription
|
||||
_attr_has_entity_name = True
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
box: SFRBox,
|
||||
description: SFRBoxButtonEntityDescription,
|
||||
system_info: SystemInfo,
|
||||
) -> None:
|
||||
"""Initialize the sensor."""
|
||||
self.entity_description = description
|
||||
self._box = box
|
||||
self._attr_unique_id = f"{system_info.mac_addr}_{description.key}"
|
||||
self._attr_device_info = {"identifiers": {(DOMAIN, system_info.mac_addr)}}
|
||||
|
||||
@with_error_wrapping
|
||||
async def async_press(self) -> None:
|
||||
"""Process the button press."""
|
||||
await self.entity_description.async_press(self._box)
|
|
@ -7,3 +7,4 @@ DEFAULT_USERNAME = "admin"
|
|||
DOMAIN = "sfr_box"
|
||||
|
||||
PLATFORMS = [Platform.BINARY_SENSOR, Platform.SENSOR]
|
||||
PLATFORMS_WITH_AUTH = [*PLATFORMS, Platform.BUTTON]
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
"""SFR Box models."""
|
||||
from dataclasses import dataclass
|
||||
|
||||
from sfrbox_api.bridge import SFRBox
|
||||
from sfrbox_api.models import DslInfo, SystemInfo
|
||||
|
||||
from .coordinator import SFRDataUpdateCoordinator
|
||||
|
@ -10,5 +11,6 @@ from .coordinator import SFRDataUpdateCoordinator
|
|||
class DomainData:
|
||||
"""Domain data for SFR Box."""
|
||||
|
||||
box: SFRBox
|
||||
dsl: SFRDataUpdateCoordinator[DslInfo]
|
||||
system: SFRDataUpdateCoordinator[SystemInfo]
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
"""Constants for SFR Box tests."""
|
||||
from homeassistant.components.binary_sensor import BinarySensorDeviceClass
|
||||
from homeassistant.components.button import ButtonDeviceClass
|
||||
from homeassistant.components.sensor import (
|
||||
ATTR_OPTIONS,
|
||||
ATTR_STATE_CLASS,
|
||||
|
@ -18,6 +19,7 @@ from homeassistant.const import (
|
|||
ATTR_UNIT_OF_MEASUREMENT,
|
||||
SIGNAL_STRENGTH_DECIBELS,
|
||||
STATE_ON,
|
||||
STATE_UNKNOWN,
|
||||
Platform,
|
||||
UnitOfDataRate,
|
||||
UnitOfElectricPotential,
|
||||
|
@ -48,6 +50,14 @@ EXPECTED_ENTITIES = {
|
|||
ATTR_UNIQUE_ID: "e4:5d:51:00:11:22_dsl_status",
|
||||
},
|
||||
],
|
||||
Platform.BUTTON: [
|
||||
{
|
||||
ATTR_DEVICE_CLASS: ButtonDeviceClass.RESTART,
|
||||
ATTR_ENTITY_ID: "button.sfr_box_reboot",
|
||||
ATTR_STATE: STATE_UNKNOWN,
|
||||
ATTR_UNIQUE_ID: "e4:5d:51:00:11:22_system_reboot",
|
||||
},
|
||||
],
|
||||
Platform.SENSOR: [
|
||||
{
|
||||
ATTR_DEFAULT_DISABLED: True,
|
||||
|
|
|
@ -0,0 +1,71 @@
|
|||
"""Test the SFR Box buttons."""
|
||||
from collections.abc import Generator
|
||||
from unittest.mock import patch
|
||||
|
||||
import pytest
|
||||
from sfrbox_api.exceptions import SFRBoxError
|
||||
|
||||
from homeassistant.components.button import DOMAIN as BUTTON_DOMAIN, SERVICE_PRESS
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import ATTR_ENTITY_ID, Platform
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.exceptions import HomeAssistantError
|
||||
|
||||
from . import check_device_registry, check_entities
|
||||
from .const import EXPECTED_ENTITIES
|
||||
|
||||
from tests.common import mock_device_registry, mock_registry
|
||||
|
||||
pytestmark = pytest.mark.usefixtures("system_get_info", "dsl_get_info")
|
||||
|
||||
|
||||
@pytest.fixture(autouse=True)
|
||||
def override_platforms() -> Generator[None, None, None]:
|
||||
"""Override PLATFORMS_WITH_AUTH."""
|
||||
with patch(
|
||||
"homeassistant.components.sfr_box.PLATFORMS_WITH_AUTH", [Platform.BUTTON]
|
||||
), patch("homeassistant.components.sfr_box.coordinator.SFRBox.authenticate"):
|
||||
yield
|
||||
|
||||
|
||||
async def test_buttons(
|
||||
hass: HomeAssistant, config_entry_with_auth: ConfigEntry
|
||||
) -> None:
|
||||
"""Test for SFR Box buttons."""
|
||||
entity_registry = mock_registry(hass)
|
||||
device_registry = mock_device_registry(hass)
|
||||
|
||||
await hass.config_entries.async_setup(config_entry_with_auth.entry_id)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
check_device_registry(device_registry, EXPECTED_ENTITIES["expected_device"])
|
||||
|
||||
expected_entities = EXPECTED_ENTITIES[Platform.BUTTON]
|
||||
assert len(entity_registry.entities) == len(expected_entities)
|
||||
|
||||
check_entities(hass, entity_registry, expected_entities)
|
||||
|
||||
# Reboot success
|
||||
service_data = {ATTR_ENTITY_ID: "button.sfr_box_reboot"}
|
||||
with patch(
|
||||
"homeassistant.components.sfr_box.button.SFRBox.system_reboot"
|
||||
) as mock_action:
|
||||
await hass.services.async_call(
|
||||
BUTTON_DOMAIN, SERVICE_PRESS, service_data=service_data, blocking=True
|
||||
)
|
||||
|
||||
assert len(mock_action.mock_calls) == 1
|
||||
assert mock_action.mock_calls[0][1] == ()
|
||||
|
||||
# Reboot failed
|
||||
service_data = {ATTR_ENTITY_ID: "button.sfr_box_reboot"}
|
||||
with patch(
|
||||
"homeassistant.components.sfr_box.button.SFRBox.system_reboot",
|
||||
side_effect=SFRBoxError,
|
||||
) as mock_action, pytest.raises(HomeAssistantError):
|
||||
await hass.services.async_call(
|
||||
BUTTON_DOMAIN, SERVICE_PRESS, service_data=service_data, blocking=True
|
||||
)
|
||||
|
||||
assert len(mock_action.mock_calls) == 1
|
||||
assert mock_action.mock_calls[0][1] == ()
|
Loading…
Reference in New Issue