138 lines
4.1 KiB
Python
138 lines
4.1 KiB
Python
"""The spotify integration."""
|
|
|
|
import aiohttp
|
|
from spotipy import Spotify, SpotifyException
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.media_player import BrowseError, BrowseMedia
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
ATTR_CREDENTIALS,
|
|
CONF_CLIENT_ID,
|
|
CONF_CLIENT_SECRET,
|
|
Platform,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
|
|
from homeassistant.helpers import config_entry_oauth2_flow, config_validation as cv
|
|
from homeassistant.helpers.config_entry_oauth2_flow import (
|
|
OAuth2Session,
|
|
async_get_config_entry_implementation,
|
|
)
|
|
from homeassistant.helpers.typing import ConfigType
|
|
|
|
from . import config_flow
|
|
from .const import (
|
|
DATA_SPOTIFY_CLIENT,
|
|
DATA_SPOTIFY_ME,
|
|
DATA_SPOTIFY_SESSION,
|
|
DOMAIN,
|
|
MEDIA_PLAYER_PREFIX,
|
|
SPOTIFY_SCOPES,
|
|
)
|
|
from .media_player import async_browse_media_internal
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
{
|
|
DOMAIN: vol.Schema(
|
|
{
|
|
vol.Inclusive(CONF_CLIENT_ID, ATTR_CREDENTIALS): cv.string,
|
|
vol.Inclusive(CONF_CLIENT_SECRET, ATTR_CREDENTIALS): cv.string,
|
|
}
|
|
)
|
|
},
|
|
extra=vol.ALLOW_EXTRA,
|
|
)
|
|
|
|
PLATFORMS = [Platform.MEDIA_PLAYER]
|
|
|
|
|
|
def is_spotify_media_type(media_content_type: str) -> bool:
|
|
"""Return whether the media_content_type is a valid Spotify media_id."""
|
|
return media_content_type.startswith(MEDIA_PLAYER_PREFIX)
|
|
|
|
|
|
def resolve_spotify_media_type(media_content_type: str) -> str:
|
|
"""Return actual spotify media_content_type."""
|
|
return media_content_type[len(MEDIA_PLAYER_PREFIX) :]
|
|
|
|
|
|
async def async_browse_media(
|
|
hass: HomeAssistant,
|
|
media_content_type: str,
|
|
media_content_id: str,
|
|
*,
|
|
can_play_artist: bool = True,
|
|
) -> BrowseMedia:
|
|
"""Browse Spotify media."""
|
|
if not (info := next(iter(hass.data[DOMAIN].values()), None)):
|
|
raise BrowseError("No Spotify accounts available")
|
|
return await async_browse_media_internal(
|
|
hass,
|
|
info[DATA_SPOTIFY_CLIENT],
|
|
info[DATA_SPOTIFY_SESSION],
|
|
info[DATA_SPOTIFY_ME],
|
|
media_content_type,
|
|
media_content_id,
|
|
can_play_artist=can_play_artist,
|
|
)
|
|
|
|
|
|
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
|
"""Set up the Spotify integration."""
|
|
if DOMAIN not in config:
|
|
return True
|
|
|
|
if CONF_CLIENT_ID in config[DOMAIN]:
|
|
config_flow.SpotifyFlowHandler.async_register_implementation(
|
|
hass,
|
|
config_entry_oauth2_flow.LocalOAuth2Implementation(
|
|
hass,
|
|
DOMAIN,
|
|
config[DOMAIN][CONF_CLIENT_ID],
|
|
config[DOMAIN][CONF_CLIENT_SECRET],
|
|
"https://accounts.spotify.com/authorize",
|
|
"https://accounts.spotify.com/api/token",
|
|
),
|
|
)
|
|
|
|
return True
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Set up Spotify from a config entry."""
|
|
implementation = await async_get_config_entry_implementation(hass, entry)
|
|
session = OAuth2Session(hass, entry, implementation)
|
|
|
|
try:
|
|
await session.async_ensure_token_valid()
|
|
except aiohttp.ClientError as err:
|
|
raise ConfigEntryNotReady from err
|
|
|
|
spotify = Spotify(auth=session.token["access_token"])
|
|
|
|
try:
|
|
current_user = await hass.async_add_executor_job(spotify.me)
|
|
except SpotifyException as err:
|
|
raise ConfigEntryNotReady from err
|
|
|
|
hass.data.setdefault(DOMAIN, {})
|
|
hass.data[DOMAIN][entry.entry_id] = {
|
|
DATA_SPOTIFY_CLIENT: spotify,
|
|
DATA_SPOTIFY_ME: current_user,
|
|
DATA_SPOTIFY_SESSION: session,
|
|
}
|
|
|
|
if not set(session.token["scope"].split(" ")).issuperset(SPOTIFY_SCOPES):
|
|
raise ConfigEntryAuthFailed
|
|
|
|
hass.config_entries.async_setup_platforms(entry, PLATFORMS)
|
|
return True
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Unload Spotify config entry."""
|
|
if unload_ok := await hass.config_entries.async_unload_platforms(entry, PLATFORMS):
|
|
del hass.data[DOMAIN][entry.entry_id]
|
|
return unload_ok
|