2020-09-07 16:22:20 +00:00
|
|
|
"""Support for media browsing."""
|
2021-03-30 00:30:51 +00:00
|
|
|
import asyncio
|
2022-02-25 07:40:28 +00:00
|
|
|
import contextlib
|
2020-09-09 12:12:11 +00:00
|
|
|
import logging
|
2020-09-07 16:22:20 +00:00
|
|
|
|
2022-02-25 07:40:28 +00:00
|
|
|
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-07 16:22:20 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
PLAYABLE_MEDIA_TYPES = [
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.ALBUM,
|
|
|
|
MediaType.ARTIST,
|
|
|
|
MediaType.TRACK,
|
2020-09-07 16:22:20 +00:00
|
|
|
]
|
|
|
|
|
2020-09-11 11:08:13 +00:00
|
|
|
CONTAINER_TYPES_SPECIFIC_MEDIA_CLASS = {
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.ALBUM: MediaClass.ALBUM,
|
|
|
|
MediaType.ARTIST: MediaClass.ARTIST,
|
|
|
|
MediaType.PLAYLIST: MediaClass.PLAYLIST,
|
|
|
|
MediaType.SEASON: MediaClass.SEASON,
|
|
|
|
MediaType.TVSHOW: MediaClass.TV_SHOW,
|
2020-09-11 11:08:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CHILD_TYPE_MEDIA_CLASS = {
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.SEASON: MediaClass.SEASON,
|
|
|
|
MediaType.ALBUM: MediaClass.ALBUM,
|
|
|
|
MediaType.ARTIST: MediaClass.ARTIST,
|
|
|
|
MediaType.MOVIE: MediaClass.MOVIE,
|
|
|
|
MediaType.PLAYLIST: MediaClass.PLAYLIST,
|
|
|
|
MediaType.TRACK: MediaClass.TRACK,
|
|
|
|
MediaType.TVSHOW: MediaClass.TV_SHOW,
|
|
|
|
MediaType.CHANNEL: MediaClass.CHANNEL,
|
|
|
|
MediaType.EPISODE: MediaClass.EPISODE,
|
2020-09-08 14:42:01 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 12:12:11 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class UnknownMediaType(BrowseError):
|
|
|
|
"""Unknown media type."""
|
|
|
|
|
2020-09-07 16:22:20 +00:00
|
|
|
|
2021-03-30 00:30:51 +00:00
|
|
|
async def build_item_response(media_library, payload, get_thumbnail_url=None):
|
2020-09-07 16:22:20 +00:00
|
|
|
"""Create response payload for the provided media query."""
|
|
|
|
search_id = payload["search_id"]
|
|
|
|
search_type = payload["search_type"]
|
|
|
|
|
2021-03-30 00:30:51 +00:00
|
|
|
_, title, media = await get_media_info(media_library, search_id, search_type)
|
|
|
|
thumbnail = await get_thumbnail_url(search_type, search_id)
|
2020-09-07 16:22:20 +00:00
|
|
|
|
|
|
|
if media is None:
|
2020-09-09 12:12:11 +00:00
|
|
|
return None
|
|
|
|
|
2021-03-30 00:30:51 +00:00
|
|
|
children = await asyncio.gather(
|
2021-07-19 08:46:09 +00:00
|
|
|
*(item_payload(item, get_thumbnail_url) for item in media)
|
2021-03-30 00:30:51 +00:00
|
|
|
)
|
2020-09-07 16:22:20 +00:00
|
|
|
|
2022-09-12 18:06:27 +00:00
|
|
|
if search_type in (MediaType.TVSHOW, MediaType.MOVIE) and search_id == "":
|
2020-10-19 07:55:17 +00:00
|
|
|
children.sort(key=lambda x: x.title.replace("The ", "", 1), reverse=False)
|
|
|
|
|
2020-09-11 11:08:13 +00:00
|
|
|
response = BrowseMedia(
|
|
|
|
media_class=CONTAINER_TYPES_SPECIFIC_MEDIA_CLASS.get(
|
2022-09-12 18:06:27 +00:00
|
|
|
search_type, MediaClass.DIRECTORY
|
2020-09-11 11:08:13 +00:00
|
|
|
),
|
2020-09-09 12:12:11 +00:00
|
|
|
media_content_id=search_id,
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_type=search_type,
|
|
|
|
title=title,
|
|
|
|
can_play=search_type in PLAYABLE_MEDIA_TYPES and search_id,
|
|
|
|
can_expand=True,
|
2020-09-09 12:12:11 +00:00
|
|
|
children=children,
|
2020-09-07 16:22:20 +00:00
|
|
|
thumbnail=thumbnail,
|
|
|
|
)
|
|
|
|
|
2020-09-11 11:08:13 +00:00
|
|
|
if search_type == "library_music":
|
2022-09-12 18:06:27 +00:00
|
|
|
response.children_media_class = MediaClass.MUSIC
|
2020-09-11 11:08:13 +00:00
|
|
|
else:
|
|
|
|
response.calculate_children_class()
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
2020-09-07 16:22:20 +00:00
|
|
|
|
2021-03-30 00:30:51 +00:00
|
|
|
async def item_payload(item, get_thumbnail_url=None):
|
2023-02-03 22:08:48 +00:00
|
|
|
"""Create response payload for a single media item.
|
2020-09-07 16:22:20 +00:00
|
|
|
|
|
|
|
Used by async_browse_media.
|
|
|
|
"""
|
2020-09-09 12:12:11 +00:00
|
|
|
title = item["label"]
|
|
|
|
|
2020-09-11 11:08:13 +00:00
|
|
|
media_class = None
|
|
|
|
|
2020-09-07 16:22:20 +00:00
|
|
|
if "songid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.TRACK
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = f"{item['songid']}"
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = True
|
|
|
|
can_expand = False
|
2020-09-07 16:22:20 +00:00
|
|
|
elif "albumid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.ALBUM
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = f"{item['albumid']}"
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = True
|
|
|
|
can_expand = True
|
2020-09-07 16:22:20 +00:00
|
|
|
elif "artistid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.ARTIST
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = f"{item['artistid']}"
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = True
|
|
|
|
can_expand = True
|
2020-09-07 16:22:20 +00:00
|
|
|
elif "movieid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.MOVIE
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = f"{item['movieid']}"
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = True
|
|
|
|
can_expand = False
|
2020-09-07 16:22:20 +00:00
|
|
|
elif "episodeid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.EPISODE
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = f"{item['episodeid']}"
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = True
|
|
|
|
can_expand = False
|
2020-09-07 16:22:20 +00:00
|
|
|
elif "seasonid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.SEASON
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = f"{item['tvshowid']}/{item['season']}"
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = False
|
|
|
|
can_expand = True
|
2020-09-07 16:22:20 +00:00
|
|
|
elif "tvshowid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.TVSHOW
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = f"{item['tvshowid']}"
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = False
|
|
|
|
can_expand = True
|
2020-09-19 15:14:20 +00:00
|
|
|
elif "channelid" in item:
|
2022-09-12 18:06:27 +00:00
|
|
|
media_content_type = MediaType.CHANNEL
|
2020-09-19 15:14:20 +00:00
|
|
|
media_content_id = f"{item['channelid']}"
|
2021-10-20 15:42:26 +00:00
|
|
|
if broadcasting := item.get("broadcastnow"):
|
2020-09-19 15:14:20 +00:00
|
|
|
show = broadcasting.get("title")
|
|
|
|
title = f"{title} - {show}"
|
|
|
|
can_play = True
|
|
|
|
can_expand = False
|
2020-09-07 16:22:20 +00:00
|
|
|
else:
|
|
|
|
# this case is for the top folder of each type
|
2020-09-19 15:14:20 +00:00
|
|
|
# possible content types: album, artist, movie, library_music, tvshow, channel
|
2022-09-12 18:06:27 +00:00
|
|
|
media_class = MediaClass.DIRECTORY
|
2020-09-09 12:12:11 +00:00
|
|
|
media_content_type = item["type"]
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id = ""
|
2020-09-09 12:12:11 +00:00
|
|
|
can_play = False
|
2020-09-08 21:42:45 +00:00
|
|
|
can_expand = True
|
2020-09-09 12:12:11 +00:00
|
|
|
|
2020-09-11 11:08:13 +00:00
|
|
|
if media_class is None:
|
|
|
|
try:
|
|
|
|
media_class = CHILD_TYPE_MEDIA_CLASS[media_content_type]
|
|
|
|
except KeyError as err:
|
|
|
|
_LOGGER.debug("Unknown media type received: %s", media_content_type)
|
|
|
|
raise UnknownMediaType from err
|
2020-09-08 21:42:45 +00:00
|
|
|
|
2021-03-30 00:30:51 +00:00
|
|
|
thumbnail = item.get("thumbnail")
|
|
|
|
if thumbnail is not None and get_thumbnail_url is not None:
|
|
|
|
thumbnail = await get_thumbnail_url(
|
|
|
|
media_content_type, media_content_id, thumbnail_url=thumbnail
|
|
|
|
)
|
|
|
|
|
2020-09-07 16:22:20 +00:00
|
|
|
return BrowseMedia(
|
|
|
|
title=title,
|
2020-09-08 21:42:45 +00:00
|
|
|
media_class=media_class,
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_type=media_content_type,
|
|
|
|
media_content_id=media_content_id,
|
|
|
|
can_play=can_play,
|
|
|
|
can_expand=can_expand,
|
|
|
|
thumbnail=thumbnail,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-02-26 08:58:45 +00:00
|
|
|
def media_source_content_filter(item: BrowseMedia) -> bool:
|
|
|
|
"""Content filter for media sources."""
|
|
|
|
# Filter out cameras using PNG over MJPEG. They don't work in Kodi.
|
|
|
|
return not (
|
|
|
|
item.media_content_id.startswith("media-source://camera/")
|
|
|
|
and item.media_content_type == "image/png"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-02-25 07:40:28 +00:00
|
|
|
async def library_payload(hass):
|
2023-02-03 22:08:48 +00:00
|
|
|
"""Create response payload to describe contents of a specific library.
|
2020-09-07 16:22:20 +00:00
|
|
|
|
|
|
|
Used by async_browse_media.
|
|
|
|
"""
|
|
|
|
library_info = BrowseMedia(
|
2022-09-12 18:06:27 +00:00
|
|
|
media_class=MediaClass.DIRECTORY,
|
2020-09-07 16:22:20 +00:00
|
|
|
media_content_id="library",
|
|
|
|
media_content_type="library",
|
|
|
|
title="Media Library",
|
|
|
|
can_play=False,
|
|
|
|
can_expand=True,
|
|
|
|
children=[],
|
|
|
|
)
|
|
|
|
|
|
|
|
library = {
|
|
|
|
"library_music": "Music",
|
2022-09-12 18:06:27 +00:00
|
|
|
MediaType.MOVIE: "Movies",
|
|
|
|
MediaType.TVSHOW: "TV shows",
|
|
|
|
MediaType.CHANNEL: "Channels",
|
2020-09-07 16:22:20 +00:00
|
|
|
}
|
2021-03-30 00:30:51 +00:00
|
|
|
|
|
|
|
library_info.children = await asyncio.gather(
|
2021-07-19 08:46:09 +00:00
|
|
|
*(
|
2020-09-07 16:22:20 +00:00
|
|
|
item_payload(
|
2021-03-30 00:30:51 +00:00
|
|
|
{
|
|
|
|
"label": item["label"],
|
|
|
|
"type": item["type"],
|
|
|
|
"uri": item["type"],
|
|
|
|
},
|
2020-09-07 16:22:20 +00:00
|
|
|
)
|
2021-03-30 00:30:51 +00:00
|
|
|
for item in [
|
|
|
|
{"label": name, "type": type_} for type_, name in library.items()
|
|
|
|
]
|
2021-07-19 08:46:09 +00:00
|
|
|
)
|
2021-03-30 00:30:51 +00:00
|
|
|
)
|
2020-09-07 16:22:20 +00:00
|
|
|
|
2022-02-25 20:19:56 +00:00
|
|
|
for child in library_info.children:
|
|
|
|
child.thumbnail = "https://brands.home-assistant.io/_/kodi/logo.png"
|
|
|
|
|
2022-02-25 07:40:28 +00:00
|
|
|
with contextlib.suppress(media_source.BrowseError):
|
2022-02-26 08:58:45 +00:00
|
|
|
item = await media_source.async_browse_media(
|
|
|
|
hass, None, content_filter=media_source_content_filter
|
|
|
|
)
|
2022-02-25 07:40:28 +00:00
|
|
|
# If domain is None, it's overview of available sources
|
|
|
|
if item.domain is None:
|
|
|
|
library_info.children.extend(item.children)
|
|
|
|
else:
|
|
|
|
library_info.children.append(item)
|
|
|
|
|
2020-09-07 16:22:20 +00:00
|
|
|
return library_info
|
2021-03-30 00:30:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def get_media_info(media_library, search_id, search_type):
|
|
|
|
"""Fetch media/album."""
|
|
|
|
thumbnail = None
|
|
|
|
title = None
|
|
|
|
media = None
|
|
|
|
|
|
|
|
properties = ["thumbnail"]
|
2022-09-12 18:06:27 +00:00
|
|
|
if search_type == MediaType.ALBUM:
|
2021-03-30 00:30:51 +00:00
|
|
|
if search_id:
|
|
|
|
album = await media_library.get_album_details(
|
|
|
|
album_id=int(search_id), properties=properties
|
|
|
|
)
|
|
|
|
thumbnail = media_library.thumbnail_url(
|
|
|
|
album["albumdetails"].get("thumbnail")
|
|
|
|
)
|
|
|
|
title = album["albumdetails"]["label"]
|
|
|
|
media = await media_library.get_songs(
|
|
|
|
album_id=int(search_id),
|
|
|
|
properties=[
|
|
|
|
"albumid",
|
|
|
|
"artist",
|
|
|
|
"duration",
|
|
|
|
"album",
|
|
|
|
"thumbnail",
|
|
|
|
"track",
|
|
|
|
],
|
|
|
|
)
|
|
|
|
media = media.get("songs")
|
|
|
|
else:
|
|
|
|
media = await media_library.get_albums(properties=properties)
|
|
|
|
media = media.get("albums")
|
|
|
|
title = "Albums"
|
|
|
|
|
2022-09-12 18:06:27 +00:00
|
|
|
elif search_type == MediaType.ARTIST:
|
2021-03-30 00:30:51 +00:00
|
|
|
if search_id:
|
|
|
|
media = await media_library.get_albums(
|
|
|
|
artist_id=int(search_id), properties=properties
|
|
|
|
)
|
|
|
|
media = media.get("albums")
|
|
|
|
artist = await media_library.get_artist_details(
|
|
|
|
artist_id=int(search_id), properties=properties
|
|
|
|
)
|
|
|
|
thumbnail = media_library.thumbnail_url(
|
|
|
|
artist["artistdetails"].get("thumbnail")
|
|
|
|
)
|
|
|
|
title = artist["artistdetails"]["label"]
|
|
|
|
else:
|
|
|
|
media = await media_library.get_artists(properties)
|
|
|
|
media = media.get("artists")
|
|
|
|
title = "Artists"
|
|
|
|
|
|
|
|
elif search_type == "library_music":
|
2022-09-12 18:06:27 +00:00
|
|
|
library = {MediaType.ALBUM: "Albums", MediaType.ARTIST: "Artists"}
|
2021-03-30 00:30:51 +00:00
|
|
|
media = [{"label": name, "type": type_} for type_, name in library.items()]
|
|
|
|
title = "Music Library"
|
|
|
|
|
2022-09-12 18:06:27 +00:00
|
|
|
elif search_type == MediaType.MOVIE:
|
2021-03-30 00:30:51 +00:00
|
|
|
if search_id:
|
|
|
|
movie = await media_library.get_movie_details(
|
|
|
|
movie_id=int(search_id), properties=properties
|
|
|
|
)
|
|
|
|
thumbnail = media_library.thumbnail_url(
|
|
|
|
movie["moviedetails"].get("thumbnail")
|
|
|
|
)
|
|
|
|
title = movie["moviedetails"]["label"]
|
|
|
|
else:
|
|
|
|
media = await media_library.get_movies(properties)
|
|
|
|
media = media.get("movies")
|
|
|
|
title = "Movies"
|
|
|
|
|
2022-09-12 18:06:27 +00:00
|
|
|
elif search_type == MediaType.TVSHOW:
|
2021-03-30 00:30:51 +00:00
|
|
|
if search_id:
|
|
|
|
media = await media_library.get_seasons(
|
|
|
|
tv_show_id=int(search_id),
|
|
|
|
properties=["thumbnail", "season", "tvshowid"],
|
|
|
|
)
|
|
|
|
media = media.get("seasons")
|
|
|
|
tvshow = await media_library.get_tv_show_details(
|
|
|
|
tv_show_id=int(search_id), properties=properties
|
|
|
|
)
|
|
|
|
thumbnail = media_library.thumbnail_url(
|
|
|
|
tvshow["tvshowdetails"].get("thumbnail")
|
|
|
|
)
|
|
|
|
title = tvshow["tvshowdetails"]["label"]
|
|
|
|
else:
|
|
|
|
media = await media_library.get_tv_shows(properties)
|
|
|
|
media = media.get("tvshows")
|
|
|
|
title = "TV Shows"
|
|
|
|
|
2022-09-12 18:06:27 +00:00
|
|
|
elif search_type == MediaType.SEASON:
|
2021-03-30 00:30:51 +00:00
|
|
|
tv_show_id, season_id = search_id.split("/", 1)
|
|
|
|
media = await media_library.get_episodes(
|
|
|
|
tv_show_id=int(tv_show_id),
|
|
|
|
season_id=int(season_id),
|
|
|
|
properties=["thumbnail", "tvshowid", "seasonid"],
|
|
|
|
)
|
|
|
|
media = media.get("episodes")
|
|
|
|
if media:
|
|
|
|
season = await media_library.get_season_details(
|
|
|
|
season_id=int(media[0]["seasonid"]), properties=properties
|
|
|
|
)
|
|
|
|
thumbnail = media_library.thumbnail_url(
|
|
|
|
season["seasondetails"].get("thumbnail")
|
|
|
|
)
|
|
|
|
title = season["seasondetails"]["label"]
|
|
|
|
|
2022-09-12 18:06:27 +00:00
|
|
|
elif search_type == MediaType.CHANNEL:
|
2021-03-30 00:30:51 +00:00
|
|
|
media = await media_library.get_channels(
|
|
|
|
channel_group_id="alltv",
|
|
|
|
properties=["thumbnail", "channeltype", "channel", "broadcastnow"],
|
|
|
|
)
|
|
|
|
media = media.get("channels")
|
|
|
|
title = "Channels"
|
|
|
|
|
|
|
|
return thumbnail, title, media
|