core/homeassistant/components/jellyfin/browse_media.py

183 lines
5.1 KiB
Python

"""Support for media browsing."""
from __future__ import annotations
import asyncio
from typing import Any
from jellyfin_apiclient_python import JellyfinClient
from homeassistant.components.media_player import BrowseError, MediaClass, MediaType
from homeassistant.components.media_player.browse_media import BrowseMedia
from homeassistant.core import HomeAssistant
from .client_wrapper import get_artwork_url
from .const import (
CONTENT_TYPE_MAP,
MEDIA_CLASS_MAP,
MEDIA_TYPE_NONE,
SUPPORTED_COLLECTION_TYPES,
)
CONTAINER_TYPES_SPECIFIC_MEDIA_CLASS: dict[str, str] = {
MediaType.MUSIC: MediaClass.MUSIC,
MediaType.SEASON: MediaClass.SEASON,
MediaType.TVSHOW: MediaClass.TV_SHOW,
"boxset": MediaClass.DIRECTORY,
"collection": MediaClass.DIRECTORY,
"library": MediaClass.DIRECTORY,
}
PLAYABLE_MEDIA_TYPES = [
MediaType.EPISODE,
MediaType.MOVIE,
MediaType.MUSIC,
]
async def item_payload(
hass: HomeAssistant,
client: JellyfinClient,
user_id: str,
item: dict[str, Any],
) -> BrowseMedia:
"""Create response payload for a single media item."""
title = item["Name"]
thumbnail = get_artwork_url(client, item, 600)
media_content_id = item["Id"]
media_content_type = CONTENT_TYPE_MAP.get(item["Type"], MEDIA_TYPE_NONE)
return BrowseMedia(
title=title,
media_content_id=media_content_id,
media_content_type=media_content_type,
media_class=MEDIA_CLASS_MAP.get(item["Type"], MediaClass.DIRECTORY),
can_play=bool(media_content_type in PLAYABLE_MEDIA_TYPES and media_content_id),
can_expand=bool(item.get("IsFolder")),
children_media_class=None,
thumbnail=thumbnail,
)
async def build_root_response(
hass: HomeAssistant, client: JellyfinClient, user_id: str
) -> BrowseMedia:
"""Create response payload for root folder."""
folders = await hass.async_add_executor_job(client.jellyfin.get_media_folders)
children = [
await item_payload(hass, client, user_id, folder)
for folder in folders["Items"]
if folder["CollectionType"] in SUPPORTED_COLLECTION_TYPES
]
return BrowseMedia(
media_content_id="",
media_content_type="root",
media_class=MediaClass.DIRECTORY,
children_media_class=MediaClass.DIRECTORY,
title="Jellyfin",
can_play=False,
can_expand=True,
children=children,
)
async def build_item_response(
hass: HomeAssistant,
client: JellyfinClient,
user_id: str,
media_content_type: str | None,
media_content_id: str,
) -> BrowseMedia:
"""Create response payload for the provided media query."""
title, media, thumbnail = await get_media_info(
hass, client, user_id, media_content_type, media_content_id
)
if title is None or media is None:
raise BrowseError(f"Media not found: {media_content_type} / {media_content_id}")
children = await asyncio.gather(
*(item_payload(hass, client, user_id, media_item) for media_item in media)
)
response = BrowseMedia(
media_class=CONTAINER_TYPES_SPECIFIC_MEDIA_CLASS.get(
str(media_content_type), MediaClass.DIRECTORY
),
media_content_id=media_content_id,
media_content_type=str(media_content_type),
title=title,
can_play=bool(media_content_type in PLAYABLE_MEDIA_TYPES and media_content_id),
can_expand=True,
children=children,
thumbnail=thumbnail,
)
response.calculate_children_class()
return response
def fetch_item(client: JellyfinClient, item_id: str) -> dict[str, Any] | None:
"""Fetch item from Jellyfin server."""
result = client.jellyfin.get_item(item_id)
if not result:
return None
item: dict[str, Any] = result
return item
def fetch_items(
client: JellyfinClient,
params: dict[str, Any],
) -> list[dict[str, Any]] | None:
"""Fetch items from Jellyfin server."""
result = client.jellyfin.user_items(params=params)
if not result or "Items" not in result or len(result["Items"]) < 1:
return None
items: list[dict[str, Any]] = result["Items"]
return [
item
for item in items
if not item.get("IsFolder")
or (item.get("IsFolder") and item.get("ChildCount", 1) > 0)
]
async def get_media_info(
hass: HomeAssistant,
client: JellyfinClient,
user_id: str,
media_content_type: str | None,
media_content_id: str,
) -> tuple[str | None, list[dict[str, Any]] | None, str | None]:
"""Fetch media info."""
thumbnail: str | None = None
title: str | None = None
media: list[dict[str, Any]] | None = None
item = await hass.async_add_executor_job(fetch_item, client, media_content_id)
if item is None:
return None, None, None
title = item["Name"]
thumbnail = get_artwork_url(client, item)
if item.get("IsFolder"):
media = await hass.async_add_executor_job(
fetch_items, client, {"parentId": media_content_id, "fields": "childCount"}
)
if not media or len(media) == 0:
media = None
return title, media, thumbnail