Files
alexei.dolgolyov 705616f8b0 feat(playlists): expose scene playlists as Home Assistant entities
One device per scene playlist (model "Scene Playlist"), each with:
- a switch (start/stop; the server cycles one playlist at a time, so starting one stops any other)
- a "Current Scene" sensor that resolves the active preset name from scene_presets while this playlist is cycling
- an "Items" diagnostic sensor (configured scene count; no state_class)

The coordinator reads scene_playlists plus the flat playlist_state from the /api/v1/snapshot payload and gains start_playlist()/stop_playlist(); __init__ registers and prunes per-playlist devices and reloads on playlist-id changes; the event listener also refreshes on the playlist_state_changed WS event. Shared device/lookup/running-state plumbing lives in a new LedGrabPlaylistEntity base (entity.py) used by both the switch and the sensors. Adds en/ru translation keys.

Note: this also lands the in-progress coordinator migration from the per-request fan-out to the single /api/v1/snapshot poll that was already present in the working tree.

Requires the led-grab server /api/v1/snapshot to emit scene_playlists + playlist_state (companion server change, tracked separately).
2026-06-08 15:59:42 +03:00

701 lines
23 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
from .entity import LedGrabPlaylistEntity
_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))
for playlist in coordinator.data.get("scene_playlists", []):
entities.append(
PlaylistCurrentSceneSensor(coordinator, playlist["id"], entry.entry_id)
)
entities.append(
PlaylistItemsSensor(coordinator, playlist["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 _PlaylistSensorBase(LedGrabPlaylistEntity, SensorEntity):
"""Sensor attached to a scene-playlist device.
Device/availability/lookup plumbing lives in :class:`LedGrabPlaylistEntity`
so it stays in sync with the playlist switch.
"""
class PlaylistCurrentSceneSensor(_PlaylistSensorBase):
"""Name of the scene preset this playlist is currently holding.
Only this playlist's value populates while it is the active (cycling) one;
every other playlist's sensor reads ``None`` (idle). The value advances as
the engine steps to the next item.
"""
_attr_icon = "mdi:playlist-star"
def __init__(
self,
coordinator: LedGrabCoordinator,
playlist_id: str,
entry_id: str,
) -> None:
super().__init__(coordinator, playlist_id, entry_id)
self._attr_unique_id = f"{playlist_id}_current_scene"
self._attr_translation_key = "playlist_current_scene"
@property
def native_value(self) -> str | None:
state = self._running_state()
if not state:
return None
preset_id = state.get("current_preset_id")
if not preset_id:
return None
return self._resolve_preset_name(preset_id)
@property
def extra_state_attributes(self) -> dict[str, Any]:
state = self._running_state()
if not state:
return {}
return {
"current_preset_id": state.get("current_preset_id"),
"current_index": state.get("current_index"),
"item_count": state.get("item_count"),
}
def _resolve_preset_name(self, preset_id: str) -> str:
"""Map a scene-preset id to its display name (fall back to the id)."""
if self.coordinator.data:
for preset in self.coordinator.data.get("scene_presets", []):
if preset.get("id") == preset_id:
return preset.get("name", preset_id)
return preset_id
class PlaylistItemsSensor(_PlaylistSensorBase):
"""Number of scene presets queued in a playlist (static config stat).
No ``state_class`` — this is a configuration cardinality that only changes
when the playlist is edited, so it shouldn't be enrolled in long-term
statistics.
"""
_attr_icon = "mdi:playlist-music"
_attr_entity_category = EntityCategory.DIAGNOSTIC
def __init__(
self,
coordinator: LedGrabCoordinator,
playlist_id: str,
entry_id: str,
) -> None:
super().__init__(coordinator, playlist_id, entry_id)
self._attr_unique_id = f"{playlist_id}_items"
self._attr_translation_key = "playlist_items"
@property
def native_value(self) -> int | None:
playlist = self._get_playlist()
if playlist is None:
return None
return len(playlist.get("items") or [])
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