9d277276b8
Adds Home Assistant entities for the foreground-process feature shipped in the media server, plus migrates existing display entities to use HA translation keys (strings.json / translations/*) so per-language UI text flows through the standard locale mechanism. Foreground entities (all share one HA "Foreground" device linked to the hub via via_device): - sensor.foreground_process — process name as state + full payload (pid, exec path, window title, fullscreen flag, monitor, geometry, is_browser, browser_page_title, browser_url, error) as attributes - sensor.window_title, sensor.pid, sensor.foreground_monitor, sensor.process_started (TIMESTAMP device class) - binary_sensor.fullscreen, binary_sensor.minimized Data flow: - ForegroundCoordinator polls GET /api/foreground every 5s (HTTP fallback) - media_player's WebSocket receiver forwards `foreground` / `foreground_update` push frames into the coordinator via apply_websocket_snapshot, so sensors update in near-real-time when WS is connected and fall back to polling otherwise Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
225 lines
7.3 KiB
Python
225 lines
7.3 KiB
Python
"""The Remote Media Player integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
from typing import Any
|
|
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import Platform
|
|
from homeassistant.core import HomeAssistant, ServiceCall
|
|
from homeassistant.helpers import config_validation as cv
|
|
|
|
from .api_client import MediaServerClient, MediaServerError
|
|
from .const import (
|
|
ATTR_FILE_PATH,
|
|
ATTR_SCRIPT_NAME,
|
|
ATTR_SCRIPT_PARAMS,
|
|
CONF_HOST,
|
|
CONF_PORT,
|
|
CONF_TOKEN,
|
|
DOMAIN,
|
|
SERVICE_EXECUTE_SCRIPT,
|
|
SERVICE_PLAY_MEDIA_FILE,
|
|
)
|
|
from .display_coordinator import DisplayCoordinator
|
|
from .foreground_coordinator import ForegroundCoordinator
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
PLATFORMS: list[Platform] = [
|
|
Platform.MEDIA_PLAYER,
|
|
Platform.BUTTON,
|
|
Platform.NUMBER,
|
|
Platform.SWITCH,
|
|
Platform.SENSOR,
|
|
Platform.BINARY_SENSOR,
|
|
Platform.SELECT,
|
|
]
|
|
|
|
# Service schema for execute_script
|
|
SERVICE_EXECUTE_SCRIPT_SCHEMA = vol.Schema(
|
|
{
|
|
vol.Required(ATTR_SCRIPT_NAME): cv.string,
|
|
vol.Optional(ATTR_SCRIPT_PARAMS, default={}): dict,
|
|
}
|
|
)
|
|
|
|
# Service schema for play_media_file
|
|
SERVICE_PLAY_MEDIA_FILE_SCHEMA = vol.Schema(
|
|
{
|
|
vol.Required(ATTR_FILE_PATH): cv.string,
|
|
}
|
|
)
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Set up Remote Media Player from a config entry.
|
|
|
|
Args:
|
|
hass: Home Assistant instance
|
|
entry: Config entry
|
|
|
|
Returns:
|
|
True if setup was successful
|
|
"""
|
|
_LOGGER.debug("Setting up Remote Media Player: %s", entry.entry_id)
|
|
|
|
# Create API client
|
|
client = MediaServerClient(
|
|
host=entry.data[CONF_HOST],
|
|
port=entry.data[CONF_PORT],
|
|
token=entry.data[CONF_TOKEN],
|
|
)
|
|
|
|
# Verify connection
|
|
if not await client.check_connection():
|
|
_LOGGER.error("Failed to connect to Media Server")
|
|
await client.close()
|
|
return False
|
|
|
|
# Create the shared display coordinator BEFORE platform setup so each
|
|
# display platform's async_setup_entry can register against the same
|
|
# data source instead of polling /api/display/monitors on its own.
|
|
display_coordinator = DisplayCoordinator(hass, client)
|
|
try:
|
|
await display_coordinator.async_config_entry_first_refresh()
|
|
except Exception as err: # noqa: BLE001 - first refresh wraps its own errors
|
|
_LOGGER.warning("Initial display monitor fetch failed, will retry: %s", err)
|
|
|
|
# Foreground coordinator — shared by sensor + binary_sensor platforms and
|
|
# nudged by the media-player WebSocket receiver when it gets a push.
|
|
foreground_coordinator = ForegroundCoordinator(hass, client)
|
|
try:
|
|
await foreground_coordinator.async_config_entry_first_refresh()
|
|
except Exception as err: # noqa: BLE001
|
|
_LOGGER.warning("Initial foreground fetch failed, will retry: %s", err)
|
|
|
|
# Store client in hass.data
|
|
hass.data.setdefault(DOMAIN, {})
|
|
hass.data[DOMAIN][entry.entry_id] = {
|
|
"client": client,
|
|
"display_coordinator": display_coordinator,
|
|
"foreground_coordinator": foreground_coordinator,
|
|
}
|
|
|
|
# Register services if not already registered
|
|
if not hass.services.has_service(DOMAIN, SERVICE_EXECUTE_SCRIPT):
|
|
async def async_execute_script(call: ServiceCall) -> dict[str, Any]:
|
|
"""Execute a script on the media server."""
|
|
script_name = call.data[ATTR_SCRIPT_NAME]
|
|
script_params = call.data.get(ATTR_SCRIPT_PARAMS, {})
|
|
|
|
_LOGGER.debug(
|
|
"Executing script '%s' with params: %s", script_name, script_params
|
|
)
|
|
|
|
# Get all clients and execute on all of them
|
|
results = {}
|
|
for entry_id, data in hass.data[DOMAIN].items():
|
|
client: MediaServerClient = data["client"]
|
|
try:
|
|
result = await client.execute_script(script_name, script_params)
|
|
results[entry_id] = result
|
|
_LOGGER.info(
|
|
"Script '%s' executed on %s: success=%s",
|
|
script_name,
|
|
entry_id,
|
|
result.get("success", False),
|
|
)
|
|
except MediaServerError as err:
|
|
_LOGGER.error(
|
|
"Failed to execute script '%s' on %s: %s",
|
|
script_name,
|
|
entry_id,
|
|
err,
|
|
)
|
|
results[entry_id] = {"success": False, "error": str(err)}
|
|
|
|
return results
|
|
|
|
hass.services.async_register(
|
|
DOMAIN,
|
|
SERVICE_EXECUTE_SCRIPT,
|
|
async_execute_script,
|
|
schema=SERVICE_EXECUTE_SCRIPT_SCHEMA,
|
|
)
|
|
|
|
# Register play_media_file service if not already registered
|
|
if not hass.services.has_service(DOMAIN, SERVICE_PLAY_MEDIA_FILE):
|
|
async def async_play_media_file(call: ServiceCall) -> None:
|
|
"""Handle play_media_file service call."""
|
|
file_path = call.data[ATTR_FILE_PATH]
|
|
_LOGGER.debug("Service play_media_file called with path: %s", file_path)
|
|
|
|
# Execute on all configured media server instances
|
|
for entry_id, data in hass.data[DOMAIN].items():
|
|
client: MediaServerClient = data["client"]
|
|
try:
|
|
await client.play_media_file(file_path)
|
|
_LOGGER.info("Started playback of %s on %s", file_path, entry_id)
|
|
except MediaServerError as err:
|
|
_LOGGER.error("Failed to play %s on %s: %s", file_path, entry_id, err)
|
|
|
|
hass.services.async_register(
|
|
DOMAIN,
|
|
SERVICE_PLAY_MEDIA_FILE,
|
|
async_play_media_file,
|
|
schema=SERVICE_PLAY_MEDIA_FILE_SCHEMA,
|
|
)
|
|
|
|
# Forward setup to platforms
|
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
|
|
|
# Register update listener for options
|
|
entry.async_on_unload(entry.add_update_listener(async_update_options))
|
|
|
|
return True
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Unload a config entry.
|
|
|
|
Args:
|
|
hass: Home Assistant instance
|
|
entry: Config entry
|
|
|
|
Returns:
|
|
True if unload was successful
|
|
"""
|
|
_LOGGER.debug("Unloading Remote Media Player: %s", entry.entry_id)
|
|
|
|
# Unload platforms
|
|
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
|
|
|
if unload_ok:
|
|
# Close client and remove data
|
|
data = hass.data[DOMAIN].pop(entry.entry_id)
|
|
|
|
# Shutdown coordinator (WebSocket cleanup)
|
|
if "coordinator" in data:
|
|
await data["coordinator"].async_shutdown()
|
|
|
|
# Close HTTP client
|
|
await data["client"].close()
|
|
|
|
# Remove services if this was the last entry
|
|
if not hass.data[DOMAIN]:
|
|
hass.services.async_remove(DOMAIN, SERVICE_EXECUTE_SCRIPT)
|
|
hass.services.async_remove(DOMAIN, SERVICE_PLAY_MEDIA_FILE)
|
|
|
|
return unload_ok
|
|
|
|
|
|
async def async_update_options(hass: HomeAssistant, entry: ConfigEntry) -> None:
|
|
"""Handle options update.
|
|
|
|
Args:
|
|
hass: Home Assistant instance
|
|
entry: Config entry
|
|
"""
|
|
_LOGGER.debug("Options updated for: %s", entry.entry_id)
|
|
await hass.config_entries.async_reload(entry.entry_id)
|