2016-01-10 05:30:26 +00:00
|
|
|
"""
|
|
|
|
homeassistant.components.media_player.universal
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
Combines multiple media players into one for a universal controller.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/media_player.universal/
|
|
|
|
"""
|
|
|
|
|
|
|
|
# pylint: disable=import-error
|
2016-01-12 02:56:07 +00:00
|
|
|
from copy import copy
|
2016-01-10 05:30:26 +00:00
|
|
|
import logging
|
|
|
|
|
2016-01-12 02:56:07 +00:00
|
|
|
from homeassistant.helpers.event import track_state_change
|
2016-01-12 05:59:57 +00:00
|
|
|
from homeassistant.helpers.service import call_from_config
|
2016-01-12 02:56:07 +00:00
|
|
|
|
2016-01-10 05:30:26 +00:00
|
|
|
from homeassistant.const import (
|
2016-01-12 02:56:07 +00:00
|
|
|
STATE_IDLE, STATE_ON, STATE_OFF, CONF_NAME,
|
|
|
|
ATTR_ENTITY_ID, ATTR_ENTITY_PICTURE,
|
2016-01-10 05:30:26 +00:00
|
|
|
SERVICE_TURN_OFF, SERVICE_TURN_ON,
|
|
|
|
SERVICE_VOLUME_UP, SERVICE_VOLUME_DOWN, SERVICE_VOLUME_SET,
|
|
|
|
SERVICE_VOLUME_MUTE,
|
|
|
|
SERVICE_MEDIA_PLAY_PAUSE, SERVICE_MEDIA_PLAY, SERVICE_MEDIA_PAUSE,
|
|
|
|
SERVICE_MEDIA_NEXT_TRACK, SERVICE_MEDIA_PREVIOUS_TRACK, SERVICE_MEDIA_SEEK)
|
|
|
|
|
|
|
|
from homeassistant.components.media_player import (
|
|
|
|
MediaPlayerDevice, DOMAIN,
|
2016-01-12 06:05:02 +00:00
|
|
|
SUPPORT_VOLUME_STEP, SUPPORT_VOLUME_SET, SUPPORT_VOLUME_MUTE,
|
2016-01-10 05:30:26 +00:00
|
|
|
SUPPORT_TURN_ON, SUPPORT_TURN_OFF,
|
|
|
|
SERVICE_PLAY_MEDIA, SERVICE_YOUTUBE_VIDEO,
|
|
|
|
ATTR_SUPPORTED_MEDIA_COMMANDS, ATTR_MEDIA_VOLUME_MUTED,
|
|
|
|
ATTR_MEDIA_CONTENT_ID, ATTR_MEDIA_CONTENT_TYPE, ATTR_MEDIA_DURATION,
|
|
|
|
ATTR_MEDIA_TITLE, ATTR_MEDIA_ARTIST, ATTR_MEDIA_ALBUM_NAME,
|
|
|
|
ATTR_MEDIA_TRACK, ATTR_MEDIA_SERIES_TITLE, ATTR_MEDIA_ALBUM_ARTIST,
|
|
|
|
ATTR_MEDIA_SEASON, ATTR_MEDIA_EPISODE, ATTR_MEDIA_CHANNEL,
|
|
|
|
ATTR_MEDIA_PLAYLIST, ATTR_APP_ID, ATTR_APP_NAME, ATTR_MEDIA_VOLUME_LEVEL,
|
|
|
|
ATTR_MEDIA_SEEK_POSITION)
|
|
|
|
|
2016-01-10 05:43:27 +00:00
|
|
|
ATTR_ACTIVE_CHILD = 'active_child'
|
|
|
|
|
2016-01-10 05:30:26 +00:00
|
|
|
CONF_ATTRS = 'attributes'
|
|
|
|
CONF_CHILDREN = 'children'
|
|
|
|
CONF_COMMANDS = 'commands'
|
|
|
|
CONF_PLATFORM = 'platform'
|
|
|
|
CONF_SERVICE = 'service'
|
|
|
|
CONF_SERVICE_DATA = 'service_data'
|
|
|
|
CONF_STATE = 'state'
|
2016-01-10 05:43:27 +00:00
|
|
|
|
2016-01-10 05:30:26 +00:00
|
|
|
OFF_STATES = [STATE_IDLE, STATE_OFF]
|
|
|
|
REQUIREMENTS = []
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
# pylint: disable=unused-argument
|
|
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
|
|
|
""" sets up the universal media players """
|
|
|
|
if not validate_config(config):
|
|
|
|
return
|
|
|
|
|
2016-01-12 02:56:07 +00:00
|
|
|
player = UniversalMediaPlayer(hass,
|
|
|
|
config[CONF_NAME],
|
2016-01-10 05:30:26 +00:00
|
|
|
config[CONF_CHILDREN],
|
|
|
|
config[CONF_COMMANDS],
|
|
|
|
config[CONF_ATTRS])
|
|
|
|
|
|
|
|
add_devices([player])
|
|
|
|
|
|
|
|
|
|
|
|
def validate_config(config):
|
|
|
|
""" validate universal media player configuration """
|
|
|
|
del config[CONF_PLATFORM]
|
|
|
|
|
|
|
|
# validate name
|
|
|
|
if CONF_NAME not in config:
|
|
|
|
_LOGGER.error('Universal Media Player configuration requires name')
|
|
|
|
return False
|
|
|
|
|
|
|
|
validate_children(config)
|
|
|
|
validate_commands(config)
|
|
|
|
validate_attributes(config)
|
|
|
|
|
|
|
|
del_keys = []
|
|
|
|
for key in config:
|
|
|
|
if key not in [CONF_NAME, CONF_CHILDREN, CONF_COMMANDS, CONF_ATTRS]:
|
|
|
|
_LOGGER.warning(
|
|
|
|
'Universal Media Player (%s) unrecognized parameter %s',
|
|
|
|
config[CONF_NAME], key)
|
|
|
|
del_keys.append(key)
|
|
|
|
for key in del_keys:
|
|
|
|
del config[key]
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
def validate_children(config):
|
|
|
|
""" validate children """
|
|
|
|
if CONF_CHILDREN not in config:
|
|
|
|
_LOGGER.info(
|
|
|
|
'No children under Universal Media Player (%s)', config[CONF_NAME])
|
|
|
|
config[CONF_CHILDREN] = []
|
|
|
|
elif not isinstance(config[CONF_CHILDREN], list):
|
|
|
|
_LOGGER.warning(
|
|
|
|
'Universal Media Player (%s) children not list in config. '
|
|
|
|
'They will be ignored.',
|
|
|
|
config[CONF_NAME])
|
|
|
|
config[CONF_CHILDREN] = []
|
|
|
|
|
|
|
|
|
|
|
|
def validate_commands(config):
|
|
|
|
""" validate commands """
|
|
|
|
if CONF_COMMANDS not in config:
|
|
|
|
config[CONF_COMMANDS] = {}
|
|
|
|
elif not isinstance(config[CONF_COMMANDS], dict):
|
|
|
|
_LOGGER.warning(
|
2016-01-12 02:56:07 +00:00
|
|
|
'Universal Media Player (%s) specified commands not dict in '
|
|
|
|
'config. They will be ignored.',
|
2016-01-10 05:30:26 +00:00
|
|
|
config[CONF_NAME])
|
|
|
|
config[CONF_COMMANDS] = {}
|
|
|
|
|
|
|
|
|
|
|
|
def validate_attributes(config):
|
|
|
|
""" validate attributes """
|
|
|
|
if CONF_ATTRS not in config:
|
|
|
|
config[CONF_ATTRS] = {}
|
|
|
|
elif not isinstance(config[CONF_ATTRS], dict):
|
|
|
|
_LOGGER.warning(
|
|
|
|
'Universal Media Player (%s) specified attributes '
|
|
|
|
'not dict in config. They will be ignored.',
|
|
|
|
config[CONF_NAME])
|
|
|
|
config[CONF_ATTRS] = {}
|
|
|
|
|
|
|
|
for key, val in config[CONF_ATTRS].items():
|
|
|
|
config[CONF_ATTRS][key] = val.split('|', 1)
|
|
|
|
|
|
|
|
|
|
|
|
class UniversalMediaPlayer(MediaPlayerDevice):
|
|
|
|
""" Represents a universal media player in HA """
|
2016-01-12 02:56:07 +00:00
|
|
|
# pylint: disable=too-many-public-methods
|
2016-01-10 05:30:26 +00:00
|
|
|
|
2016-01-12 02:56:07 +00:00
|
|
|
def __init__(self, hass, name, children, commands, attributes):
|
|
|
|
# pylint: disable=too-many-arguments
|
|
|
|
self.hass = hass
|
2016-01-10 05:30:26 +00:00
|
|
|
self._name = name
|
|
|
|
self._children = children
|
|
|
|
self._cmds = commands
|
|
|
|
self._attrs = attributes
|
2016-01-12 06:34:25 +00:00
|
|
|
self._child_state = None
|
2016-01-10 05:30:26 +00:00
|
|
|
|
2016-01-12 02:56:07 +00:00
|
|
|
track_state_change(hass, self.dependencies, self.update_state)
|
2016-01-10 05:30:26 +00:00
|
|
|
|
2016-01-12 06:16:57 +00:00
|
|
|
def _entity_lkp(self, entity_id=None, state_attr=None):
|
2016-01-10 05:30:26 +00:00
|
|
|
""" Looks up an entity state from hass """
|
|
|
|
if entity_id is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
state_obj = self.hass.states.get(entity_id)
|
|
|
|
|
|
|
|
if state_obj is None:
|
|
|
|
return
|
|
|
|
|
2016-01-11 01:34:30 +00:00
|
|
|
if state_attr:
|
|
|
|
return state_obj.attributes.get(state_attr)
|
|
|
|
return state_obj.state
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
def _override_or_child_attr(self, attr_name):
|
|
|
|
""" returns either the override or the active child for attr_name """
|
|
|
|
if attr_name in self._attrs:
|
2016-01-12 06:16:57 +00:00
|
|
|
return self._entity_lkp(*self._attrs[attr_name])
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
return self._child_attr(attr_name)
|
|
|
|
|
|
|
|
def _child_attr(self, attr_name):
|
|
|
|
""" returns the active child's attr """
|
|
|
|
active_child = self.active_child_state
|
|
|
|
return active_child.attributes.get(attr_name) if active_child else None
|
|
|
|
|
2016-01-12 06:39:02 +00:00
|
|
|
def _override_or_child_service(self, service_name, service_data):
|
2016-01-10 05:30:26 +00:00
|
|
|
""" calls either a specified or active child's service """
|
|
|
|
if service_name in self._cmds:
|
2016-01-12 05:59:57 +00:00
|
|
|
call_from_config(
|
|
|
|
self.hass, self._cmds[service_name], blocking=True)
|
2016-01-10 05:30:26 +00:00
|
|
|
return
|
|
|
|
|
2016-01-12 06:39:02 +00:00
|
|
|
self._child_service(service_name, service_data)
|
2016-01-10 05:30:26 +00:00
|
|
|
|
2016-01-12 06:39:02 +00:00
|
|
|
def _child_service(self, service_name, service_data):
|
2016-01-10 05:30:26 +00:00
|
|
|
""" calls the active child's specified service """
|
|
|
|
active_child = self.active_child_state
|
|
|
|
service_data[ATTR_ENTITY_ID] = active_child.entity_id
|
|
|
|
|
|
|
|
self.hass.services.call(DOMAIN, service_name, service_data,
|
|
|
|
blocking=True)
|
|
|
|
|
2016-01-12 02:56:07 +00:00
|
|
|
@property
|
|
|
|
def dependencies(self):
|
|
|
|
""" List of entity ids of entities that the mp depends on for state """
|
|
|
|
depend = copy(self._children)
|
|
|
|
for entity in self._attrs.values():
|
|
|
|
depend.append(entity[0])
|
|
|
|
return depend
|
|
|
|
|
2016-01-10 05:30:26 +00:00
|
|
|
@property
|
|
|
|
def master_state(self):
|
|
|
|
""" gets the master state from entity or none """
|
|
|
|
if CONF_STATE in self._attrs:
|
2016-01-12 06:16:57 +00:00
|
|
|
master_state = self._entity_lkp(*self._attrs[CONF_STATE])
|
2016-01-11 01:34:30 +00:00
|
|
|
return master_state if master_state else STATE_OFF
|
2016-01-10 05:30:26 +00:00
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
2016-01-12 06:34:25 +00:00
|
|
|
def _cache_active_child_state(self):
|
2016-01-10 05:30:26 +00:00
|
|
|
""" The state of the active child or None """
|
2016-01-12 05:43:34 +00:00
|
|
|
for child_name in self._children:
|
2016-01-12 06:16:57 +00:00
|
|
|
child_state = self.hass.states.get(child_name)
|
2016-01-10 05:30:26 +00:00
|
|
|
if child_state and child_state.state not in OFF_STATES:
|
2016-01-12 06:34:25 +00:00
|
|
|
self._child_state = child_state
|
|
|
|
return
|
|
|
|
self._child_state = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def active_child_state(self):
|
|
|
|
""" the state of the active child or none """
|
|
|
|
return self._child_state
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
""" name of universal player """
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""
|
|
|
|
Current state of media player
|
|
|
|
|
|
|
|
Off if master state is off
|
|
|
|
ELSE Status of first active child
|
|
|
|
ELSE master state or off
|
|
|
|
"""
|
|
|
|
master_state = self.master_state # avoid multiple lookups
|
|
|
|
if master_state == STATE_OFF:
|
|
|
|
return STATE_OFF
|
|
|
|
|
|
|
|
active_child = self.active_child_state
|
|
|
|
if active_child:
|
|
|
|
return active_child.state
|
|
|
|
|
|
|
|
return master_state if master_state else STATE_OFF
|
|
|
|
|
|
|
|
@property
|
|
|
|
def volume_level(self):
|
|
|
|
""" Volume level of entity specified in attributes or active child """
|
|
|
|
return self._child_attr(ATTR_MEDIA_VOLUME_LEVEL)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_volume_muted(self):
|
|
|
|
""" boolean if volume is muted """
|
2016-01-12 02:56:07 +00:00
|
|
|
return self._override_or_child_attr(ATTR_MEDIA_VOLUME_MUTED) \
|
|
|
|
in [True, STATE_ON]
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_content_id(self):
|
|
|
|
""" Content ID of current playing media. """
|
|
|
|
return self._child_attr(ATTR_MEDIA_CONTENT_ID)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_content_type(self):
|
|
|
|
""" Content type of current playing media. """
|
|
|
|
return self._child_attr(ATTR_MEDIA_CONTENT_TYPE)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_duration(self):
|
|
|
|
""" Duration of current playing media in seconds. """
|
|
|
|
return self._child_attr(ATTR_MEDIA_DURATION)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_image_url(self):
|
|
|
|
""" Image url of current playing media. """
|
|
|
|
return self._child_attr(ATTR_ENTITY_PICTURE)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_title(self):
|
|
|
|
""" Title of current playing media. """
|
|
|
|
return self._child_attr(ATTR_MEDIA_TITLE)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_artist(self):
|
|
|
|
""" Artist of current playing media. (Music track only) """
|
|
|
|
return self._child_attr(ATTR_MEDIA_ARTIST)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_album_name(self):
|
|
|
|
""" Album name of current playing media. (Music track only) """
|
|
|
|
return self._child_attr(ATTR_MEDIA_ALBUM_NAME)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_album_artist(self):
|
|
|
|
""" Album arist of current playing media. (Music track only) """
|
|
|
|
return self._child_attr(ATTR_MEDIA_ALBUM_ARTIST)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_track(self):
|
|
|
|
""" Track number of current playing media. (Music track only) """
|
|
|
|
return self._child_attr(ATTR_MEDIA_TRACK)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_series_title(self):
|
|
|
|
""" Series title of current playing media. (TV Show only)"""
|
|
|
|
return self._child_attr(ATTR_MEDIA_SERIES_TITLE)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_season(self):
|
|
|
|
""" Season of current playing media. (TV Show only) """
|
|
|
|
return self._child_attr(ATTR_MEDIA_SEASON)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_episode(self):
|
|
|
|
""" Episode of current playing media. (TV Show only) """
|
|
|
|
return self._child_attr(ATTR_MEDIA_EPISODE)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_channel(self):
|
|
|
|
""" Channel currently playing. """
|
|
|
|
return self._child_attr(ATTR_MEDIA_CHANNEL)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_playlist(self):
|
|
|
|
""" Title of Playlist currently playing. """
|
|
|
|
return self._child_attr(ATTR_MEDIA_PLAYLIST)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def app_id(self):
|
|
|
|
""" ID of the current running app. """
|
|
|
|
return self._child_attr(ATTR_APP_ID)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def app_name(self):
|
|
|
|
""" Name of the current running app. """
|
|
|
|
return self._child_attr(ATTR_APP_NAME)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_media_commands(self):
|
|
|
|
""" Flags of media commands that are supported. """
|
|
|
|
flags = self._child_attr(ATTR_SUPPORTED_MEDIA_COMMANDS) or 0
|
|
|
|
|
|
|
|
if SERVICE_TURN_ON in self._cmds:
|
|
|
|
flags |= SUPPORT_TURN_ON
|
|
|
|
if SERVICE_TURN_OFF in self._cmds:
|
|
|
|
flags |= SUPPORT_TURN_OFF
|
|
|
|
|
|
|
|
if any([cmd in self._cmds for cmd in [SERVICE_VOLUME_UP,
|
|
|
|
SERVICE_VOLUME_DOWN]]):
|
2016-01-12 06:05:02 +00:00
|
|
|
flags |= SUPPORT_VOLUME_STEP
|
2016-01-10 05:30:26 +00:00
|
|
|
flags &= ~SUPPORT_VOLUME_SET
|
|
|
|
|
|
|
|
if SERVICE_VOLUME_MUTE in self._cmds and \
|
|
|
|
ATTR_MEDIA_VOLUME_MUTED in self._attrs:
|
|
|
|
flags |= SUPPORT_VOLUME_MUTE
|
|
|
|
|
|
|
|
return flags
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
2016-01-10 05:43:27 +00:00
|
|
|
""" Extra attributes a device wants to expose. """
|
|
|
|
active_child = self.active_child_state
|
|
|
|
return {ATTR_ACTIVE_CHILD: active_child.entity_id} \
|
|
|
|
if active_child else {}
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
def turn_on(self):
|
|
|
|
""" turn the media player on. """
|
|
|
|
self._override_or_child_service(SERVICE_TURN_ON)
|
|
|
|
|
|
|
|
def turn_off(self):
|
|
|
|
""" turn the media player off. """
|
|
|
|
self._override_or_child_service(SERVICE_TURN_OFF)
|
|
|
|
|
|
|
|
def mute_volume(self, is_volume_muted):
|
|
|
|
""" mute the volume. """
|
|
|
|
data = {ATTR_MEDIA_VOLUME_MUTED: is_volume_muted}
|
2016-01-12 06:39:02 +00:00
|
|
|
self._override_or_child_service(SERVICE_VOLUME_MUTE, data)
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
def set_volume_level(self, volume_level):
|
|
|
|
""" set volume level, range 0..1. """
|
|
|
|
data = {ATTR_MEDIA_VOLUME_LEVEL: volume_level}
|
2016-01-12 06:39:02 +00:00
|
|
|
self._child_service(SERVICE_VOLUME_SET, data)
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
def media_play(self):
|
|
|
|
""" Send play commmand. """
|
|
|
|
self._child_service(SERVICE_MEDIA_PLAY)
|
|
|
|
|
|
|
|
def media_pause(self):
|
|
|
|
""" Send pause command. """
|
|
|
|
self._child_service(SERVICE_MEDIA_PAUSE)
|
|
|
|
|
|
|
|
def media_previous_track(self):
|
|
|
|
""" Send previous track command. """
|
|
|
|
self._child_service(SERVICE_MEDIA_PREVIOUS_TRACK)
|
|
|
|
|
|
|
|
def media_next_track(self):
|
|
|
|
""" Send next track command. """
|
|
|
|
self._child_service(SERVICE_MEDIA_NEXT_TRACK)
|
|
|
|
|
|
|
|
def media_seek(self, position):
|
|
|
|
""" Send seek command. """
|
|
|
|
data = {ATTR_MEDIA_SEEK_POSITION: position}
|
2016-01-12 06:39:02 +00:00
|
|
|
self._child_service(SERVICE_MEDIA_SEEK, data)
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
def play_youtube(self, media_id):
|
|
|
|
""" Plays a YouTube media. """
|
2016-01-12 06:39:02 +00:00
|
|
|
data = {'media_id': media_id}
|
|
|
|
self._child_service(SERVICE_YOUTUBE_VIDEO, data)
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
def play_media(self, media_type, media_id):
|
|
|
|
""" Plays a piece of media. """
|
2016-01-12 06:39:02 +00:00
|
|
|
data = {'media_type': media_type, 'media_id': media_id}
|
|
|
|
self._child_service(SERVICE_PLAY_MEDIA, data)
|
2016-01-10 05:30:26 +00:00
|
|
|
|
|
|
|
def volume_up(self):
|
|
|
|
""" volume_up media player. """
|
|
|
|
self._override_or_child_service(SERVICE_VOLUME_UP)
|
|
|
|
|
|
|
|
def volume_down(self):
|
|
|
|
""" volume_down media player. """
|
|
|
|
self._override_or_child_service(SERVICE_VOLUME_DOWN)
|
|
|
|
|
|
|
|
def media_play_pause(self):
|
|
|
|
""" media_play_pause media player. """
|
|
|
|
self._child_service(SERVICE_MEDIA_PLAY_PAUSE)
|
2016-01-12 02:56:07 +00:00
|
|
|
|
|
|
|
def update_state(self, *_):
|
|
|
|
""" event to trigger a state update in HA """
|
2016-01-12 06:34:25 +00:00
|
|
|
self._cache_active_child_state()
|
2016-01-12 02:56:07 +00:00
|
|
|
self.update_ha_state()
|