2022-01-06 11:15:40 +00:00
|
|
|
"""Support for FRITZ!Box devices."""
|
2021-04-25 13:48:03 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-06-24 15:02:41 +00:00
|
|
|
import datetime
|
2015-12-09 19:22:40 +00:00
|
|
|
import logging
|
|
|
|
|
2022-01-11 07:27:33 +00:00
|
|
|
from homeassistant.components.device_tracker import SOURCE_TYPE_ROUTER
|
2021-04-25 10:10:33 +00:00
|
|
|
from homeassistant.components.device_tracker.config_entry import ScannerEntity
|
2022-01-11 07:27:33 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2021-04-25 10:10:33 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2021-06-13 14:45:35 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2015-12-09 19:22:40 +00:00
|
|
|
|
2021-09-29 03:59:03 +00:00
|
|
|
from .common import (
|
2022-01-20 11:43:32 +00:00
|
|
|
AvmWrapper,
|
2021-09-29 03:59:03 +00:00
|
|
|
FritzData,
|
|
|
|
FritzDevice,
|
|
|
|
FritzDeviceBase,
|
|
|
|
device_filter_out_from_trackers,
|
|
|
|
)
|
2021-07-16 11:38:37 +00:00
|
|
|
from .const import DATA_FRITZ, DOMAIN
|
2015-12-09 19:22:40 +00:00
|
|
|
|
2021-04-25 10:10:33 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-09-02 04:28:03 +00:00
|
|
|
|
2021-04-25 10:10:33 +00:00
|
|
|
YAML_DEFAULT_HOST = "169.254.1.1"
|
|
|
|
YAML_DEFAULT_USERNAME = "admin"
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
2021-06-13 14:45:35 +00:00
|
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
2021-04-25 10:10:33 +00:00
|
|
|
) -> None:
|
|
|
|
"""Set up device tracker for FRITZ!Box component."""
|
|
|
|
_LOGGER.debug("Starting FRITZ!Box device tracker")
|
2022-01-20 11:43:32 +00:00
|
|
|
avm_wrapper: AvmWrapper = hass.data[DOMAIN][entry.entry_id]
|
2021-06-13 14:45:35 +00:00
|
|
|
data_fritz: FritzData = hass.data[DATA_FRITZ]
|
2021-04-25 10:10:33 +00:00
|
|
|
|
|
|
|
@callback
|
2022-01-06 11:15:40 +00:00
|
|
|
def update_avm_device() -> None:
|
|
|
|
"""Update the values of AVM device."""
|
2022-01-20 11:43:32 +00:00
|
|
|
_async_add_entities(avm_wrapper, async_add_entities, data_fritz)
|
2021-04-25 10:10:33 +00:00
|
|
|
|
2021-05-14 16:46:37 +00:00
|
|
|
entry.async_on_unload(
|
2022-01-20 11:43:32 +00:00
|
|
|
async_dispatcher_connect(hass, avm_wrapper.signal_device_new, update_avm_device)
|
2021-05-14 16:46:37 +00:00
|
|
|
)
|
2020-01-01 21:03:38 +00:00
|
|
|
|
2022-01-06 11:15:40 +00:00
|
|
|
update_avm_device()
|
2020-01-01 21:03:38 +00:00
|
|
|
|
2015-12-09 19:22:40 +00:00
|
|
|
|
2021-04-25 10:10:33 +00:00
|
|
|
@callback
|
2021-06-13 14:45:35 +00:00
|
|
|
def _async_add_entities(
|
2022-01-20 11:43:32 +00:00
|
|
|
avm_wrapper: AvmWrapper,
|
2021-06-13 14:45:35 +00:00
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
data_fritz: FritzData,
|
|
|
|
) -> None:
|
2022-01-06 11:15:40 +00:00
|
|
|
"""Add new tracker entities from the AVM device."""
|
2021-04-29 18:10:36 +00:00
|
|
|
|
2021-04-25 10:10:33 +00:00
|
|
|
new_tracked = []
|
2022-01-20 11:43:32 +00:00
|
|
|
if avm_wrapper.unique_id not in data_fritz.tracked:
|
|
|
|
data_fritz.tracked[avm_wrapper.unique_id] = set()
|
2021-04-25 10:10:33 +00:00
|
|
|
|
2022-01-20 11:43:32 +00:00
|
|
|
for mac, device in avm_wrapper.devices.items():
|
2021-10-01 14:18:49 +00:00
|
|
|
if device_filter_out_from_trackers(mac, device, data_fritz.tracked.values()):
|
2021-04-25 10:10:33 +00:00
|
|
|
continue
|
|
|
|
|
2022-01-20 11:43:32 +00:00
|
|
|
new_tracked.append(FritzBoxTracker(avm_wrapper, device))
|
|
|
|
data_fritz.tracked[avm_wrapper.unique_id].add(mac)
|
2021-04-25 10:10:33 +00:00
|
|
|
|
|
|
|
if new_tracked:
|
|
|
|
async_add_entities(new_tracked)
|
|
|
|
|
|
|
|
|
2021-07-16 11:38:37 +00:00
|
|
|
class FritzBoxTracker(FritzDeviceBase, ScannerEntity):
|
2022-01-06 11:15:40 +00:00
|
|
|
"""This class queries a FRITZ!Box device."""
|
2021-04-25 10:10:33 +00:00
|
|
|
|
2022-01-20 11:43:32 +00:00
|
|
|
def __init__(self, avm_wrapper: AvmWrapper, device: FritzDevice) -> None:
|
2021-04-25 10:10:33 +00:00
|
|
|
"""Initialize a FRITZ!Box device."""
|
2022-01-20 11:43:32 +00:00
|
|
|
super().__init__(avm_wrapper, device)
|
2021-06-24 15:02:41 +00:00
|
|
|
self._last_activity: datetime.datetime | None = device.last_activity
|
2021-04-25 10:10:33 +00:00
|
|
|
|
|
|
|
@property
|
2021-06-13 14:45:35 +00:00
|
|
|
def is_connected(self) -> bool:
|
2021-04-25 10:10:33 +00:00
|
|
|
"""Return device status."""
|
2022-01-20 11:43:32 +00:00
|
|
|
return self._avm_wrapper.devices[self._mac].is_connected
|
2021-04-25 10:10:33 +00:00
|
|
|
|
|
|
|
@property
|
2021-06-13 14:45:35 +00:00
|
|
|
def unique_id(self) -> str:
|
2021-04-25 10:10:33 +00:00
|
|
|
"""Return device unique id."""
|
2021-07-16 11:38:37 +00:00
|
|
|
return f"{self._mac}_tracker"
|
2021-04-25 10:10:33 +00:00
|
|
|
|
2022-01-05 07:16:43 +00:00
|
|
|
@property
|
|
|
|
def mac_address(self) -> str:
|
|
|
|
"""Return mac_address."""
|
|
|
|
return self._mac
|
|
|
|
|
2021-04-25 10:10:33 +00:00
|
|
|
@property
|
2021-06-13 14:45:35 +00:00
|
|
|
def icon(self) -> str:
|
2021-04-25 10:10:33 +00:00
|
|
|
"""Return device icon."""
|
|
|
|
if self.is_connected:
|
|
|
|
return "mdi:lan-connect"
|
|
|
|
return "mdi:lan-disconnect"
|
|
|
|
|
2021-05-24 14:54:57 +00:00
|
|
|
@property
|
|
|
|
def extra_state_attributes(self) -> dict[str, str]:
|
|
|
|
"""Return the attributes."""
|
|
|
|
attrs: dict[str, str] = {}
|
2022-01-20 11:43:32 +00:00
|
|
|
device = self._avm_wrapper.devices[self._mac]
|
2021-12-30 22:23:55 +00:00
|
|
|
self._last_activity = device.last_activity
|
2021-05-24 14:54:57 +00:00
|
|
|
if self._last_activity is not None:
|
|
|
|
attrs["last_time_reachable"] = self._last_activity.isoformat(
|
|
|
|
timespec="seconds"
|
|
|
|
)
|
2021-12-30 22:23:55 +00:00
|
|
|
if device.connected_to:
|
|
|
|
attrs["connected_to"] = device.connected_to
|
|
|
|
if device.connection_type:
|
|
|
|
attrs["connection_type"] = device.connection_type
|
|
|
|
if device.ssid:
|
|
|
|
attrs["ssid"] = device.ssid
|
2021-05-24 14:54:57 +00:00
|
|
|
return attrs
|
|
|
|
|
2021-07-16 11:38:37 +00:00
|
|
|
@property
|
|
|
|
def source_type(self) -> str:
|
|
|
|
"""Return tracker source type."""
|
|
|
|
return SOURCE_TYPE_ROUTER
|