2022-08-16 18:08:35 +00:00
|
|
|
"""Base entity for the Fully Kiosk Browser integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
2023-11-23 09:38:32 +00:00
|
|
|
import json
|
|
|
|
|
2023-11-13 16:40:57 +00:00
|
|
|
from yarl import URL
|
|
|
|
|
2023-11-23 09:38:32 +00:00
|
|
|
from homeassistant.components import mqtt
|
2023-11-02 20:59:25 +00:00
|
|
|
from homeassistant.const import ATTR_CONNECTIONS
|
2023-11-23 09:38:32 +00:00
|
|
|
from homeassistant.core import CALLBACK_TYPE, callback
|
2023-08-11 02:04:26 +00:00
|
|
|
from homeassistant.helpers.device_registry import CONNECTION_NETWORK_MAC, DeviceInfo
|
|
|
|
from homeassistant.helpers.entity import Entity
|
2022-08-16 18:08:35 +00:00
|
|
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
|
|
|
|
|
|
|
from .const import DOMAIN
|
|
|
|
from .coordinator import FullyKioskDataUpdateCoordinator
|
|
|
|
|
|
|
|
|
2023-06-27 11:34:07 +00:00
|
|
|
def valid_global_mac_address(mac: str | None) -> bool:
|
|
|
|
"""Check if a MAC address is valid, non-locally administered address."""
|
|
|
|
if not isinstance(mac, str):
|
|
|
|
return False
|
|
|
|
try:
|
|
|
|
first_octet = int(mac.split(":")[0], 16)
|
|
|
|
# If the second least-significant bit is set, it's a locally administered address, should not be used as an ID
|
|
|
|
return not bool(first_octet & 0x2)
|
|
|
|
except ValueError:
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2022-08-16 18:08:35 +00:00
|
|
|
class FullyKioskEntity(CoordinatorEntity[FullyKioskDataUpdateCoordinator], Entity):
|
|
|
|
"""Defines a Fully Kiosk Browser entity."""
|
|
|
|
|
|
|
|
_attr_has_entity_name = True
|
|
|
|
|
|
|
|
def __init__(self, coordinator: FullyKioskDataUpdateCoordinator) -> None:
|
|
|
|
"""Initialize the Fully Kiosk Browser entity."""
|
|
|
|
super().__init__(coordinator=coordinator)
|
2023-11-13 16:40:57 +00:00
|
|
|
|
|
|
|
url = URL.build(
|
|
|
|
scheme="https" if coordinator.use_ssl else "http",
|
|
|
|
host=coordinator.data["ip4"],
|
|
|
|
port=2323,
|
|
|
|
)
|
|
|
|
|
2023-02-10 08:19:09 +00:00
|
|
|
device_info = DeviceInfo(
|
2022-08-16 18:08:35 +00:00
|
|
|
identifiers={(DOMAIN, coordinator.data["deviceID"])},
|
|
|
|
name=coordinator.data["deviceName"],
|
|
|
|
manufacturer=coordinator.data["deviceManufacturer"],
|
|
|
|
model=coordinator.data["deviceModel"],
|
|
|
|
sw_version=coordinator.data["appVersionName"],
|
2023-11-13 16:40:57 +00:00
|
|
|
configuration_url=str(url),
|
2022-08-16 18:08:35 +00:00
|
|
|
)
|
2023-06-27 11:34:07 +00:00
|
|
|
if "Mac" in coordinator.data and valid_global_mac_address(
|
|
|
|
coordinator.data["Mac"]
|
|
|
|
):
|
2023-11-02 20:59:25 +00:00
|
|
|
device_info[ATTR_CONNECTIONS] = {
|
2023-02-10 08:19:09 +00:00
|
|
|
(CONNECTION_NETWORK_MAC, coordinator.data["Mac"])
|
|
|
|
}
|
|
|
|
self._attr_device_info = device_info
|
2023-11-23 09:38:32 +00:00
|
|
|
|
|
|
|
async def mqtt_subscribe(
|
|
|
|
self, event: str | None, event_callback: CALLBACK_TYPE
|
|
|
|
) -> CALLBACK_TYPE | None:
|
|
|
|
"""Subscribe to MQTT for a given event."""
|
|
|
|
data = self.coordinator.data
|
|
|
|
if (
|
|
|
|
event is None
|
|
|
|
or not mqtt.mqtt_config_entry_enabled(self.hass)
|
|
|
|
or not data["settings"]["mqttEnabled"]
|
|
|
|
):
|
|
|
|
return None
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def message_callback(message: mqtt.ReceiveMessage) -> None:
|
|
|
|
payload = json.loads(message.payload)
|
|
|
|
event_callback(**payload)
|
|
|
|
|
|
|
|
topic_template = data["settings"]["mqttEventTopic"]
|
|
|
|
topic = (
|
|
|
|
topic_template.replace("$appId", "fully")
|
|
|
|
.replace("$event", event)
|
|
|
|
.replace("$deviceId", data["deviceID"])
|
|
|
|
)
|
|
|
|
return await mqtt.async_subscribe(self.hass, topic, message_callback)
|