204 lines
5.8 KiB
Python
204 lines
5.8 KiB
Python
"""Support for the Unitymedia Horizon HD Recorder."""
|
|
from datetime import timedelta
|
|
import logging
|
|
|
|
from horimote import Client, keys
|
|
from horimote.exceptions import AuthenticationError
|
|
import voluptuous as vol
|
|
|
|
from homeassistant import util
|
|
from homeassistant.components.media_player import PLATFORM_SCHEMA, MediaPlayerEntity
|
|
from homeassistant.components.media_player.const import (
|
|
MEDIA_TYPE_CHANNEL,
|
|
SUPPORT_NEXT_TRACK,
|
|
SUPPORT_PAUSE,
|
|
SUPPORT_PLAY,
|
|
SUPPORT_PLAY_MEDIA,
|
|
SUPPORT_PREVIOUS_TRACK,
|
|
SUPPORT_TURN_OFF,
|
|
SUPPORT_TURN_ON,
|
|
)
|
|
from homeassistant.const import (
|
|
CONF_HOST,
|
|
CONF_NAME,
|
|
CONF_PORT,
|
|
STATE_OFF,
|
|
STATE_PAUSED,
|
|
STATE_PLAYING,
|
|
)
|
|
from homeassistant.exceptions import PlatformNotReady
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
DEFAULT_NAME = "Horizon"
|
|
DEFAULT_PORT = 5900
|
|
|
|
MIN_TIME_BETWEEN_FORCED_SCANS = timedelta(seconds=1)
|
|
MIN_TIME_BETWEEN_SCANS = timedelta(seconds=10)
|
|
|
|
SUPPORT_HORIZON = (
|
|
SUPPORT_NEXT_TRACK
|
|
| SUPPORT_PAUSE
|
|
| SUPPORT_PLAY
|
|
| SUPPORT_PLAY_MEDIA
|
|
| SUPPORT_PREVIOUS_TRACK
|
|
| SUPPORT_TURN_ON
|
|
| SUPPORT_TURN_OFF
|
|
)
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
{
|
|
vol.Required(CONF_HOST): cv.string,
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
}
|
|
)
|
|
|
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
|
"""Set up the Horizon platform."""
|
|
|
|
host = config[CONF_HOST]
|
|
name = config[CONF_NAME]
|
|
port = config[CONF_PORT]
|
|
|
|
try:
|
|
client = Client(host, port=port)
|
|
except AuthenticationError as msg:
|
|
_LOGGER.error("Authentication to %s at %s failed: %s", name, host, msg)
|
|
return
|
|
except OSError as msg:
|
|
# occurs if horizon box is offline
|
|
_LOGGER.error("Connection to %s at %s failed: %s", name, host, msg)
|
|
raise PlatformNotReady from msg
|
|
|
|
_LOGGER.info("Connection to %s at %s established", name, host)
|
|
|
|
add_entities([HorizonDevice(client, name, keys)], True)
|
|
|
|
|
|
class HorizonDevice(MediaPlayerEntity):
|
|
"""Representation of a Horizon HD Recorder."""
|
|
|
|
def __init__(self, client, name, remote_keys):
|
|
"""Initialize the remote."""
|
|
self._client = client
|
|
self._name = name
|
|
self._state = None
|
|
self._keys = remote_keys
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of the remote."""
|
|
return self._name
|
|
|
|
@property
|
|
def state(self):
|
|
"""Return the state of the device."""
|
|
return self._state
|
|
|
|
@property
|
|
def supported_features(self):
|
|
"""Flag media player features that are supported."""
|
|
return SUPPORT_HORIZON
|
|
|
|
@util.Throttle(MIN_TIME_BETWEEN_SCANS, MIN_TIME_BETWEEN_FORCED_SCANS)
|
|
def update(self):
|
|
"""Update State using the media server running on the Horizon."""
|
|
try:
|
|
if self._client.is_powered_on():
|
|
self._state = STATE_PLAYING
|
|
else:
|
|
self._state = STATE_OFF
|
|
except OSError:
|
|
self._state = STATE_OFF
|
|
|
|
def turn_on(self):
|
|
"""Turn the device on."""
|
|
if self._state == STATE_OFF:
|
|
self._send_key(self._keys.POWER)
|
|
|
|
def turn_off(self):
|
|
"""Turn the device off."""
|
|
if self._state != STATE_OFF:
|
|
self._send_key(self._keys.POWER)
|
|
|
|
def media_previous_track(self):
|
|
"""Channel down."""
|
|
self._send_key(self._keys.CHAN_DOWN)
|
|
self._state = STATE_PLAYING
|
|
|
|
def media_next_track(self):
|
|
"""Channel up."""
|
|
self._send_key(self._keys.CHAN_UP)
|
|
self._state = STATE_PLAYING
|
|
|
|
def media_play(self):
|
|
"""Send play command."""
|
|
self._send_key(self._keys.PAUSE)
|
|
self._state = STATE_PLAYING
|
|
|
|
def media_pause(self):
|
|
"""Send pause command."""
|
|
self._send_key(self._keys.PAUSE)
|
|
self._state = STATE_PAUSED
|
|
|
|
def media_play_pause(self):
|
|
"""Send play/pause command."""
|
|
self._send_key(self._keys.PAUSE)
|
|
if self._state == STATE_PAUSED:
|
|
self._state = STATE_PLAYING
|
|
else:
|
|
self._state = STATE_PAUSED
|
|
|
|
def play_media(self, media_type, media_id, **kwargs):
|
|
"""Play media / switch to channel."""
|
|
if MEDIA_TYPE_CHANNEL == media_type:
|
|
try:
|
|
self._select_channel(int(media_id))
|
|
self._state = STATE_PLAYING
|
|
except ValueError:
|
|
_LOGGER.error("Invalid channel: %s", media_id)
|
|
else:
|
|
_LOGGER.error(
|
|
"Invalid media type %s. Supported type: %s",
|
|
media_type,
|
|
MEDIA_TYPE_CHANNEL,
|
|
)
|
|
|
|
def _select_channel(self, channel):
|
|
"""Select a channel (taken from einder library, thx)."""
|
|
self._send(channel=channel)
|
|
|
|
def _send_key(self, key):
|
|
"""Send a key to the Horizon device."""
|
|
self._send(key=key)
|
|
|
|
def _send(self, key=None, channel=None):
|
|
"""Send a key to the Horizon device."""
|
|
|
|
try:
|
|
if key:
|
|
self._client.send_key(key)
|
|
elif channel:
|
|
self._client.select_channel(channel)
|
|
except OSError as msg:
|
|
_LOGGER.error("%s disconnected: %s. Trying to reconnect", self._name, msg)
|
|
|
|
# for reconnect, first gracefully disconnect
|
|
self._client.disconnect()
|
|
|
|
try:
|
|
self._client.connect()
|
|
self._client.authorize()
|
|
except AuthenticationError as msg:
|
|
_LOGGER.error("Authentication to %s failed: %s", self._name, msg)
|
|
return
|
|
except OSError as msg:
|
|
# occurs if horizon box is offline
|
|
_LOGGER.error("Reconnect to %s failed: %s", self._name, msg)
|
|
return
|
|
|
|
self._send(key=key, channel=channel)
|