2019-02-14 15:01:46 +00:00
|
|
|
"""Support for Harmony Hub devices."""
|
2018-12-19 13:21:40 +00:00
|
|
|
import json
|
2018-01-21 06:35:38 +00:00
|
|
|
import logging
|
2017-05-19 14:39:13 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2018-07-18 09:54:27 +00:00
|
|
|
from homeassistant.components import remote
|
2017-05-19 14:39:13 +00:00
|
|
|
from homeassistant.components.remote import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_ACTIVITY,
|
|
|
|
ATTR_DELAY_SECS,
|
|
|
|
ATTR_DEVICE,
|
|
|
|
ATTR_HOLD_SECS,
|
|
|
|
ATTR_NUM_REPEATS,
|
|
|
|
DEFAULT_DELAY_SECS,
|
2021-03-03 03:57:36 +00:00
|
|
|
SUPPORT_ACTIVITY,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2021-01-14 08:45:32 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2021-04-20 13:09:46 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2020-04-19 20:33:51 +00:00
|
|
|
from homeassistant.helpers import entity_platform
|
2019-10-19 03:58:07 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2020-03-20 01:43:44 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2020-07-07 14:02:22 +00:00
|
|
|
from homeassistant.helpers.restore_state import RestoreEntity
|
2020-03-20 01:43:44 +00:00
|
|
|
|
|
|
|
from .const import (
|
|
|
|
ACTIVITY_POWER_OFF,
|
2020-07-31 19:55:38 +00:00
|
|
|
ATTR_ACTIVITY_STARTING,
|
2020-07-07 14:02:22 +00:00
|
|
|
ATTR_DEVICES_LIST,
|
|
|
|
ATTR_LAST_ACTIVITY,
|
2020-03-20 01:43:44 +00:00
|
|
|
DOMAIN,
|
2021-04-18 07:44:29 +00:00
|
|
|
HARMONY_DATA,
|
2020-03-20 01:43:44 +00:00
|
|
|
HARMONY_OPTIONS_UPDATE,
|
2020-07-07 14:02:22 +00:00
|
|
|
PREVIOUS_ACTIVE_ACTIVITY,
|
2020-03-20 01:43:44 +00:00
|
|
|
SERVICE_CHANGE_CHANNEL,
|
|
|
|
SERVICE_SYNC,
|
2020-04-02 16:46:10 +00:00
|
|
|
)
|
2021-08-28 19:10:19 +00:00
|
|
|
from .entity import HarmonyEntity
|
2021-01-04 23:21:14 +00:00
|
|
|
from .subscriber import HarmonyCallback
|
2019-11-27 17:14:46 +00:00
|
|
|
|
2017-05-19 14:39:13 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2020-04-19 20:33:51 +00:00
|
|
|
# We want to fire remote commands right away
|
|
|
|
PARALLEL_UPDATES = 0
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_CHANNEL = "channel"
|
2018-12-30 01:22:27 +00:00
|
|
|
|
2021-05-03 16:41:16 +00:00
|
|
|
HARMONY_CHANGE_CHANNEL_SCHEMA = {
|
|
|
|
vol.Required(ATTR_CHANNEL): cv.positive_int,
|
|
|
|
}
|
2019-01-12 18:02:00 +00:00
|
|
|
|
2017-05-19 14:39:13 +00:00
|
|
|
|
2020-03-19 16:29:51 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities
|
|
|
|
):
|
|
|
|
"""Set up the Harmony config entry."""
|
|
|
|
|
2021-04-18 07:44:29 +00:00
|
|
|
data = hass.data[DOMAIN][entry.entry_id][HARMONY_DATA]
|
2020-03-19 16:29:51 +00:00
|
|
|
|
2021-01-04 23:21:14 +00:00
|
|
|
_LOGGER.debug("HarmonyData : %s", data)
|
2020-03-19 16:29:51 +00:00
|
|
|
|
2021-01-04 23:21:14 +00:00
|
|
|
default_activity = entry.options.get(ATTR_ACTIVITY)
|
|
|
|
delay_secs = entry.options.get(ATTR_DELAY_SECS, DEFAULT_DELAY_SECS)
|
|
|
|
|
|
|
|
harmony_conf_file = hass.config.path(f"harmony_{entry.unique_id}.conf")
|
|
|
|
device = HarmonyRemote(data, default_activity, delay_secs, harmony_conf_file)
|
2020-03-19 16:29:51 +00:00
|
|
|
async_add_entities([device])
|
2020-03-20 01:43:44 +00:00
|
|
|
|
2021-05-03 16:34:28 +00:00
|
|
|
platform = entity_platform.async_get_current_platform()
|
2020-03-20 01:43:44 +00:00
|
|
|
|
2020-04-19 20:33:51 +00:00
|
|
|
platform.async_register_entity_service(
|
2020-08-27 11:56:20 +00:00
|
|
|
SERVICE_SYNC,
|
2021-05-03 16:41:16 +00:00
|
|
|
{},
|
2020-08-27 11:56:20 +00:00
|
|
|
"sync",
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-04-19 20:33:51 +00:00
|
|
|
platform.async_register_entity_service(
|
|
|
|
SERVICE_CHANGE_CHANNEL, HARMONY_CHANGE_CHANNEL_SCHEMA, "change_channel"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-01-12 18:02:00 +00:00
|
|
|
|
2017-05-19 14:39:13 +00:00
|
|
|
|
2021-08-28 19:10:19 +00:00
|
|
|
class HarmonyRemote(HarmonyEntity, remote.RemoteEntity, RestoreEntity):
|
2017-05-19 14:39:13 +00:00
|
|
|
"""Remote representation used to control a Harmony device."""
|
|
|
|
|
2021-01-04 23:21:14 +00:00
|
|
|
def __init__(self, data, activity, delay_secs, out_path):
|
2017-05-19 14:39:13 +00:00
|
|
|
"""Initialize HarmonyRemote class."""
|
2021-08-28 19:10:19 +00:00
|
|
|
super().__init__(data=data)
|
2017-05-19 14:39:13 +00:00
|
|
|
self._state = None
|
2020-07-27 16:31:30 +00:00
|
|
|
self._current_activity = ACTIVITY_POWER_OFF
|
2020-03-20 01:43:44 +00:00
|
|
|
self.default_activity = activity
|
2020-07-31 19:55:38 +00:00
|
|
|
self._activity_starting = None
|
2020-08-03 10:55:15 +00:00
|
|
|
self._is_initial_update = True
|
2020-03-20 01:43:44 +00:00
|
|
|
self.delay_secs = delay_secs
|
2020-07-07 14:02:22 +00:00
|
|
|
self._last_activity = None
|
2021-01-04 23:21:14 +00:00
|
|
|
self._config_path = out_path
|
2021-08-28 19:10:19 +00:00
|
|
|
self._attr_unique_id = data.unique_id
|
|
|
|
self._attr_device_info = self._data.device_info(DOMAIN)
|
|
|
|
self._attr_name = data.name
|
|
|
|
self._attr_supported_features = SUPPORT_ACTIVITY
|
2021-03-03 03:57:36 +00:00
|
|
|
|
2020-03-20 01:43:44 +00:00
|
|
|
async def _async_update_options(self, data):
|
|
|
|
"""Change options when the options flow does."""
|
|
|
|
if ATTR_DELAY_SECS in data:
|
|
|
|
self.delay_secs = data[ATTR_DELAY_SECS]
|
|
|
|
|
|
|
|
if ATTR_ACTIVITY in data:
|
|
|
|
self.default_activity = data[ATTR_ACTIVITY]
|
2020-03-19 16:29:51 +00:00
|
|
|
|
2021-01-04 23:21:14 +00:00
|
|
|
def _setup_callbacks(self):
|
2020-06-20 02:50:42 +00:00
|
|
|
callbacks = {
|
2021-04-20 13:09:46 +00:00
|
|
|
"connected": self.async_got_connected,
|
|
|
|
"disconnected": self.async_got_disconnected,
|
|
|
|
"config_updated": self.async_new_config,
|
|
|
|
"activity_starting": self.async_new_activity,
|
|
|
|
"activity_started": self.async_new_activity_finished,
|
2020-06-20 02:50:42 +00:00
|
|
|
}
|
2021-01-04 23:21:14 +00:00
|
|
|
|
|
|
|
self.async_on_remove(self._data.async_subscribe(HarmonyCallback(**callbacks)))
|
2020-06-20 02:50:42 +00:00
|
|
|
|
2021-04-20 13:09:46 +00:00
|
|
|
@callback
|
|
|
|
def async_new_activity_finished(self, activity_info: tuple) -> None:
|
2020-07-31 19:55:38 +00:00
|
|
|
"""Call for finished updated current activity."""
|
|
|
|
self._activity_starting = None
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2018-10-01 06:56:50 +00:00
|
|
|
async def async_added_to_hass(self):
|
2017-11-09 16:57:41 +00:00
|
|
|
"""Complete the initialization."""
|
2020-07-07 14:02:22 +00:00
|
|
|
await super().async_added_to_hass()
|
|
|
|
|
2021-08-28 19:10:19 +00:00
|
|
|
_LOGGER.debug("%s: Harmony Hub added", self.name)
|
2021-01-04 23:21:14 +00:00
|
|
|
|
|
|
|
self.async_on_remove(self._clear_disconnection_delay)
|
|
|
|
self._setup_callbacks()
|
2019-01-16 07:35:29 +00:00
|
|
|
|
2020-04-18 18:36:15 +00:00
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass,
|
|
|
|
f"{HARMONY_OPTIONS_UPDATE}-{self.unique_id}",
|
|
|
|
self._async_update_options,
|
|
|
|
)
|
2020-03-20 01:43:44 +00:00
|
|
|
)
|
|
|
|
|
2019-01-16 07:35:29 +00:00
|
|
|
# Store Harmony HUB config, this will also update our current
|
|
|
|
# activity
|
2021-04-20 13:09:46 +00:00
|
|
|
await self.async_new_config()
|
2019-01-16 07:35:29 +00:00
|
|
|
|
2020-07-07 14:02:22 +00:00
|
|
|
# Restore the last activity so we know
|
|
|
|
# how what to turn on if nothing
|
|
|
|
# is specified
|
|
|
|
last_state = await self.async_get_last_state()
|
|
|
|
if not last_state:
|
|
|
|
return
|
|
|
|
if ATTR_LAST_ACTIVITY not in last_state.attributes:
|
|
|
|
return
|
|
|
|
if self.is_on:
|
|
|
|
return
|
|
|
|
|
|
|
|
self._last_activity = last_state.attributes[ATTR_LAST_ACTIVITY]
|
|
|
|
|
2021-03-03 03:57:36 +00:00
|
|
|
@property
|
|
|
|
def current_activity(self):
|
|
|
|
"""Return the current activity."""
|
|
|
|
return self._current_activity
|
|
|
|
|
|
|
|
@property
|
|
|
|
def activity_list(self):
|
|
|
|
"""Return the available activities."""
|
|
|
|
return self._data.activity_names
|
|
|
|
|
2017-05-19 14:39:13 +00:00
|
|
|
@property
|
2021-03-11 15:57:47 +00:00
|
|
|
def extra_state_attributes(self):
|
2017-05-19 14:39:13 +00:00
|
|
|
"""Add platform specific attributes."""
|
2020-07-07 14:02:22 +00:00
|
|
|
return {
|
2020-07-31 19:55:38 +00:00
|
|
|
ATTR_ACTIVITY_STARTING: self._activity_starting,
|
2021-01-04 23:21:14 +00:00
|
|
|
ATTR_DEVICES_LIST: self._data.device_names,
|
2020-07-07 14:02:22 +00:00
|
|
|
ATTR_LAST_ACTIVITY: self._last_activity,
|
|
|
|
}
|
2017-05-19 14:39:13 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return False if PowerOff is the current activity, otherwise True."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self._current_activity not in [None, "PowerOff"]
|
2017-05-19 14:39:13 +00:00
|
|
|
|
2021-04-20 13:09:46 +00:00
|
|
|
@callback
|
|
|
|
def async_new_activity(self, activity_info: tuple) -> None:
|
2018-12-30 01:22:27 +00:00
|
|
|
"""Call for updating the current activity."""
|
|
|
|
activity_id, activity_name = activity_info
|
2021-08-28 19:10:19 +00:00
|
|
|
_LOGGER.debug("%s: activity reported as: %s", self.name, activity_name)
|
2018-12-19 13:21:40 +00:00
|
|
|
self._current_activity = activity_name
|
2020-07-31 19:55:38 +00:00
|
|
|
if self._is_initial_update:
|
|
|
|
self._is_initial_update = False
|
|
|
|
else:
|
|
|
|
self._activity_starting = activity_name
|
2020-07-07 14:02:22 +00:00
|
|
|
if activity_id != -1:
|
|
|
|
# Save the activity so we can restore
|
|
|
|
# to that activity if none is specified
|
|
|
|
# when turning on
|
|
|
|
self._last_activity = activity_name
|
2018-12-30 01:22:27 +00:00
|
|
|
self._state = bool(activity_id != -1)
|
2020-04-01 21:19:51 +00:00
|
|
|
self.async_write_ha_state()
|
2018-12-19 13:21:40 +00:00
|
|
|
|
2021-04-20 13:09:46 +00:00
|
|
|
async def async_new_config(self, _=None):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Call for updating the current activity."""
|
2021-08-28 19:10:19 +00:00
|
|
|
_LOGGER.debug("%s: configuration has been updated", self.name)
|
2021-04-20 13:09:46 +00:00
|
|
|
self.async_new_activity(self._data.current_activity)
|
2018-12-30 01:22:27 +00:00
|
|
|
await self.hass.async_add_executor_job(self.write_config_file)
|
|
|
|
|
2018-12-19 13:21:40 +00:00
|
|
|
async def async_turn_on(self, **kwargs):
|
2017-05-19 14:39:13 +00:00
|
|
|
"""Start an activity from the Harmony device."""
|
2018-12-30 01:22:27 +00:00
|
|
|
_LOGGER.debug("%s: Turn On", self.name)
|
|
|
|
|
2020-03-20 01:43:44 +00:00
|
|
|
activity = kwargs.get(ATTR_ACTIVITY, self.default_activity)
|
2017-05-19 14:39:13 +00:00
|
|
|
|
2020-07-07 14:02:22 +00:00
|
|
|
if not activity or activity == PREVIOUS_ACTIVE_ACTIVITY:
|
|
|
|
if self._last_activity:
|
|
|
|
activity = self._last_activity
|
|
|
|
else:
|
2021-01-04 23:21:14 +00:00
|
|
|
all_activities = self._data.activity_names
|
2020-07-07 14:02:22 +00:00
|
|
|
if all_activities:
|
|
|
|
activity = all_activities[0]
|
|
|
|
|
2017-05-19 14:39:13 +00:00
|
|
|
if activity:
|
2021-01-04 23:21:14 +00:00
|
|
|
await self._data.async_start_activity(activity)
|
2017-05-19 14:39:13 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error("%s: No activity specified with turn_on service", self.name)
|
2017-05-19 14:39:13 +00:00
|
|
|
|
2018-12-19 13:21:40 +00:00
|
|
|
async def async_turn_off(self, **kwargs):
|
2017-05-19 14:39:13 +00:00
|
|
|
"""Start the PowerOff activity."""
|
2021-01-04 23:21:14 +00:00
|
|
|
await self._data.async_power_off()
|
2017-05-19 14:39:13 +00:00
|
|
|
|
2018-12-19 13:21:40 +00:00
|
|
|
async def async_send_command(self, command, **kwargs):
|
2017-11-09 16:57:41 +00:00
|
|
|
"""Send a list of commands to one device."""
|
2018-12-30 01:22:27 +00:00
|
|
|
_LOGGER.debug("%s: Send Command", self.name)
|
2017-11-09 16:57:41 +00:00
|
|
|
device = kwargs.get(ATTR_DEVICE)
|
2017-08-01 03:52:39 +00:00
|
|
|
if device is None:
|
2018-12-30 01:22:27 +00:00
|
|
|
_LOGGER.error("%s: Missing required argument: device", self.name)
|
2017-08-01 03:52:39 +00:00
|
|
|
return
|
2017-11-09 16:57:41 +00:00
|
|
|
|
2019-03-02 23:54:03 +00:00
|
|
|
num_repeats = kwargs[ATTR_NUM_REPEATS]
|
2020-03-20 01:43:44 +00:00
|
|
|
delay_secs = kwargs.get(ATTR_DELAY_SECS, self.delay_secs)
|
2019-03-02 23:54:03 +00:00
|
|
|
hold_secs = kwargs[ATTR_HOLD_SECS]
|
2021-01-04 23:21:14 +00:00
|
|
|
await self._data.async_send_command(
|
|
|
|
command, device, num_repeats, delay_secs, hold_secs
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2017-11-09 16:57:41 +00:00
|
|
|
|
2019-01-12 18:02:00 +00:00
|
|
|
async def change_channel(self, channel):
|
|
|
|
"""Change the channel using Harmony remote."""
|
2021-01-04 23:21:14 +00:00
|
|
|
await self._data.change_channel(channel)
|
2019-01-12 18:02:00 +00:00
|
|
|
|
2018-12-19 13:21:40 +00:00
|
|
|
async def sync(self):
|
2017-05-19 14:39:13 +00:00
|
|
|
"""Sync the Harmony device with the web service."""
|
2021-01-04 23:21:14 +00:00
|
|
|
if await self._data.sync():
|
2018-12-30 01:22:27 +00:00
|
|
|
await self.hass.async_add_executor_job(self.write_config_file)
|
2018-12-19 13:21:40 +00:00
|
|
|
|
|
|
|
def write_config_file(self):
|
2021-01-04 23:21:14 +00:00
|
|
|
"""Write Harmony configuration file.
|
|
|
|
|
|
|
|
This is a handy way for users to figure out the available commands for automations.
|
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug(
|
2020-02-13 16:27:00 +00:00
|
|
|
"%s: Writing hub configuration to file: %s", self.name, self._config_path
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2021-01-04 23:21:14 +00:00
|
|
|
json_config = self._data.json_config
|
|
|
|
if json_config is None:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.warning("%s: No configuration received from hub", self.name)
|
2018-12-30 01:22:27 +00:00
|
|
|
return
|
|
|
|
|
2018-12-19 13:21:40 +00:00
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
with open(self._config_path, "w+", encoding="utf-8") as file_out:
|
2021-01-04 23:21:14 +00:00
|
|
|
json.dump(json_config, file_out, sort_keys=True, indent=4)
|
2020-04-04 20:09:11 +00:00
|
|
|
except OSError as exc:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"%s: Unable to write HUB configuration to %s: %s",
|
|
|
|
self.name,
|
|
|
|
self._config_path,
|
|
|
|
exc,
|
|
|
|
)
|