Coverage for custom_components/supernotify/transports/media_player.py: 100%
43 statements
« prev ^ index » next coverage.py v7.10.6, created at 2026-04-01 15:06 +0000
« prev ^ index » next coverage.py v7.10.6, created at 2026-04-01 15:06 +0000
1import logging
2import urllib.parse
3from typing import TYPE_CHECKING, Any
5from homeassistant.const import (
6 ATTR_ENTITY_ID,
7)
9from custom_components.supernotify.const import (
10 ATTR_MEDIA,
11 ATTR_MEDIA_SNAPSHOT_URL,
12 OPTION_TARGET_CATEGORIES,
13 OPTION_TARGET_SELECT,
14 TRANSPORT_MEDIA,
15)
16from custom_components.supernotify.model import DebugTrace, TransportConfig, TransportFeature
17from custom_components.supernotify.transport import Transport
19if TYPE_CHECKING:
20 from custom_components.supernotify.envelope import Envelope
22RE_VALID_MEDIA_PLAYER = r"media_player\.[A-Za-z0-9_]+"
24_LOGGER = logging.getLogger(__name__)
27class MediaPlayerTransport(Transport):
28 name = TRANSPORT_MEDIA
30 def __init__(self, *args: Any, **kwargs: Any) -> None:
31 super().__init__(*args, **kwargs)
33 @property
34 def supported_features(self) -> TransportFeature:
35 return TransportFeature.IMAGES | TransportFeature.VIDEO
37 @property
38 def default_config(self) -> TransportConfig:
39 config = TransportConfig()
40 config.delivery_defaults.action = "media_player.play_media"
41 config.delivery_defaults.options = {
42 OPTION_TARGET_SELECT: [RE_VALID_MEDIA_PLAYER],
43 OPTION_TARGET_CATEGORIES: [ATTR_ENTITY_ID],
44 }
45 return config
47 async def deliver(self, envelope: Envelope, debug_trace: DebugTrace | None = None) -> bool: # noqa: ARG002
48 _LOGGER.debug("SUPERNOTIFY notify_media: %s", envelope.data)
50 data: dict[str, Any] = envelope.data or {}
51 media_players: list[str] = envelope.target.entity_ids or []
52 media_type: str = data.get("media_content_type", "image")
53 if not media_players:
54 _LOGGER.debug("SUPERNOTIFY skipping media show, no targets")
55 return False
57 snapshot_url = data.get(ATTR_MEDIA, {}).get(ATTR_MEDIA_SNAPSHOT_URL)
58 if snapshot_url is None:
59 # fallback to older idiosyncratic way for backward compatibility
60 snapshot_url = data.get(ATTR_MEDIA_SNAPSHOT_URL)
61 if snapshot_url is None:
62 _LOGGER.debug("SUPERNOTIFY skipping media player, no snapshot url")
63 return False
64 # absolutize relative URL for external URl, probably preferred by Alexa Show etc
65 snapshot_url = urllib.parse.urljoin(self.hass_api.external_url, snapshot_url)
67 action_data: dict[str, Any] = {"media": {"media_content_id": snapshot_url, "media_content_type": media_type}}
68 if data and data.get("announce"):
69 action_data["announce"] = data.get("announce")
70 if data and data.get("enqueue"):
71 action_data["enqueue"] = data.get("enqueue")
73 return await self.call_action(envelope, action_data=action_data, target_data={ATTR_ENTITY_ID: media_players})