2019-09-10 20:05:46 +00:00
|
|
|
"""Helpers to deal with Cast devices."""
|
2021-02-12 17:54:00 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-01-06 08:23:18 +00:00
|
|
|
from typing import Optional
|
2019-09-10 20:05:46 +00:00
|
|
|
|
|
|
|
import attr
|
2021-01-06 08:23:18 +00:00
|
|
|
from pychromecast import dial
|
2020-04-09 08:58:19 +00:00
|
|
|
from pychromecast.const import CAST_MANUFACTURERS
|
2019-09-10 20:05:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
@attr.s(slots=True, frozen=True)
|
|
|
|
class ChromecastInfo:
|
|
|
|
"""Class to hold all data about a chromecast for creating connections.
|
|
|
|
|
|
|
|
This also has the same attributes as the mDNS fields by zeroconf.
|
|
|
|
"""
|
|
|
|
|
2021-03-18 21:58:19 +00:00
|
|
|
services: set | None = attr.ib()
|
|
|
|
uuid: str | None = attr.ib(
|
2020-07-14 17:30:30 +00:00
|
|
|
converter=attr.converters.optional(str), default=None
|
2019-09-10 20:05:46 +00:00
|
|
|
) # always convert UUID to string if not None
|
2021-01-06 08:23:18 +00:00
|
|
|
_manufacturer = attr.ib(type=Optional[str], default=None)
|
2020-07-14 17:30:30 +00:00
|
|
|
model_name: str = attr.ib(default="")
|
2021-03-18 21:58:19 +00:00
|
|
|
friendly_name: str | None = attr.ib(default=None)
|
2021-02-26 12:43:53 +00:00
|
|
|
is_audio_group = attr.ib(type=Optional[bool], default=False)
|
2021-01-06 08:23:18 +00:00
|
|
|
is_dynamic_group = attr.ib(type=Optional[bool], default=None)
|
2019-09-10 20:05:46 +00:00
|
|
|
|
|
|
|
@property
|
2021-01-06 08:23:18 +00:00
|
|
|
def is_information_complete(self) -> bool:
|
|
|
|
"""Return if all information is filled out."""
|
|
|
|
want_dynamic_group = self.is_audio_group
|
|
|
|
have_dynamic_group = self.is_dynamic_group is not None
|
|
|
|
have_all_except_dynamic_group = all(
|
|
|
|
attr.astuple(
|
|
|
|
self,
|
|
|
|
filter=attr.filters.exclude(
|
|
|
|
attr.fields(ChromecastInfo).is_dynamic_group
|
|
|
|
),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return have_all_except_dynamic_group and (
|
|
|
|
not want_dynamic_group or have_dynamic_group
|
|
|
|
)
|
2019-09-10 20:05:46 +00:00
|
|
|
|
2020-04-09 08:58:19 +00:00
|
|
|
@property
|
2021-07-05 09:14:41 +00:00
|
|
|
def manufacturer(self) -> str | None:
|
2020-04-09 08:58:19 +00:00
|
|
|
"""Return the manufacturer."""
|
2021-01-06 08:23:18 +00:00
|
|
|
if self._manufacturer:
|
|
|
|
return self._manufacturer
|
2020-04-09 08:58:19 +00:00
|
|
|
if not self.model_name:
|
|
|
|
return None
|
|
|
|
return CAST_MANUFACTURERS.get(self.model_name.lower(), "Google Inc.")
|
|
|
|
|
2021-02-12 17:54:00 +00:00
|
|
|
def fill_out_missing_chromecast_info(self) -> ChromecastInfo:
|
2021-01-06 08:23:18 +00:00
|
|
|
"""Return a new ChromecastInfo object with missing attributes filled in.
|
|
|
|
|
|
|
|
Uses blocking HTTP / HTTPS.
|
|
|
|
"""
|
|
|
|
if self.is_information_complete:
|
|
|
|
# We have all information, no need to check HTTP API.
|
|
|
|
return self
|
|
|
|
|
|
|
|
# Fill out missing group information via HTTP API.
|
|
|
|
if self.is_audio_group:
|
|
|
|
is_dynamic_group = False
|
|
|
|
http_group_status = None
|
|
|
|
if self.uuid:
|
|
|
|
http_group_status = dial.get_multizone_status(
|
2021-02-26 12:43:53 +00:00
|
|
|
None,
|
2021-01-06 08:23:18 +00:00
|
|
|
services=self.services,
|
|
|
|
zconf=ChromeCastZeroconf.get_zeroconf(),
|
|
|
|
)
|
|
|
|
if http_group_status is not None:
|
|
|
|
is_dynamic_group = any(
|
|
|
|
str(g.uuid) == self.uuid
|
|
|
|
for g in http_group_status.dynamic_groups
|
|
|
|
)
|
|
|
|
|
|
|
|
return ChromecastInfo(
|
|
|
|
services=self.services,
|
|
|
|
uuid=self.uuid,
|
|
|
|
friendly_name=self.friendly_name,
|
|
|
|
model_name=self.model_name,
|
2021-02-26 12:43:53 +00:00
|
|
|
is_audio_group=True,
|
2021-01-06 08:23:18 +00:00
|
|
|
is_dynamic_group=is_dynamic_group,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Fill out some missing information (friendly_name, uuid) via HTTP dial.
|
|
|
|
http_device_status = dial.get_device_status(
|
2021-02-26 12:43:53 +00:00
|
|
|
None, services=self.services, zconf=ChromeCastZeroconf.get_zeroconf()
|
2021-01-06 08:23:18 +00:00
|
|
|
)
|
|
|
|
if http_device_status is None:
|
|
|
|
# HTTP dial didn't give us any new information.
|
|
|
|
return self
|
|
|
|
|
|
|
|
return ChromecastInfo(
|
|
|
|
services=self.services,
|
|
|
|
uuid=(self.uuid or http_device_status.uuid),
|
|
|
|
friendly_name=(self.friendly_name or http_device_status.friendly_name),
|
|
|
|
manufacturer=(self.manufacturer or http_device_status.manufacturer),
|
|
|
|
model_name=(self.model_name or http_device_status.model_name),
|
|
|
|
)
|
|
|
|
|
2019-09-10 20:05:46 +00:00
|
|
|
|
|
|
|
class ChromeCastZeroconf:
|
|
|
|
"""Class to hold a zeroconf instance."""
|
|
|
|
|
|
|
|
__zconf = None
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def set_zeroconf(cls, zconf):
|
|
|
|
"""Set zeroconf."""
|
|
|
|
cls.__zconf = zconf
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_zeroconf(cls):
|
|
|
|
"""Get zeroconf."""
|
|
|
|
return cls.__zconf
|
|
|
|
|
|
|
|
|
|
|
|
class CastStatusListener:
|
|
|
|
"""Helper class to handle pychromecast status callbacks.
|
|
|
|
|
|
|
|
Necessary because a CastDevice entity can create a new socket client
|
|
|
|
and therefore callbacks from multiple chromecast connections can
|
|
|
|
potentially arrive. This class allows invalidating past chromecast objects.
|
|
|
|
"""
|
|
|
|
|
2021-01-06 08:23:18 +00:00
|
|
|
def __init__(self, cast_device, chromecast, mz_mgr, mz_only=False):
|
2019-09-10 20:05:46 +00:00
|
|
|
"""Initialize the status listener."""
|
|
|
|
self._cast_device = cast_device
|
|
|
|
self._uuid = chromecast.uuid
|
|
|
|
self._valid = True
|
|
|
|
self._mz_mgr = mz_mgr
|
|
|
|
|
2021-01-06 08:23:18 +00:00
|
|
|
if cast_device._cast_info.is_audio_group:
|
|
|
|
self._mz_mgr.add_multizone(chromecast)
|
|
|
|
if mz_only:
|
|
|
|
return
|
|
|
|
|
2019-09-10 20:05:46 +00:00
|
|
|
chromecast.register_status_listener(self)
|
|
|
|
chromecast.socket_client.media_controller.register_status_listener(self)
|
|
|
|
chromecast.register_connection_listener(self)
|
2021-01-06 08:23:18 +00:00
|
|
|
if not cast_device._cast_info.is_audio_group:
|
2019-09-10 20:05:46 +00:00
|
|
|
self._mz_mgr.register_listener(chromecast.uuid, self)
|
|
|
|
|
|
|
|
def new_cast_status(self, cast_status):
|
|
|
|
"""Handle reception of a new CastStatus."""
|
|
|
|
if self._valid:
|
|
|
|
self._cast_device.new_cast_status(cast_status)
|
|
|
|
|
|
|
|
def new_media_status(self, media_status):
|
|
|
|
"""Handle reception of a new MediaStatus."""
|
|
|
|
if self._valid:
|
|
|
|
self._cast_device.new_media_status(media_status)
|
|
|
|
|
|
|
|
def new_connection_status(self, connection_status):
|
|
|
|
"""Handle reception of a new ConnectionStatus."""
|
|
|
|
if self._valid:
|
|
|
|
self._cast_device.new_connection_status(connection_status)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def added_to_multizone(group_uuid):
|
|
|
|
"""Handle the cast added to a group."""
|
|
|
|
|
|
|
|
def removed_from_multizone(self, group_uuid):
|
|
|
|
"""Handle the cast removed from a group."""
|
|
|
|
if self._valid:
|
|
|
|
self._cast_device.multizone_new_media_status(group_uuid, None)
|
|
|
|
|
|
|
|
def multizone_new_cast_status(self, group_uuid, cast_status):
|
|
|
|
"""Handle reception of a new CastStatus for a group."""
|
|
|
|
|
|
|
|
def multizone_new_media_status(self, group_uuid, media_status):
|
|
|
|
"""Handle reception of a new MediaStatus for a group."""
|
|
|
|
if self._valid:
|
|
|
|
self._cast_device.multizone_new_media_status(group_uuid, media_status)
|
|
|
|
|
|
|
|
def invalidate(self):
|
|
|
|
"""Invalidate this status listener.
|
|
|
|
|
|
|
|
All following callbacks won't be forwarded.
|
|
|
|
"""
|
|
|
|
# pylint: disable=protected-access
|
|
|
|
if self._cast_device._cast_info.is_audio_group:
|
|
|
|
self._mz_mgr.remove_multizone(self._uuid)
|
|
|
|
else:
|
|
|
|
self._mz_mgr.deregister_listener(self._uuid, self)
|
|
|
|
self._valid = False
|