2020-09-09 21:22:26 +00:00
|
|
|
"""Support for media browsing."""
|
2022-01-24 16:34:09 +00:00
|
|
|
from __future__ import annotations
|
2020-09-09 21:22:26 +00:00
|
|
|
|
2022-01-24 16:34:09 +00:00
|
|
|
from collections.abc import Callable
|
|
|
|
from functools import partial
|
|
|
|
|
|
|
|
from homeassistant.components import media_source
|
2022-09-12 18:06:27 +00:00
|
|
|
from homeassistant.components.media_player import (
|
|
|
|
BrowseError,
|
|
|
|
BrowseMedia,
|
|
|
|
MediaClass,
|
|
|
|
MediaType,
|
2020-09-09 21:22:26 +00:00
|
|
|
)
|
2022-01-24 16:34:09 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.network import is_internal_request
|
|
|
|
|
|
|
|
from .coordinator import RokuDataUpdateCoordinator
|
2022-02-12 02:52:31 +00:00
|
|
|
from .helpers import format_channel_name
|
2020-09-09 21:22:26 +00:00
|
|
|
|
|
|
|
CONTENT_TYPE_MEDIA_CLASS = {
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.APP: MediaClass.APP,
|
|
|
|
MediaType.APPS: MediaClass.APP,
|
|
|
|
MediaType.CHANNEL: MediaClass.CHANNEL,
|
|
|
|
MediaType.CHANNELS: MediaClass.CHANNEL,
|
2020-09-19 16:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CONTAINER_TYPES_SPECIFIC_MEDIA_CLASS = {
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.APPS: MediaClass.DIRECTORY,
|
|
|
|
MediaType.CHANNELS: MediaClass.DIRECTORY,
|
2020-09-09 21:22:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PLAYABLE_MEDIA_TYPES = [
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.APP,
|
|
|
|
MediaType.CHANNEL,
|
2020-09-09 21:22:26 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
EXPANDABLE_MEDIA_TYPES = [
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.APPS,
|
|
|
|
MediaType.CHANNELS,
|
2020-09-09 21:22:26 +00:00
|
|
|
]
|
|
|
|
|
2022-01-24 16:34:09 +00:00
|
|
|
GetBrowseImageUrlType = Callable[[str, str, "str | None"], str]
|
|
|
|
|
|
|
|
|
|
|
|
def get_thumbnail_url_full(
|
|
|
|
coordinator: RokuDataUpdateCoordinator,
|
|
|
|
is_internal: bool,
|
|
|
|
get_browse_image_url: GetBrowseImageUrlType,
|
|
|
|
media_content_type: str,
|
|
|
|
media_content_id: str,
|
|
|
|
media_image_id: str | None = None,
|
|
|
|
) -> str | None:
|
|
|
|
"""Get thumbnail URL."""
|
|
|
|
if is_internal:
|
2022-09-12 18:06:27 +00:00
|
|
|
if media_content_type == MediaType.APP and media_content_id:
|
2022-01-24 16:34:09 +00:00
|
|
|
return coordinator.roku.app_icon_url(media_content_id)
|
|
|
|
return None
|
|
|
|
|
|
|
|
return get_browse_image_url(
|
|
|
|
media_content_type,
|
2022-09-09 11:18:24 +00:00
|
|
|
media_content_id,
|
2022-01-24 16:34:09 +00:00
|
|
|
media_image_id,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_browse_media(
|
2022-02-06 04:17:31 +00:00
|
|
|
hass: HomeAssistant,
|
2022-01-24 16:34:09 +00:00
|
|
|
coordinator: RokuDataUpdateCoordinator,
|
|
|
|
get_browse_image_url: GetBrowseImageUrlType,
|
|
|
|
media_content_id: str | None,
|
|
|
|
media_content_type: str | None,
|
2022-02-06 04:17:31 +00:00
|
|
|
) -> BrowseMedia:
|
2022-01-24 16:34:09 +00:00
|
|
|
"""Browse media."""
|
|
|
|
if media_content_id is None:
|
|
|
|
return await root_payload(
|
|
|
|
hass,
|
|
|
|
coordinator,
|
|
|
|
get_browse_image_url,
|
|
|
|
)
|
|
|
|
|
|
|
|
if media_source.is_media_source_id(media_content_id):
|
|
|
|
return await media_source.async_browse_media(hass, media_content_id)
|
|
|
|
|
|
|
|
payload = {
|
|
|
|
"search_type": media_content_type,
|
|
|
|
"search_id": media_content_id,
|
|
|
|
}
|
|
|
|
|
|
|
|
response = await hass.async_add_executor_job(
|
|
|
|
build_item_response,
|
|
|
|
coordinator,
|
|
|
|
payload,
|
|
|
|
partial(
|
|
|
|
get_thumbnail_url_full,
|
|
|
|
coordinator,
|
|
|
|
is_internal_request(hass),
|
|
|
|
get_browse_image_url,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
if response is None:
|
|
|
|
raise BrowseError(f"Media not found: {media_content_type} / {media_content_id}")
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
|
async def root_payload(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
coordinator: RokuDataUpdateCoordinator,
|
|
|
|
get_browse_image_url: GetBrowseImageUrlType,
|
2022-02-06 04:17:31 +00:00
|
|
|
) -> BrowseMedia:
|
2022-01-24 16:34:09 +00:00
|
|
|
"""Return root payload for Roku."""
|
|
|
|
device = coordinator.data
|
|
|
|
|
|
|
|
children = [
|
|
|
|
item_payload(
|
2022-09-12 18:06:27 +00:00
|
|
|
{"title": "Apps", "type": MediaType.APPS},
|
2022-01-24 16:34:09 +00:00
|
|
|
coordinator,
|
|
|
|
get_browse_image_url,
|
|
|
|
)
|
|
|
|
]
|
|
|
|
|
|
|
|
if device.info.device_type == "tv" and len(device.channels) > 0:
|
|
|
|
children.append(
|
|
|
|
item_payload(
|
2022-09-12 18:06:27 +00:00
|
|
|
{"title": "TV Channels", "type": MediaType.CHANNELS},
|
2022-01-24 16:34:09 +00:00
|
|
|
coordinator,
|
|
|
|
get_browse_image_url,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2022-02-25 20:19:56 +00:00
|
|
|
for child in children:
|
|
|
|
child.thumbnail = "https://brands.home-assistant.io/_/roku/logo.png"
|
|
|
|
|
2022-01-24 16:34:09 +00:00
|
|
|
try:
|
|
|
|
browse_item = await media_source.async_browse_media(hass, None)
|
|
|
|
|
|
|
|
# If domain is None, it's overview of available sources
|
|
|
|
if browse_item.domain is None:
|
|
|
|
if browse_item.children is not None:
|
|
|
|
children.extend(browse_item.children)
|
|
|
|
else:
|
|
|
|
children.append(browse_item)
|
|
|
|
except media_source.BrowseError:
|
|
|
|
pass
|
2020-09-09 21:22:26 +00:00
|
|
|
|
2022-01-24 16:34:09 +00:00
|
|
|
if len(children) == 1:
|
|
|
|
return await async_browse_media(
|
|
|
|
hass,
|
|
|
|
coordinator,
|
|
|
|
get_browse_image_url,
|
|
|
|
children[0].media_content_id,
|
|
|
|
children[0].media_content_type,
|
|
|
|
)
|
|
|
|
|
|
|
|
return BrowseMedia(
|
|
|
|
title="Roku",
|
2022-09-12 18:06:27 +00:00
|
|
|
media_class=MediaClass.DIRECTORY,
|
2022-01-24 16:34:09 +00:00
|
|
|
media_content_id="",
|
|
|
|
media_content_type="root",
|
|
|
|
can_play=False,
|
|
|
|
can_expand=True,
|
|
|
|
children=children,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def build_item_response(
|
|
|
|
coordinator: RokuDataUpdateCoordinator,
|
|
|
|
payload: dict,
|
|
|
|
get_browse_image_url: GetBrowseImageUrlType,
|
|
|
|
) -> BrowseMedia | None:
|
2020-09-09 21:22:26 +00:00
|
|
|
"""Create response payload for the provided media query."""
|
|
|
|
search_id = payload["search_id"]
|
|
|
|
search_type = payload["search_type"]
|
|
|
|
|
|
|
|
thumbnail = None
|
|
|
|
title = None
|
|
|
|
media = None
|
2020-09-19 16:02:15 +00:00
|
|
|
children_media_class = None
|
2020-09-09 21:22:26 +00:00
|
|
|
|
2022-09-12 18:06:27 +00:00
|
|
|
if search_type == MediaType.APPS:
|
2020-09-09 21:22:26 +00:00
|
|
|
title = "Apps"
|
|
|
|
media = [
|
2022-09-12 18:06:27 +00:00
|
|
|
{"app_id": item.app_id, "title": item.name, "type": MediaType.APP}
|
2020-09-09 21:22:26 +00:00
|
|
|
for item in coordinator.data.apps
|
|
|
|
]
|
2022-09-12 18:06:27 +00:00
|
|
|
children_media_class = MediaClass.APP
|
|
|
|
elif search_type == MediaType.CHANNELS:
|
2022-01-24 16:34:09 +00:00
|
|
|
title = "TV Channels"
|
2020-09-09 21:22:26 +00:00
|
|
|
media = [
|
|
|
|
{
|
2022-02-12 02:52:31 +00:00
|
|
|
"channel_number": channel.number,
|
|
|
|
"title": format_channel_name(channel.number, channel.name),
|
2022-09-12 18:06:27 +00:00
|
|
|
"type": MediaType.CHANNEL,
|
2020-09-09 21:22:26 +00:00
|
|
|
}
|
2022-02-12 02:52:31 +00:00
|
|
|
for channel in coordinator.data.channels
|
2020-09-09 21:22:26 +00:00
|
|
|
]
|
2022-09-12 18:06:27 +00:00
|
|
|
children_media_class = MediaClass.CHANNEL
|
2020-09-09 21:22:26 +00:00
|
|
|
|
2022-01-24 16:34:09 +00:00
|
|
|
if title is None or media is None:
|
2020-09-09 21:22:26 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
return BrowseMedia(
|
2020-09-19 16:02:15 +00:00
|
|
|
media_class=CONTAINER_TYPES_SPECIFIC_MEDIA_CLASS.get(
|
2022-09-12 18:06:27 +00:00
|
|
|
search_type, MediaClass.DIRECTORY
|
2020-09-19 16:02:15 +00:00
|
|
|
),
|
2020-09-09 21:22:26 +00:00
|
|
|
media_content_id=search_id,
|
|
|
|
media_content_type=search_type,
|
|
|
|
title=title,
|
|
|
|
can_play=search_type in PLAYABLE_MEDIA_TYPES and search_id,
|
|
|
|
can_expand=True,
|
2022-01-24 16:34:09 +00:00
|
|
|
children=[
|
|
|
|
item_payload(item, coordinator, get_browse_image_url) for item in media
|
|
|
|
],
|
2020-09-19 16:02:15 +00:00
|
|
|
children_media_class=children_media_class,
|
2020-09-09 21:22:26 +00:00
|
|
|
thumbnail=thumbnail,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-01-24 16:34:09 +00:00
|
|
|
def item_payload(
|
|
|
|
item: dict,
|
|
|
|
coordinator: RokuDataUpdateCoordinator,
|
|
|
|
get_browse_image_url: GetBrowseImageUrlType,
|
2022-02-06 04:17:31 +00:00
|
|
|
) -> BrowseMedia:
|
2023-02-03 22:08:48 +00:00
|
|
|
"""Create response payload for a single media item.
|
2020-09-09 21:22:26 +00:00
|
|
|
|
|
|
|
Used by async_browse_media.
|
|
|
|
"""
|
|
|
|
thumbnail = None
|
|
|
|
|
|
|
|
if "app_id" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.APP
|
2020-09-09 21:22:26 +00:00
|
|
|
media_content_id = item["app_id"]
|
2022-01-24 16:34:09 +00:00
|
|
|
thumbnail = get_browse_image_url(media_content_type, media_content_id, None)
|
2020-09-09 21:22:26 +00:00
|
|
|
elif "channel_number" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.CHANNEL
|
2020-09-09 21:22:26 +00:00
|
|
|
media_content_id = item["channel_number"]
|
|
|
|
else:
|
|
|
|
media_content_type = item["type"]
|
|
|
|
media_content_id = ""
|
|
|
|
|
|
|
|
title = item["title"]
|
|
|
|
can_play = media_content_type in PLAYABLE_MEDIA_TYPES and media_content_id
|
|
|
|
can_expand = media_content_type in EXPANDABLE_MEDIA_TYPES
|
|
|
|
|
|
|
|
return BrowseMedia(
|
|
|
|
title=title,
|
|
|
|
media_class=CONTENT_TYPE_MEDIA_CLASS[media_content_type],
|
|
|
|
media_content_type=media_content_type,
|
|
|
|
media_content_id=media_content_id,
|
|
|
|
can_play=can_play,
|
|
|
|
can_expand=can_expand,
|
|
|
|
thumbnail=thumbnail,
|
|
|
|
)
|