core/homeassistant/components/tplink_omada/device_tracker.py

108 lines
3.4 KiB
Python
Raw Normal View History

"""Connected Wi-Fi device scanners for TP-Link Omada access points."""
import logging
from tplink_omada_client.clients import OmadaWirelessClient
from homeassistant.components.device_tracker import ScannerEntity, SourceType
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.update_coordinator import CoordinatorEntity
from .config_flow import CONF_SITE
from .const import DOMAIN
from .controller import OmadaClientsCoordinator, OmadaSiteController
_LOGGER = logging.getLogger(__name__)
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up device trackers and scanners."""
controller: OmadaSiteController = hass.data[DOMAIN][config_entry.entry_id]
clients_coordinator = controller.get_clients_coordinator()
site_id = config_entry.data[CONF_SITE]
# Add all known WiFi devices as potentially tracked devices. They will only be
# tracked if the user enables the entity.
async_add_entities(
[
OmadaClientScannerEntity(
site_id, client.mac, client.name, clients_coordinator
)
async for client in controller.omada_client.get_known_clients()
if isinstance(client, OmadaWirelessClient)
]
)
class OmadaClientScannerEntity(
CoordinatorEntity[OmadaClientsCoordinator], ScannerEntity
):
"""Entity for a client connected to the Omada network."""
_client_details: OmadaWirelessClient | None = None
def __init__(
self,
site_id: str,
client_id: str,
display_name: str,
coordinator: OmadaClientsCoordinator,
) -> None:
"""Initialize the scanner."""
super().__init__(coordinator)
self._site_id = site_id
self._client_id = client_id
self._attr_name = display_name
@property
def source_type(self) -> SourceType:
"""Return the source type of the device."""
return SourceType.ROUTER
def _do_update(self) -> None:
self._client_details = self.coordinator.data.get(self._client_id)
async def async_added_to_hass(self) -> None:
"""When entity is added to hass."""
await super().async_added_to_hass()
self._do_update()
@callback
def _handle_coordinator_update(self) -> None:
"""Handle updated data from the coordinator."""
self._do_update()
self.async_write_ha_state()
@property
def ip_address(self) -> str | None:
"""Return the primary ip address of the device."""
return self._client_details.ip if self._client_details else None
@property
def mac_address(self) -> str | None:
"""Return the mac address of the device."""
return self._client_id
@property
def hostname(self) -> str | None:
"""Return hostname of the device."""
return self._client_details.host_name if self._client_details else None
@property
def is_connected(self) -> bool:
"""Return true if the device is connected to the network."""
return self._client_details.is_active if self._client_details else False
@property
def unique_id(self) -> str | None:
"""Return the unique id of the device."""
return f"scanner_{self._site_id}_{self._client_id}"