2019-04-03 15:40:03 +00:00
|
|
|
"""Support for Russound multizone controllers using RIO Protocol."""
|
2019-12-01 05:21:40 +00:00
|
|
|
from russound_rio import Russound
|
2017-07-31 12:42:55 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
from homeassistant.components.media_player import PLATFORM_SCHEMA, 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_MUSIC,
|
|
|
|
SUPPORT_SELECT_SOURCE,
|
|
|
|
SUPPORT_TURN_OFF,
|
|
|
|
SUPPORT_TURN_ON,
|
|
|
|
SUPPORT_VOLUME_MUTE,
|
|
|
|
SUPPORT_VOLUME_SET,
|
|
|
|
)
|
2017-07-31 12:42:55 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_HOST,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_PORT,
|
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
|
|
STATE_OFF,
|
|
|
|
STATE_ON,
|
|
|
|
)
|
2018-09-09 12:26:06 +00:00
|
|
|
from homeassistant.core import callback
|
2017-07-31 12:42:55 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_RUSSOUND = (
|
|
|
|
SUPPORT_VOLUME_MUTE
|
|
|
|
| SUPPORT_VOLUME_SET
|
|
|
|
| SUPPORT_TURN_ON
|
|
|
|
| SUPPORT_TURN_OFF
|
|
|
|
| SUPPORT_SELECT_SOURCE
|
|
|
|
)
|
2017-07-31 12:42:55 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=9621): cv.port,
|
|
|
|
}
|
|
|
|
)
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def async_setup_platform(hass, config, async_add_entities, discovery_info=None):
|
2017-07-31 12:42:55 +00:00
|
|
|
"""Set up the Russound RIO platform."""
|
2018-09-09 12:26:06 +00:00
|
|
|
|
2017-07-31 12:42:55 +00:00
|
|
|
host = config.get(CONF_HOST)
|
|
|
|
port = config.get(CONF_PORT)
|
|
|
|
|
|
|
|
russ = Russound(hass.loop, host, port)
|
|
|
|
|
2018-10-01 06:58:21 +00:00
|
|
|
await russ.connect()
|
2017-07-31 12:42:55 +00:00
|
|
|
|
2018-09-09 12:26:06 +00:00
|
|
|
# Discover sources and zones
|
2018-10-01 06:58:21 +00:00
|
|
|
sources = await russ.enumerate_sources()
|
|
|
|
valid_zones = await russ.enumerate_zones()
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
devices = []
|
|
|
|
for zone_id, name in valid_zones:
|
2018-10-01 06:58:21 +00:00
|
|
|
await russ.watch_zone(zone_id)
|
2017-07-31 12:42:55 +00:00
|
|
|
dev = RussoundZoneDevice(russ, zone_id, name, sources)
|
|
|
|
devices.append(dev)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def on_stop(event):
|
|
|
|
"""Shutdown cleanly when hass stops."""
|
|
|
|
hass.loop.create_task(russ.close())
|
|
|
|
|
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, on_stop)
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
async_add_entities(devices)
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
class RussoundZoneDevice(MediaPlayerEntity):
|
2017-07-31 12:42:55 +00:00
|
|
|
"""Representation of a Russound Zone."""
|
|
|
|
|
|
|
|
def __init__(self, russ, zone_id, name, sources):
|
|
|
|
"""Initialize the zone device."""
|
|
|
|
super().__init__()
|
|
|
|
self._name = name
|
|
|
|
self._russ = russ
|
|
|
|
self._zone_id = zone_id
|
|
|
|
self._sources = sources
|
|
|
|
|
|
|
|
def _zone_var(self, name, default=None):
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._russ.get_cached_zone_variable(self._zone_id, name, default)
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
def _source_var(self, name, default=None):
|
2019-07-31 19:25:30 +00:00
|
|
|
current = int(self._zone_var("currentsource", 0))
|
2017-07-31 12:42:55 +00:00
|
|
|
if current:
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._russ.get_cached_source_variable(current, name, default)
|
2017-07-31 12:42:55 +00:00
|
|
|
return default
|
|
|
|
|
|
|
|
def _source_na_var(self, name):
|
|
|
|
"""Will replace invalid values with None."""
|
2019-07-31 19:25:30 +00:00
|
|
|
current = int(self._zone_var("currentsource", 0))
|
2017-07-31 12:42:55 +00:00
|
|
|
if current:
|
2019-07-31 19:25:30 +00:00
|
|
|
value = self._russ.get_cached_source_variable(current, name, None)
|
2017-07-31 12:42:55 +00:00
|
|
|
if value in (None, "", "------"):
|
|
|
|
return None
|
|
|
|
return value
|
2018-07-23 08:16:05 +00:00
|
|
|
return None
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
def _zone_callback_handler(self, zone_id, *args):
|
|
|
|
if zone_id == self._zone_id:
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
|
|
|
def _source_callback_handler(self, source_id, *args):
|
2019-07-31 19:25:30 +00:00
|
|
|
current = int(self._zone_var("currentsource", 0))
|
2017-07-31 12:42:55 +00:00
|
|
|
if source_id == current:
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
2018-10-01 06:58:21 +00:00
|
|
|
async def async_added_to_hass(self):
|
2017-07-31 12:42:55 +00:00
|
|
|
"""Register callback handlers."""
|
|
|
|
self._russ.add_zone_callback(self._zone_callback_handler)
|
|
|
|
self._russ.add_source_callback(self._source_callback_handler)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""No polling needed."""
|
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the zone."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._zone_var("name", self._name)
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the device."""
|
2019-07-31 19:25:30 +00:00
|
|
|
status = self._zone_var("status", "OFF")
|
|
|
|
if status == "ON":
|
2017-07-31 12:42:55 +00:00
|
|
|
return STATE_ON
|
2019-07-31 19:25:30 +00:00
|
|
|
if status == "OFF":
|
2017-07-31 12:42:55 +00:00
|
|
|
return STATE_OFF
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag media player features that are supported."""
|
|
|
|
return SUPPORT_RUSSOUND
|
|
|
|
|
|
|
|
@property
|
|
|
|
def source(self):
|
|
|
|
"""Get the currently selected source."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._source_na_var("name")
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def source_list(self):
|
|
|
|
"""Return a list of available input sources."""
|
|
|
|
return [x[1] for x in self._sources]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_content_type(self):
|
|
|
|
"""Content type of current playing media."""
|
|
|
|
return MEDIA_TYPE_MUSIC
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_title(self):
|
|
|
|
"""Title of current playing media."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._source_na_var("songname")
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_artist(self):
|
|
|
|
"""Artist of current playing media, music track only."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._source_na_var("artistname")
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_album_name(self):
|
|
|
|
"""Album name of current playing media, music track only."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._source_na_var("albumname")
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_image_url(self):
|
|
|
|
"""Image url of current playing media."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._source_na_var("coverarturl")
|
2017-07-31 12:42:55 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def volume_level(self):
|
|
|
|
"""Volume level of the media player (0..1).
|
|
|
|
|
|
|
|
Value is returned based on a range (0..50).
|
|
|
|
Therefore float divide by 50 to get to the required range.
|
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
return float(self._zone_var("volume", 0)) / 50.0
|
2017-07-31 12:42:55 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_turn_off(self):
|
2017-07-31 12:42:55 +00:00
|
|
|
"""Turn off the zone."""
|
2020-01-29 21:59:45 +00:00
|
|
|
await self._russ.send_zone_event(self._zone_id, "ZoneOff")
|
2017-07-31 12:42:55 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_turn_on(self):
|
2017-07-31 12:42:55 +00:00
|
|
|
"""Turn on the zone."""
|
2020-01-29 21:59:45 +00:00
|
|
|
await self._russ.send_zone_event(self._zone_id, "ZoneOn")
|
2017-07-31 12:42:55 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_set_volume_level(self, volume):
|
2017-07-31 12:42:55 +00:00
|
|
|
"""Set the volume level."""
|
|
|
|
rvol = int(volume * 50.0)
|
2020-01-29 21:59:45 +00:00
|
|
|
await self._russ.send_zone_event(self._zone_id, "KeyPress", "Volume", rvol)
|
2017-07-31 12:42:55 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_select_source(self, source):
|
2017-07-31 12:42:55 +00:00
|
|
|
"""Select the source input for this zone."""
|
|
|
|
for source_id, name in self._sources:
|
|
|
|
if name.lower() != source.lower():
|
|
|
|
continue
|
2020-01-29 21:59:45 +00:00
|
|
|
await self._russ.send_zone_event(self._zone_id, "SelectSource", source_id)
|
|
|
|
break
|