core/homeassistant/components/fritz/button.py

187 lines
6.0 KiB
Python

"""Switches for AVM Fritz!Box buttons."""
from __future__ import annotations
from collections.abc import Callable
from dataclasses import dataclass
import logging
from typing import Any, Final
from homeassistant.components.button import (
ButtonDeviceClass,
ButtonEntity,
ButtonEntityDescription,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import EntityCategory
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers.device_registry import CONNECTION_NETWORK_MAC, DeviceInfo
from homeassistant.helpers.dispatcher import async_dispatcher_connect
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from .const import BUTTON_TYPE_WOL, CONNECTION_TYPE_LAN, DATA_FRITZ, DOMAIN, MeshRoles
from .coordinator import AvmWrapper, FritzData, FritzDevice, _is_tracked
from .entity import FritzDeviceBase
_LOGGER = logging.getLogger(__name__)
@dataclass(frozen=True, kw_only=True)
class FritzButtonDescription(ButtonEntityDescription):
"""Class to describe a Button entity."""
press_action: Callable[[AvmWrapper], Any]
BUTTONS: Final = [
FritzButtonDescription(
key="firmware_update",
translation_key="firmware_update",
device_class=ButtonDeviceClass.UPDATE,
entity_category=EntityCategory.CONFIG,
press_action=lambda avm_wrapper: avm_wrapper.async_trigger_firmware_update(),
),
FritzButtonDescription(
key="reboot",
device_class=ButtonDeviceClass.RESTART,
entity_category=EntityCategory.CONFIG,
press_action=lambda avm_wrapper: avm_wrapper.async_trigger_reboot(),
),
FritzButtonDescription(
key="reconnect",
translation_key="reconnect",
device_class=ButtonDeviceClass.RESTART,
entity_category=EntityCategory.CONFIG,
press_action=lambda avm_wrapper: avm_wrapper.async_trigger_reconnect(),
),
FritzButtonDescription(
key="cleanup",
translation_key="cleanup",
entity_category=EntityCategory.CONFIG,
press_action=lambda avm_wrapper: avm_wrapper.async_trigger_cleanup(),
),
]
async def async_setup_entry(
hass: HomeAssistant,
entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set buttons for device."""
_LOGGER.debug("Setting up buttons")
avm_wrapper: AvmWrapper = hass.data[DOMAIN][entry.entry_id]
entities_list: list[ButtonEntity] = [
FritzButton(avm_wrapper, entry.title, button) for button in BUTTONS
]
if avm_wrapper.mesh_role == MeshRoles.SLAVE:
async_add_entities(entities_list)
return
data_fritz: FritzData = hass.data[DATA_FRITZ]
entities_list += _async_wol_buttons_list(avm_wrapper, data_fritz)
async_add_entities(entities_list)
@callback
def async_update_avm_device() -> None:
"""Update the values of the AVM device."""
async_add_entities(_async_wol_buttons_list(avm_wrapper, data_fritz))
entry.async_on_unload(
async_dispatcher_connect(
hass, avm_wrapper.signal_device_new, async_update_avm_device
)
)
class FritzButton(ButtonEntity):
"""Defines a Fritz!Box base button."""
entity_description: FritzButtonDescription
_attr_has_entity_name = True
def __init__(
self,
avm_wrapper: AvmWrapper,
device_friendly_name: str,
description: FritzButtonDescription,
) -> None:
"""Initialize Fritz!Box button."""
self.entity_description = description
self.avm_wrapper = avm_wrapper
self._attr_unique_id = f"{self.avm_wrapper.unique_id}-{description.key}"
self._attr_device_info = DeviceInfo(
connections={(CONNECTION_NETWORK_MAC, avm_wrapper.mac)},
name=device_friendly_name,
)
async def async_press(self) -> None:
"""Triggers Fritz!Box service."""
await self.entity_description.press_action(self.avm_wrapper)
@callback
def _async_wol_buttons_list(
avm_wrapper: AvmWrapper,
data_fritz: FritzData,
) -> list[FritzBoxWOLButton]:
"""Add new WOL button entities from the AVM device."""
_LOGGER.debug("Setting up %s buttons", BUTTON_TYPE_WOL)
new_wols: list[FritzBoxWOLButton] = []
if avm_wrapper.unique_id not in data_fritz.wol_buttons:
data_fritz.wol_buttons[avm_wrapper.unique_id] = set()
for mac, device in avm_wrapper.devices.items():
if _is_tracked(mac, data_fritz.wol_buttons.values()):
_LOGGER.debug("Skipping wol button creation for device %s", device.hostname)
continue
if device.connection_type != CONNECTION_TYPE_LAN:
_LOGGER.debug(
"Skipping wol button creation for device %s, not connected via LAN",
device.hostname,
)
continue
new_wols.append(FritzBoxWOLButton(avm_wrapper, device))
data_fritz.wol_buttons[avm_wrapper.unique_id].add(mac)
_LOGGER.debug("Creating %s wol buttons", len(new_wols))
return new_wols
class FritzBoxWOLButton(FritzDeviceBase, ButtonEntity):
"""Defines a FRITZ!Box Tools Wake On LAN button."""
_attr_icon = "mdi:lan-pending"
_attr_entity_registry_enabled_default = False
def __init__(self, avm_wrapper: AvmWrapper, device: FritzDevice) -> None:
"""Initialize Fritz!Box WOL button."""
super().__init__(avm_wrapper, device)
self._name = f"{self.hostname} Wake on LAN"
self._attr_unique_id = f"{self._mac}_wake_on_lan"
self._is_available = True
self._attr_device_info = DeviceInfo(
connections={(CONNECTION_NETWORK_MAC, self._mac)},
default_manufacturer="AVM",
default_model="FRITZ!Box Tracked device",
default_name=device.hostname,
via_device=(
DOMAIN,
avm_wrapper.unique_id,
),
)
async def async_press(self) -> None:
"""Press the button."""
if self.mac_address:
await self._avm_wrapper.async_wake_on_lan(self.mac_address)