2019-04-03 15:40:03 +00:00
|
|
|
"""Support for the DirecTV receivers."""
|
2021-03-17 22:43:55 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2018-10-29 20:06:37 +00:00
|
|
|
import logging
|
2019-12-05 05:23:05 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
from directv import DIRECTV
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-09-03 21:27:13 +00:00
|
|
|
from homeassistant.components.media_player import (
|
2021-12-09 11:06:04 +00:00
|
|
|
MediaPlayerDeviceClass,
|
2020-09-03 21:27:13 +00:00
|
|
|
MediaPlayerEntity,
|
|
|
|
)
|
2019-02-08 22:18:18 +00:00
|
|
|
from homeassistant.components.media_player.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
MEDIA_TYPE_CHANNEL,
|
|
|
|
MEDIA_TYPE_MOVIE,
|
2020-04-16 01:08:54 +00:00
|
|
|
MEDIA_TYPE_MUSIC,
|
2019-07-31 19:25:30 +00:00
|
|
|
MEDIA_TYPE_TVSHOW,
|
|
|
|
SUPPORT_NEXT_TRACK,
|
|
|
|
SUPPORT_PAUSE,
|
|
|
|
SUPPORT_PLAY,
|
|
|
|
SUPPORT_PLAY_MEDIA,
|
|
|
|
SUPPORT_PREVIOUS_TRACK,
|
|
|
|
SUPPORT_STOP,
|
|
|
|
SUPPORT_TURN_OFF,
|
|
|
|
SUPPORT_TURN_ON,
|
|
|
|
)
|
2020-03-11 19:28:38 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2020-03-31 22:35:32 +00:00
|
|
|
from homeassistant.const import STATE_OFF, STATE_PAUSED, STATE_PLAYING
|
2021-04-17 10:48:03 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2021-04-29 10:28:14 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2020-03-11 19:28:38 +00:00
|
|
|
from homeassistant.util import dt as dt_util
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-02-25 00:30:00 +00:00
|
|
|
from .const import (
|
|
|
|
ATTR_MEDIA_CURRENTLY_RECORDING,
|
|
|
|
ATTR_MEDIA_RATING,
|
|
|
|
ATTR_MEDIA_RECORDED,
|
|
|
|
ATTR_MEDIA_START_TIME,
|
2020-03-11 19:28:38 +00:00
|
|
|
DOMAIN,
|
2020-02-25 00:30:00 +00:00
|
|
|
)
|
2021-06-23 14:10:29 +00:00
|
|
|
from .entity import DIRECTVEntity
|
2018-10-29 20:06:37 +00:00
|
|
|
|
2020-02-25 00:30:00 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-04-16 01:08:54 +00:00
|
|
|
KNOWN_MEDIA_TYPES = [MEDIA_TYPE_MOVIE, MEDIA_TYPE_MUSIC, MEDIA_TYPE_TVSHOW]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_DTV = (
|
|
|
|
SUPPORT_PAUSE
|
|
|
|
| SUPPORT_TURN_ON
|
|
|
|
| SUPPORT_TURN_OFF
|
|
|
|
| SUPPORT_PLAY_MEDIA
|
|
|
|
| SUPPORT_STOP
|
|
|
|
| SUPPORT_NEXT_TRACK
|
|
|
|
| SUPPORT_PREVIOUS_TRACK
|
|
|
|
| SUPPORT_PLAY
|
|
|
|
)
|
|
|
|
|
|
|
|
SUPPORT_DTV_CLIENT = (
|
|
|
|
SUPPORT_PAUSE
|
|
|
|
| SUPPORT_PLAY_MEDIA
|
|
|
|
| SUPPORT_STOP
|
|
|
|
| SUPPORT_NEXT_TRACK
|
|
|
|
| SUPPORT_PREVIOUS_TRACK
|
|
|
|
| SUPPORT_PLAY
|
|
|
|
)
|
|
|
|
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-11 19:28:38 +00:00
|
|
|
async def async_setup_entry(
|
2021-04-17 10:48:03 +00:00
|
|
|
hass: HomeAssistant,
|
2020-03-11 19:28:38 +00:00
|
|
|
entry: ConfigEntry,
|
2021-04-29 10:28:14 +00:00
|
|
|
async_add_entities: AddEntitiesCallback,
|
2021-09-15 05:58:04 +00:00
|
|
|
) -> None:
|
2020-03-11 19:28:38 +00:00
|
|
|
"""Set up the DirecTV config entry."""
|
2020-03-31 22:35:32 +00:00
|
|
|
dtv = hass.data[DOMAIN][entry.entry_id]
|
2020-03-11 19:28:38 +00:00
|
|
|
entities = []
|
2018-11-05 15:19:03 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
for location in dtv.device.locations:
|
2020-03-11 19:28:38 +00:00
|
|
|
entities.append(
|
2020-03-31 22:35:32 +00:00
|
|
|
DIRECTVMediaPlayer(
|
2020-08-27 11:56:20 +00:00
|
|
|
dtv=dtv,
|
|
|
|
name=str.title(location.name),
|
|
|
|
address=location.address,
|
2020-03-11 19:28:38 +00:00
|
|
|
)
|
|
|
|
)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-11 19:28:38 +00:00
|
|
|
async_add_entities(entities, True)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
class DIRECTVMediaPlayer(DIRECTVEntity, MediaPlayerEntity):
|
2017-09-23 15:15:46 +00:00
|
|
|
"""Representation of a DirecTV receiver on the network."""
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
def __init__(self, *, dtv: DIRECTV, name: str, address: str = "0") -> None:
|
|
|
|
"""Initialize DirecTV media player."""
|
|
|
|
super().__init__(
|
2020-08-27 11:56:20 +00:00
|
|
|
dtv=dtv,
|
|
|
|
address=address,
|
2020-03-31 22:35:32 +00:00
|
|
|
)
|
|
|
|
|
2021-06-23 14:10:29 +00:00
|
|
|
self._attr_unique_id = self._device_id
|
|
|
|
self._attr_name = name
|
2021-12-09 11:06:04 +00:00
|
|
|
self._attr_device_class = MediaPlayerDeviceClass.RECEIVER
|
2021-06-23 14:10:29 +00:00
|
|
|
self._attr_available = False
|
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
self._is_recorded = None
|
2016-07-26 06:20:56 +00:00
|
|
|
self._is_standby = True
|
2020-03-31 22:35:32 +00:00
|
|
|
self._last_position = None
|
2018-10-29 20:06:37 +00:00
|
|
|
self._last_update = None
|
|
|
|
self._paused = None
|
2020-03-31 22:35:32 +00:00
|
|
|
self._program = None
|
|
|
|
self._state = None
|
2018-10-29 20:06:37 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_update(self):
|
2016-07-26 06:20:56 +00:00
|
|
|
"""Retrieve latest state."""
|
2020-03-31 22:35:32 +00:00
|
|
|
self._state = await self.dtv.state(self._address)
|
2021-06-23 14:10:29 +00:00
|
|
|
self._attr_available = self._state.available
|
2020-03-31 22:35:32 +00:00
|
|
|
self._is_standby = self._state.standby
|
|
|
|
self._program = self._state.program
|
2018-12-07 06:26:49 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby:
|
2021-06-23 14:10:29 +00:00
|
|
|
self._attr_assumed_state = False
|
2020-03-31 22:35:32 +00:00
|
|
|
self._is_recorded = None
|
|
|
|
self._last_position = None
|
|
|
|
self._last_update = None
|
|
|
|
self._paused = None
|
|
|
|
elif self._program is not None:
|
|
|
|
self._paused = self._last_position == self._program.position
|
|
|
|
self._is_recorded = self._program.recorded
|
|
|
|
self._last_position = self._program.position
|
|
|
|
self._last_update = self._state.at
|
2021-06-23 14:10:29 +00:00
|
|
|
self._attr_assumed_state = self._is_recorded
|
2018-12-07 06:26:49 +00:00
|
|
|
|
2018-10-29 20:06:37 +00:00
|
|
|
@property
|
2021-03-11 15:51:03 +00:00
|
|
|
def extra_state_attributes(self):
|
2018-10-29 20:06:37 +00:00
|
|
|
"""Return device specific state attributes."""
|
2020-10-06 16:08:53 +00:00
|
|
|
if self._is_standby:
|
|
|
|
return {}
|
|
|
|
return {
|
|
|
|
ATTR_MEDIA_CURRENTLY_RECORDING: self.media_currently_recording,
|
|
|
|
ATTR_MEDIA_RATING: self.media_rating,
|
|
|
|
ATTR_MEDIA_RECORDED: self.media_recorded,
|
|
|
|
ATTR_MEDIA_START_TIME: self.media_start_time,
|
|
|
|
}
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
# MediaPlayerEntity properties and methods
|
2016-07-26 06:20:56 +00:00
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the device."""
|
|
|
|
if self._is_standby:
|
|
|
|
return STATE_OFF
|
2018-10-29 20:06:37 +00:00
|
|
|
|
|
|
|
# For recorded media we can determine if it is paused or not.
|
|
|
|
# For live media we're unable to determine and will always return
|
|
|
|
# playing instead.
|
|
|
|
if self._paused:
|
|
|
|
return STATE_PAUSED
|
|
|
|
|
2017-07-06 03:02:16 +00:00
|
|
|
return STATE_PLAYING
|
2016-07-26 06:20:56 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_content_id(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Return the content ID of current playing media."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2016-07-26 06:20:56 +00:00
|
|
|
return None
|
2018-10-29 20:06:37 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.program_id
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2018-10-29 20:06:37 +00:00
|
|
|
@property
|
|
|
|
def media_content_type(self):
|
|
|
|
"""Return the content type of current playing media."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2018-10-29 20:06:37 +00:00
|
|
|
return None
|
|
|
|
|
2020-04-16 01:08:54 +00:00
|
|
|
if self._program.program_type in KNOWN_MEDIA_TYPES:
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.program_type
|
2018-10-29 20:06:37 +00:00
|
|
|
|
|
|
|
return MEDIA_TYPE_MOVIE
|
|
|
|
|
2016-07-26 06:20:56 +00:00
|
|
|
@property
|
|
|
|
def media_duration(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Return the duration of current playing media in seconds."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2016-07-26 06:20:56 +00:00
|
|
|
return None
|
2018-10-29 20:06:37 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.duration
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2018-10-29 20:06:37 +00:00
|
|
|
@property
|
|
|
|
def media_position(self):
|
|
|
|
"""Position of current playing media in seconds."""
|
|
|
|
if self._is_standby:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return self._last_position
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_position_updated_at(self):
|
2020-03-31 22:35:32 +00:00
|
|
|
"""When was the position of the current playing media valid."""
|
2018-10-29 20:06:37 +00:00
|
|
|
if self._is_standby:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return self._last_update
|
|
|
|
|
2016-07-26 06:20:56 +00:00
|
|
|
@property
|
|
|
|
def media_title(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Return the title of current playing media."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2016-07-26 06:20:56 +00:00
|
|
|
return None
|
2018-10-29 20:06:37 +00:00
|
|
|
|
2020-04-16 01:08:54 +00:00
|
|
|
if self.media_content_type == MEDIA_TYPE_MUSIC:
|
|
|
|
return self._program.music_title
|
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.title
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-04-16 01:08:54 +00:00
|
|
|
@property
|
|
|
|
def media_artist(self):
|
|
|
|
"""Artist of current playing media, music track only."""
|
|
|
|
if self._is_standby or self._program is None:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return self._program.music_artist
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_album_name(self):
|
|
|
|
"""Album name of current playing media, music track only."""
|
|
|
|
if self._is_standby or self._program is None:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return self._program.music_album
|
|
|
|
|
2016-07-26 06:20:56 +00:00
|
|
|
@property
|
|
|
|
def media_series_title(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Return the title of current episode of TV show."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2016-07-26 06:20:56 +00:00
|
|
|
return None
|
2018-10-29 20:06:37 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.episode_title
|
2018-10-29 20:06:37 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_channel(self):
|
|
|
|
"""Return the channel current playing media."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2018-10-29 20:06:37 +00:00
|
|
|
return None
|
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return f"{self._program.channel_name} ({self._program.channel})"
|
2018-10-29 20:06:37 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def source(self):
|
|
|
|
"""Name of the current input source."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2018-10-29 20:06:37 +00:00
|
|
|
return None
|
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.channel
|
2016-07-26 06:20:56 +00:00
|
|
|
|
|
|
|
@property
|
2017-02-08 04:42:45 +00:00
|
|
|
def supported_features(self):
|
|
|
|
"""Flag media player features that are supported."""
|
2018-11-19 10:47:00 +00:00
|
|
|
return SUPPORT_DTV_CLIENT if self._is_client else SUPPORT_DTV
|
2016-07-26 06:20:56 +00:00
|
|
|
|
|
|
|
@property
|
2018-10-29 20:06:37 +00:00
|
|
|
def media_currently_recording(self):
|
|
|
|
"""If the media is currently being recorded or not."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2018-10-29 20:06:37 +00:00
|
|
|
return None
|
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.recording
|
2016-07-26 06:20:56 +00:00
|
|
|
|
|
|
|
@property
|
2018-10-29 20:06:37 +00:00
|
|
|
def media_rating(self):
|
|
|
|
"""TV Rating of the current playing media."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2016-07-26 06:20:56 +00:00
|
|
|
return None
|
2017-07-06 06:30:01 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return self._program.rating
|
2018-10-29 20:06:37 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_recorded(self):
|
|
|
|
"""If the media was recorded or live."""
|
|
|
|
if self._is_standby:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return self._is_recorded
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_start_time(self):
|
|
|
|
"""Start time the program aired."""
|
2020-03-31 22:35:32 +00:00
|
|
|
if self._is_standby or self._program is None:
|
2018-10-29 20:06:37 +00:00
|
|
|
return None
|
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
return dt_util.as_local(self._program.start_time)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_turn_on(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Turn on the receiver."""
|
2018-11-19 10:47:00 +00:00
|
|
|
if self._is_client:
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Turn on %s", self.name)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.remote("poweron", self._address)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_turn_off(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Turn off the receiver."""
|
2018-11-19 10:47:00 +00:00
|
|
|
if self._is_client:
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Turn off %s", self.name)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.remote("poweroff", self._address)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_media_play(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Send play command."""
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Play on %s", self.name)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.remote("play", self._address)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_media_pause(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Send pause command."""
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Pause on %s", self.name)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.remote("pause", self._address)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_media_stop(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Send stop command."""
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Stop on %s", self.name)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.remote("stop", self._address)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_media_previous_track(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Send rewind command."""
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Rewind on %s", self.name)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.remote("rew", self._address)
|
2016-07-26 06:20:56 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_media_next_track(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Send fast forward command."""
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Fast forward on %s", self.name)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.remote("ffwd", self._address)
|
2018-10-29 20:06:37 +00:00
|
|
|
|
2020-03-31 22:35:32 +00:00
|
|
|
async def async_play_media(self, media_type, media_id, **kwargs):
|
2018-10-29 20:06:37 +00:00
|
|
|
"""Select input source."""
|
2018-11-20 23:05:25 +00:00
|
|
|
if media_type != MEDIA_TYPE_CHANNEL:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Invalid media type %s. Only %s is supported",
|
|
|
|
media_type,
|
|
|
|
MEDIA_TYPE_CHANNEL,
|
|
|
|
)
|
2018-11-20 23:05:25 +00:00
|
|
|
return
|
|
|
|
|
2021-06-23 14:10:29 +00:00
|
|
|
_LOGGER.debug("Changing channel on %s to %s", self.name, media_id)
|
2020-03-31 22:35:32 +00:00
|
|
|
await self.dtv.tune(media_id, self._address)
|