Files
ledgrab-haos-integration/custom_components/ledgrab/sensor.py
T
alexei.dolgolyov a666d9eb9c feat: server telemetry, update entity, sync-clock controls
- Server device exposing CPU/RAM/GPU/temperature/battery sensors via
  /api/v1/system/performance, plus last-restart timestamp (cached with
  jitter threshold so the recorder doesn't see poll wobble) and version.
- Update entity backed by /api/v1/system/update — installs via
  /apply, hides the install button when the server reports
  can_auto_update=false.
- Sync-clock entities: reset button, speed number, running switch, and
  the event listener now refreshes on entity_changed events too.
- Bump manifest to 0.4.0.
2026-04-27 01:35:42 +03:00

605 lines
20 KiB
Python

"""Sensor platform for LED Screen Controller."""
from __future__ import annotations
from datetime import datetime
import logging
from typing import Any
from homeassistant.components.sensor import (
SensorDeviceClass,
SensorEntity,
SensorStateClass,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import (
PERCENTAGE,
UnitOfInformation,
UnitOfTemperature,
UnitOfTime,
)
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity import EntityCategory
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.update_coordinator import CoordinatorEntity
from .const import (
DOMAIN,
TARGET_TYPE_HA_LIGHT,
DATA_COORDINATOR,
)
from .coordinator import LedGrabCoordinator
_LOGGER = logging.getLogger(__name__)
async def async_setup_entry(
hass: HomeAssistant,
entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up LED Screen Controller sensors."""
data = hass.data[DOMAIN][entry.entry_id]
coordinator: LedGrabCoordinator = data[DATA_COORDINATOR]
entities: list[SensorEntity] = []
if coordinator.data and "targets" in coordinator.data:
for target_id, target_data in coordinator.data["targets"].items():
entities.append(LedGrabFPSSensor(coordinator, target_id, entry.entry_id))
entities.append(
LedGrabStatusSensor(coordinator, target_id, entry.entry_id)
)
# Add mapped lights sensor for HA Light targets
info = target_data["info"]
if info.get("target_type") == TARGET_TYPE_HA_LIGHT:
entities.append(HALightMappedLightsSensor(coordinator, target_id, entry.entry_id))
if coordinator.data:
for clock in coordinator.data.get("sync_clocks", []):
entities.append(SyncClockElapsedSensor(coordinator, clock["id"], entry.entry_id))
entities.extend(_build_server_sensors(coordinator, entry.entry_id))
async_add_entities(entities)
def _build_server_sensors(
coordinator: LedGrabCoordinator,
entry_id: str,
) -> list[SensorEntity]:
"""Build the server-level telemetry sensors.
GPU/CPU-temp/battery sensors are only registered when the first refresh
actually carries those readings — we don't want phantom unavailable
entities on hardware that doesn't support a metric. A reload picks up
metrics that come online later (e.g. user installs LibreHardwareMonitor).
"""
system = (coordinator.data or {}).get("system") or {}
perf = system.get("performance") or {}
health = system.get("health")
sensors: list[SensorEntity] = [
ServerCpuPercentSensor(coordinator, entry_id),
ServerRamPercentSensor(coordinator, entry_id),
ServerAppCpuPercentSensor(coordinator, entry_id),
ServerAppRamSensor(coordinator, entry_id),
ServerVersionSensor(coordinator, entry_id),
]
if health is not None:
# Only meaningful when /health actually responded on first refresh.
sensors.append(ServerLastRestartSensor(coordinator, entry_id))
if perf.get("gpu") is not None:
sensors.append(ServerGpuUtilizationSensor(coordinator, entry_id))
sensors.append(ServerGpuTemperatureSensor(coordinator, entry_id))
if perf.get("cpu_temp_c") is not None:
sensors.append(ServerCpuTemperatureSensor(coordinator, entry_id))
if perf.get("battery_percent") is not None:
sensors.append(ServerBatterySensor(coordinator, entry_id))
return sensors
class LedGrabFPSSensor(CoordinatorEntity, SensorEntity):
"""FPS sensor for a LED Screen Controller target."""
_attr_has_entity_name = True
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_native_unit_of_measurement = "FPS"
_attr_icon = "mdi:speedometer"
_attr_suggested_display_precision = 1
def __init__(
self,
coordinator: LedGrabCoordinator,
target_id: str,
entry_id: str,
) -> None:
"""Initialize the sensor."""
super().__init__(coordinator)
self._target_id = target_id
self._entry_id = entry_id
self._attr_unique_id = f"{target_id}_fps"
self._attr_translation_key = "fps"
@property
def device_info(self) -> dict[str, Any]:
"""Return device information."""
return {"identifiers": {(DOMAIN, self._target_id)}}
@property
def native_value(self) -> float | None:
"""Return the FPS value."""
target_data = self._get_target_data()
if not target_data or not target_data.get("state"):
return None
state = target_data["state"]
if not state.get("processing"):
return None
return state.get("fps_actual")
@property
def extra_state_attributes(self) -> dict[str, Any]:
"""Return additional attributes."""
target_data = self._get_target_data()
if not target_data or not target_data.get("state"):
return {}
return {"fps_target": target_data["state"].get("fps_target")}
@property
def available(self) -> bool:
"""Return if entity is available."""
return self._get_target_data() is not None
def _get_target_data(self) -> dict[str, Any] | None:
if not self.coordinator.data:
return None
return self.coordinator.data.get("targets", {}).get(self._target_id)
class LedGrabStatusSensor(CoordinatorEntity, SensorEntity):
"""Status sensor for a LED Screen Controller target."""
_attr_has_entity_name = True
_attr_icon = "mdi:information-outline"
_attr_device_class = SensorDeviceClass.ENUM
_attr_options = ["processing", "idle", "error", "unavailable"]
def __init__(
self,
coordinator: LedGrabCoordinator,
target_id: str,
entry_id: str,
) -> None:
"""Initialize the sensor."""
super().__init__(coordinator)
self._target_id = target_id
self._entry_id = entry_id
self._attr_unique_id = f"{target_id}_status"
self._attr_translation_key = "status"
@property
def device_info(self) -> dict[str, Any]:
"""Return device information."""
return {"identifiers": {(DOMAIN, self._target_id)}}
@property
def native_value(self) -> str:
"""Return the status."""
target_data = self._get_target_data()
if not target_data:
return "unavailable"
state = target_data.get("state")
if not state:
return "unavailable"
if state.get("processing"):
errors = state.get("errors", [])
if errors:
return "error"
return "processing"
return "idle"
@property
def available(self) -> bool:
"""Return if entity is available."""
return self._get_target_data() is not None
def _get_target_data(self) -> dict[str, Any] | None:
if not self.coordinator.data:
return None
return self.coordinator.data.get("targets", {}).get(self._target_id)
class SyncClockElapsedSensor(CoordinatorEntity, SensorEntity):
"""Elapsed seconds since the sync clock last reset.
Disabled by default — the value advances on every coordinator poll
(~3s) while the clock is running, which would write a recorder row
each refresh. Power users can enable it explicitly.
"""
_attr_has_entity_name = True
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_device_class = SensorDeviceClass.DURATION
_attr_native_unit_of_measurement = UnitOfTime.SECONDS
_attr_icon = "mdi:timer-outline"
_attr_entity_category = EntityCategory.DIAGNOSTIC
_attr_entity_registry_enabled_default = False
_attr_suggested_display_precision = 1
def __init__(
self,
coordinator: LedGrabCoordinator,
clock_id: str,
entry_id: str,
) -> None:
super().__init__(coordinator)
self._clock_id = clock_id
self._entry_id = entry_id
self._attr_unique_id = f"{clock_id}_elapsed"
self._attr_translation_key = "sync_clock_elapsed"
@property
def device_info(self) -> dict[str, Any]:
return {"identifiers": {(DOMAIN, self._clock_id)}}
@property
def native_value(self) -> float | None:
clock = self._get_clock()
if not clock:
return None
elapsed = clock.get("elapsed_time")
return float(elapsed) if elapsed is not None else None
@property
def available(self) -> bool:
return self._get_clock() is not None
def _get_clock(self) -> dict[str, Any] | None:
if not self.coordinator.data:
return None
for clock in self.coordinator.data.get("sync_clocks", []):
if clock.get("id") == self._clock_id:
return clock
return None
class HALightMappedLightsSensor(CoordinatorEntity, SensorEntity):
"""Sensor showing the number of mapped HA lights for an HA Light target."""
_attr_has_entity_name = True
_attr_icon = "mdi:lightbulb-group"
def __init__(
self,
coordinator: LedGrabCoordinator,
target_id: str,
entry_id: str,
) -> None:
"""Initialize the sensor."""
super().__init__(coordinator)
self._target_id = target_id
self._entry_id = entry_id
self._attr_unique_id = f"{target_id}_mapped_lights"
self._attr_translation_key = "mapped_lights"
@property
def device_info(self) -> dict[str, Any]:
"""Return device information."""
return {"identifiers": {(DOMAIN, self._target_id)}}
@property
def native_value(self) -> int | None:
"""Return the number of mapped lights."""
target_data = self._get_target_data()
if not target_data:
return None
mappings = target_data.get("info", {}).get("light_mappings", [])
return len(mappings)
@property
def extra_state_attributes(self) -> dict[str, Any]:
"""Return light mapping details as attributes."""
target_data = self._get_target_data()
if not target_data:
return {}
mappings = target_data.get("info", {}).get("light_mappings", [])
entity_ids = [m.get("entity_id", "") for m in mappings]
return {
"entity_ids": entity_ids,
"mappings": [
{
"entity_id": m.get("entity_id", ""),
"led_start": m.get("led_start", 0),
"led_end": m.get("led_end", -1),
"brightness_scale": m.get("brightness_scale", 1.0),
}
for m in mappings
],
}
@property
def available(self) -> bool:
"""Return if entity is available."""
return self._get_target_data() is not None
def _get_target_data(self) -> dict[str, Any] | None:
if not self.coordinator.data:
return None
return self.coordinator.data.get("targets", {}).get(self._target_id)
# ─────────────────────────── Server-level sensors ───────────────────────────
class _ServerSensorBase(CoordinatorEntity, SensorEntity):
"""Shared plumbing for sensors attached to the synthetic Server device."""
_attr_has_entity_name = True
_attr_entity_category = EntityCategory.DIAGNOSTIC
def __init__(
self,
coordinator: LedGrabCoordinator,
entry_id: str,
unique_suffix: str,
translation_key: str,
) -> None:
super().__init__(coordinator)
self._entry_id = entry_id
self._attr_unique_id = f"{entry_id}_server_{unique_suffix}"
self._attr_translation_key = translation_key
@property
def device_info(self) -> dict[str, Any]:
return {"identifiers": {(DOMAIN, f"{self._entry_id}_server")}}
def _perf(self) -> dict[str, Any] | None:
if not self.coordinator.data:
return None
system = self.coordinator.data.get("system") or {}
return system.get("performance")
@property
def available(self) -> bool:
# Performance metrics are missing whenever /api/v1/system/performance
# fails — surface that as unavailable instead of stale numbers.
return self._perf() is not None
class ServerCpuPercentSensor(_ServerSensorBase):
"""System-wide CPU usage on the led-grab server."""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_native_unit_of_measurement = PERCENTAGE
_attr_icon = "mdi:cpu-64-bit"
_attr_suggested_display_precision = 1
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "cpu_percent", "server_cpu_percent")
@property
def native_value(self) -> float | None:
perf = self._perf()
return None if perf is None else perf.get("cpu_percent")
@property
def extra_state_attributes(self) -> dict[str, Any]:
perf = self._perf() or {}
return {"cpu_name": perf.get("cpu_name")}
class ServerRamPercentSensor(_ServerSensorBase):
"""System-wide RAM usage on the led-grab server."""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_native_unit_of_measurement = PERCENTAGE
_attr_icon = "mdi:memory"
_attr_suggested_display_precision = 1
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "ram_percent", "server_ram_percent")
@property
def native_value(self) -> float | None:
perf = self._perf()
return None if perf is None else perf.get("ram_percent")
@property
def extra_state_attributes(self) -> dict[str, Any]:
perf = self._perf() or {}
return {
"used_mb": perf.get("ram_used_mb"),
"total_mb": perf.get("ram_total_mb"),
}
class ServerAppCpuPercentSensor(_ServerSensorBase):
"""CPU usage of the led-grab process itself."""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_native_unit_of_measurement = PERCENTAGE
_attr_icon = "mdi:application-cog"
_attr_suggested_display_precision = 1
_attr_entity_registry_enabled_default = False
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "app_cpu_percent", "server_app_cpu_percent")
@property
def native_value(self) -> float | None:
perf = self._perf()
return None if perf is None else perf.get("app_cpu_percent")
class ServerAppRamSensor(_ServerSensorBase):
"""Resident memory of the led-grab process."""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_device_class = SensorDeviceClass.DATA_SIZE
_attr_native_unit_of_measurement = UnitOfInformation.MEGABYTES
_attr_icon = "mdi:memory"
_attr_suggested_display_precision = 1
_attr_entity_registry_enabled_default = False
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "app_ram_mb", "server_app_ram")
@property
def native_value(self) -> float | None:
perf = self._perf()
return None if perf is None else perf.get("app_ram_mb")
class ServerGpuUtilizationSensor(_ServerSensorBase):
"""GPU utilization (NVML)."""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_native_unit_of_measurement = PERCENTAGE
_attr_icon = "mdi:expansion-card"
_attr_suggested_display_precision = 1
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "gpu_utilization", "server_gpu_utilization")
@property
def native_value(self) -> float | None:
perf = self._perf()
if perf is None:
return None
gpu = perf.get("gpu") or {}
return gpu.get("utilization")
@property
def extra_state_attributes(self) -> dict[str, Any]:
perf = self._perf() or {}
gpu = perf.get("gpu") or {}
return {
"name": gpu.get("name"),
"memory_used_mb": gpu.get("memory_used_mb"),
"memory_total_mb": gpu.get("memory_total_mb"),
"app_memory_mb": gpu.get("app_memory_mb"),
}
@property
def available(self) -> bool:
perf = self._perf()
return perf is not None and perf.get("gpu") is not None
class ServerGpuTemperatureSensor(_ServerSensorBase):
"""GPU temperature (NVML)."""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_device_class = SensorDeviceClass.TEMPERATURE
_attr_native_unit_of_measurement = UnitOfTemperature.CELSIUS
_attr_suggested_display_precision = 0
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "gpu_temp", "server_gpu_temp")
@property
def native_value(self) -> float | None:
perf = self._perf()
if perf is None:
return None
gpu = perf.get("gpu") or {}
return gpu.get("temperature_c")
@property
def available(self) -> bool:
perf = self._perf()
return perf is not None and (perf.get("gpu") or {}).get("temperature_c") is not None
class ServerCpuTemperatureSensor(_ServerSensorBase):
"""Hottest CPU/SoC thermal-zone reading.
Only registered when the first refresh actually reports a temperature —
Windows needs LibreHardwareMonitor / OpenHardwareMonitor publishing WMI
sensors, otherwise the field stays null.
"""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_device_class = SensorDeviceClass.TEMPERATURE
_attr_native_unit_of_measurement = UnitOfTemperature.CELSIUS
_attr_suggested_display_precision = 0
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "cpu_temp", "server_cpu_temp")
@property
def native_value(self) -> float | None:
perf = self._perf()
return None if perf is None else perf.get("cpu_temp_c")
class ServerBatterySensor(_ServerSensorBase):
"""Battery charge level (laptops / portable hosts)."""
_attr_state_class = SensorStateClass.MEASUREMENT
_attr_device_class = SensorDeviceClass.BATTERY
_attr_native_unit_of_measurement = PERCENTAGE
_attr_suggested_display_precision = 0
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "battery", "server_battery")
@property
def native_value(self) -> float | None:
perf = self._perf()
return None if perf is None else perf.get("battery_percent")
@property
def extra_state_attributes(self) -> dict[str, Any]:
perf = self._perf() or {}
return {"temperature_c": perf.get("battery_temp_c")}
class ServerLastRestartSensor(_ServerSensorBase):
"""Timestamp of the last server start.
Derived from ``/health.uptime_seconds``; the coordinator caches the
computed boot time and only refreshes it when the candidate moves more
than a few seconds, so the recorder doesn't get poll-jitter writes.
"""
_attr_device_class = SensorDeviceClass.TIMESTAMP
_attr_icon = "mdi:restart"
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "last_restart", "server_last_restart")
@property
def native_value(self) -> datetime | None:
if not self.coordinator.data:
return None
system = self.coordinator.data.get("system") or {}
return system.get("boot_time")
@property
def available(self) -> bool:
return self.native_value is not None
class ServerVersionSensor(_ServerSensorBase):
"""Reported led-grab server version."""
_attr_icon = "mdi:tag-outline"
def __init__(self, coordinator: LedGrabCoordinator, entry_id: str) -> None:
super().__init__(coordinator, entry_id, "version", "server_version")
@property
def native_value(self) -> str | None:
version = self.coordinator.server_version
return version if version and version != "unknown" else None
@property
def available(self) -> bool:
return self.native_value is not None