Add media source support to Kodi (#67203)
parent
8a74295d6f
commit
406fbca4bc
|
@ -1,7 +1,9 @@
|
||||||
"""Support for media browsing."""
|
"""Support for media browsing."""
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import contextlib
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
|
from homeassistant.components import media_source
|
||||||
from homeassistant.components.media_player import BrowseError, BrowseMedia
|
from homeassistant.components.media_player import BrowseError, BrowseMedia
|
||||||
from homeassistant.components.media_player.const import (
|
from homeassistant.components.media_player.const import (
|
||||||
MEDIA_CLASS_ALBUM,
|
MEDIA_CLASS_ALBUM,
|
||||||
|
@ -184,7 +186,7 @@ async def item_payload(item, get_thumbnail_url=None):
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def library_payload():
|
async def library_payload(hass):
|
||||||
"""
|
"""
|
||||||
Create response payload to describe contents of a specific library.
|
Create response payload to describe contents of a specific library.
|
||||||
|
|
||||||
|
@ -222,6 +224,14 @@ async def library_payload():
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
with contextlib.suppress(media_source.BrowseError):
|
||||||
|
item = await media_source.async_browse_media(hass, None)
|
||||||
|
# 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)
|
||||||
|
|
||||||
return library_info
|
return library_info
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
"domain": "kodi",
|
"domain": "kodi",
|
||||||
"name": "Kodi",
|
"name": "Kodi",
|
||||||
"documentation": "https://www.home-assistant.io/integrations/kodi",
|
"documentation": "https://www.home-assistant.io/integrations/kodi",
|
||||||
|
"after_dependencies": ["media_source"],
|
||||||
"requirements": ["pykodi==0.2.7"],
|
"requirements": ["pykodi==0.2.7"],
|
||||||
"codeowners": ["@OnFreund", "@cgtobi"],
|
"codeowners": ["@OnFreund", "@cgtobi"],
|
||||||
"zeroconf": ["_xbmc-jsonrpc-h._tcp.local."],
|
"zeroconf": ["_xbmc-jsonrpc-h._tcp.local."],
|
||||||
|
|
|
@ -5,6 +5,7 @@ from datetime import timedelta
|
||||||
from functools import wraps
|
from functools import wraps
|
||||||
import logging
|
import logging
|
||||||
import re
|
import re
|
||||||
|
from typing import Any
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
|
|
||||||
import jsonrpc_base
|
import jsonrpc_base
|
||||||
|
@ -12,7 +13,11 @@ from jsonrpc_base.jsonrpc import ProtocolError, TransportError
|
||||||
from pykodi import CannotConnectError
|
from pykodi import CannotConnectError
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
|
from homeassistant.components import media_source
|
||||||
from homeassistant.components.media_player import PLATFORM_SCHEMA, MediaPlayerEntity
|
from homeassistant.components.media_player import PLATFORM_SCHEMA, MediaPlayerEntity
|
||||||
|
from homeassistant.components.media_player.browse_media import (
|
||||||
|
async_process_play_media_url,
|
||||||
|
)
|
||||||
from homeassistant.components.media_player.const import (
|
from homeassistant.components.media_player.const import (
|
||||||
MEDIA_TYPE_ALBUM,
|
MEDIA_TYPE_ALBUM,
|
||||||
MEDIA_TYPE_ARTIST,
|
MEDIA_TYPE_ARTIST,
|
||||||
|
@ -24,6 +29,7 @@ from homeassistant.components.media_player.const import (
|
||||||
MEDIA_TYPE_SEASON,
|
MEDIA_TYPE_SEASON,
|
||||||
MEDIA_TYPE_TRACK,
|
MEDIA_TYPE_TRACK,
|
||||||
MEDIA_TYPE_TVSHOW,
|
MEDIA_TYPE_TVSHOW,
|
||||||
|
MEDIA_TYPE_URL,
|
||||||
MEDIA_TYPE_VIDEO,
|
MEDIA_TYPE_VIDEO,
|
||||||
SUPPORT_BROWSE_MEDIA,
|
SUPPORT_BROWSE_MEDIA,
|
||||||
SUPPORT_NEXT_TRACK,
|
SUPPORT_NEXT_TRACK,
|
||||||
|
@ -691,8 +697,15 @@ class KodiEntity(MediaPlayerEntity):
|
||||||
await self._kodi.media_seek(position)
|
await self._kodi.media_seek(position)
|
||||||
|
|
||||||
@cmd
|
@cmd
|
||||||
async def async_play_media(self, media_type, media_id, **kwargs):
|
async def async_play_media(
|
||||||
|
self, media_type: str, media_id: str, **kwargs: Any
|
||||||
|
) -> None:
|
||||||
"""Send the play_media command to the media player."""
|
"""Send the play_media command to the media player."""
|
||||||
|
if media_source.is_media_source_id(media_id):
|
||||||
|
media_type = MEDIA_TYPE_URL
|
||||||
|
play_item = await media_source.async_resolve_media(self.hass, media_id)
|
||||||
|
media_id = play_item.url
|
||||||
|
|
||||||
media_type_lower = media_type.lower()
|
media_type_lower = media_type.lower()
|
||||||
|
|
||||||
if media_type_lower == MEDIA_TYPE_CHANNEL:
|
if media_type_lower == MEDIA_TYPE_CHANNEL:
|
||||||
|
@ -700,7 +713,7 @@ class KodiEntity(MediaPlayerEntity):
|
||||||
elif media_type_lower == MEDIA_TYPE_PLAYLIST:
|
elif media_type_lower == MEDIA_TYPE_PLAYLIST:
|
||||||
await self._kodi.play_playlist(int(media_id))
|
await self._kodi.play_playlist(int(media_id))
|
||||||
elif media_type_lower == "directory":
|
elif media_type_lower == "directory":
|
||||||
await self._kodi.play_directory(str(media_id))
|
await self._kodi.play_directory(media_id)
|
||||||
elif media_type_lower in [
|
elif media_type_lower in [
|
||||||
MEDIA_TYPE_ARTIST,
|
MEDIA_TYPE_ARTIST,
|
||||||
MEDIA_TYPE_ALBUM,
|
MEDIA_TYPE_ALBUM,
|
||||||
|
@ -719,7 +732,9 @@ class KodiEntity(MediaPlayerEntity):
|
||||||
{MAP_KODI_MEDIA_TYPES[media_type_lower]: int(media_id)}
|
{MAP_KODI_MEDIA_TYPES[media_type_lower]: int(media_id)}
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
await self._kodi.play_file(str(media_id))
|
media_id = async_process_play_media_url(self.hass, media_id)
|
||||||
|
|
||||||
|
await self._kodi.play_file(media_id)
|
||||||
|
|
||||||
@cmd
|
@cmd
|
||||||
async def async_set_shuffle(self, shuffle):
|
async def async_set_shuffle(self, shuffle):
|
||||||
|
@ -898,7 +913,10 @@ class KodiEntity(MediaPlayerEntity):
|
||||||
)
|
)
|
||||||
|
|
||||||
if media_content_type in [None, "library"]:
|
if media_content_type in [None, "library"]:
|
||||||
return await library_payload()
|
return await library_payload(self.hass)
|
||||||
|
|
||||||
|
if media_source.is_media_source_id(media_content_id):
|
||||||
|
return await media_source.async_browse_media(self.hass, media_content_id)
|
||||||
|
|
||||||
payload = {
|
payload = {
|
||||||
"search_type": media_content_type,
|
"search_type": media_content_type,
|
||||||
|
|
Loading…
Reference in New Issue