2021-05-16 09:11:35 +00:00
|
|
|
"""Class representing Sonos favorites."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from collections.abc import Iterator
|
|
|
|
import logging
|
2021-06-16 15:30:05 +00:00
|
|
|
from typing import Any
|
2021-05-16 09:11:35 +00:00
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
from pysonos import SoCo
|
2021-05-16 09:11:35 +00:00
|
|
|
from pysonos.data_structures import DidlFavorite
|
|
|
|
from pysonos.exceptions import SoCoException
|
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
2021-05-16 09:11:35 +00:00
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
from .const import SONOS_FAVORITES_UPDATED
|
|
|
|
from .household_coordinator import SonosHouseholdCoordinator
|
2021-05-16 09:11:35 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
class SonosFavorites(SonosHouseholdCoordinator):
|
|
|
|
"""Coordinator class for Sonos favorites."""
|
2021-05-16 09:11:35 +00:00
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
def __init__(self, *args: Any) -> None:
|
2021-05-16 09:11:35 +00:00
|
|
|
"""Initialize the data."""
|
2021-06-16 15:30:05 +00:00
|
|
|
super().__init__(*args)
|
2021-05-16 09:11:35 +00:00
|
|
|
self._favorites: list[DidlFavorite] = []
|
|
|
|
|
|
|
|
def __iter__(self) -> Iterator:
|
|
|
|
"""Return an iterator for the known favorites."""
|
|
|
|
favorites = self._favorites.copy()
|
|
|
|
return iter(favorites)
|
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
async def async_update_entities(self, soco: SoCo) -> bool:
|
|
|
|
"""Update the cache and update entities."""
|
|
|
|
try:
|
|
|
|
await self.hass.async_add_executor_job(self.update_cache, soco)
|
|
|
|
except (OSError, SoCoException) as err:
|
|
|
|
_LOGGER.warning("Error requesting favorites from %s: %s", soco, err)
|
|
|
|
return False
|
2021-05-16 09:11:35 +00:00
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
async_dispatcher_send(
|
|
|
|
self.hass, f"{SONOS_FAVORITES_UPDATED}-{self.household_id}"
|
|
|
|
)
|
|
|
|
return True
|
2021-05-16 09:11:35 +00:00
|
|
|
|
2021-06-16 15:30:05 +00:00
|
|
|
def update_cache(self, soco: SoCo) -> None:
|
2021-05-16 09:11:35 +00:00
|
|
|
"""Request new Sonos favorites from a speaker."""
|
2021-06-16 15:30:05 +00:00
|
|
|
new_favorites = soco.music_library.get_sonos_favorites()
|
2021-05-16 09:11:35 +00:00
|
|
|
self._favorites = []
|
2021-06-16 15:30:05 +00:00
|
|
|
|
2021-05-16 09:11:35 +00:00
|
|
|
for fav in new_favorites:
|
|
|
|
try:
|
|
|
|
# exclude non-playable favorites with no linked resources
|
|
|
|
if fav.reference.resources:
|
|
|
|
self._favorites.append(fav)
|
|
|
|
except SoCoException as ex:
|
|
|
|
# Skip unknown types
|
|
|
|
_LOGGER.error("Unhandled favorite '%s': %s", fav.title, ex)
|
2021-06-16 15:30:05 +00:00
|
|
|
|
2021-05-16 09:11:35 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"Cached %s favorites for household %s",
|
|
|
|
len(self._favorites),
|
|
|
|
self.household_id,
|
|
|
|
)
|