2019-05-10 16:34:28 +00:00
|
|
|
"""Support for a Genius Hub system."""
|
|
|
|
from datetime import timedelta
|
2019-04-16 21:54:46 +00:00
|
|
|
import logging
|
2019-10-02 16:27:13 +00:00
|
|
|
from typing import Any, Dict, Optional
|
2019-04-16 21:54:46 +00:00
|
|
|
|
2019-08-04 22:06:36 +00:00
|
|
|
import aiohttp
|
2019-04-16 21:54:46 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-08-20 17:43:39 +00:00
|
|
|
from geniushubclient import GeniusHub
|
2019-05-10 16:34:28 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_TEMPERATURE,
|
|
|
|
CONF_HOST,
|
|
|
|
CONF_MAC,
|
|
|
|
CONF_PASSWORD,
|
|
|
|
CONF_TOKEN,
|
|
|
|
CONF_USERNAME,
|
|
|
|
TEMP_CELSIUS,
|
|
|
|
)
|
2019-08-20 17:43:39 +00:00
|
|
|
from homeassistant.core import callback
|
2019-04-16 21:54:46 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
|
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
|
|
|
from homeassistant.helpers.discovery import async_load_platform
|
2019-08-20 17:43:39 +00:00
|
|
|
from homeassistant.helpers.dispatcher import (
|
|
|
|
async_dispatcher_send,
|
|
|
|
async_dispatcher_connect,
|
|
|
|
)
|
|
|
|
from homeassistant.helpers.entity import Entity
|
2019-05-10 16:34:28 +00:00
|
|
|
from homeassistant.helpers.event import async_track_time_interval
|
2019-10-02 16:27:13 +00:00
|
|
|
from homeassistant.helpers.typing import ConfigType, HomeAssistantType
|
|
|
|
import homeassistant.util.dt as dt_util
|
|
|
|
|
|
|
|
ATTR_DURATION = "duration"
|
2019-04-16 21:54:46 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "geniushub"
|
2019-04-16 21:54:46 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
# temperature is repeated here, as it gives access to high-precision temps
|
|
|
|
GH_ZONE_ATTRS = ["mode", "temperature", "type", "occupied", "override"]
|
|
|
|
GH_DEVICE_ATTRS = {
|
|
|
|
"luminance": "luminance",
|
|
|
|
"measuredTemperature": "measured_temperature",
|
|
|
|
"occupancyTrigger": "occupancy_trigger",
|
|
|
|
"setback": "setback",
|
|
|
|
"setTemperature": "set_temperature",
|
|
|
|
"wakeupInterval": "wakeup_interval",
|
|
|
|
}
|
|
|
|
|
2019-05-10 16:34:28 +00:00
|
|
|
SCAN_INTERVAL = timedelta(seconds=60)
|
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
MAC_ADDRESS_REGEXP = r"^([0-9A-F]{2}:){5}([0-9A-F]{2})$"
|
|
|
|
|
|
|
|
V1_API_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_TOKEN): cv.string,
|
|
|
|
vol.Required(CONF_MAC): vol.Match(MAC_ADDRESS_REGEXP),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
V3_API_SCHEMA = vol.Schema(
|
2019-07-31 19:25:30 +00:00
|
|
|
{
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
2019-10-02 16:27:13 +00:00
|
|
|
vol.Optional(CONF_MAC): vol.Match(MAC_ADDRESS_REGEXP),
|
2019-07-31 19:25:30 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
2019-10-02 16:27:13 +00:00
|
|
|
{DOMAIN: vol.Any(V3_API_SCHEMA, V1_API_SCHEMA)}, extra=vol.ALLOW_EXTRA
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-04-16 21:54:46 +00:00
|
|
|
|
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
async def async_setup(hass: HomeAssistantType, config: ConfigType) -> bool:
|
2019-04-16 21:54:46 +00:00
|
|
|
"""Create a Genius Hub system."""
|
2019-10-02 16:27:13 +00:00
|
|
|
hass.data[DOMAIN] = {}
|
|
|
|
|
|
|
|
kwargs = dict(config[DOMAIN])
|
2019-04-18 12:37:52 +00:00
|
|
|
if CONF_HOST in kwargs:
|
2019-07-31 19:25:30 +00:00
|
|
|
args = (kwargs.pop(CONF_HOST),)
|
2019-04-18 12:37:52 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
args = (kwargs.pop(CONF_TOKEN),)
|
2019-10-02 16:27:13 +00:00
|
|
|
hub_uid = kwargs.pop(CONF_MAC, None)
|
2019-04-18 12:37:52 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
client = GeniusHub(*args, **kwargs, session=async_get_clientsession(hass))
|
|
|
|
|
|
|
|
broker = hass.data[DOMAIN]["broker"] = GeniusBroker(hass, client, hub_uid)
|
2019-08-04 22:06:36 +00:00
|
|
|
|
2019-04-16 21:54:46 +00:00
|
|
|
try:
|
2019-10-02 16:27:13 +00:00
|
|
|
await client.update()
|
2019-08-04 22:06:36 +00:00
|
|
|
except aiohttp.ClientResponseError as err:
|
|
|
|
_LOGGER.error("Setup failed, check your configuration, %s", err)
|
2019-04-16 21:54:46 +00:00
|
|
|
return False
|
2019-08-04 22:06:36 +00:00
|
|
|
broker.make_debug_log_entries()
|
2019-04-16 21:54:46 +00:00
|
|
|
|
2019-08-04 22:06:36 +00:00
|
|
|
async_track_time_interval(hass, broker.async_update, SCAN_INTERVAL)
|
2019-04-16 21:54:46 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
for platform in ["climate", "water_heater", "sensor", "binary_sensor"]:
|
|
|
|
hass.async_create_task(async_load_platform(hass, platform, DOMAIN, {}, config))
|
2019-05-14 21:30:26 +00:00
|
|
|
|
2019-04-16 21:54:46 +00:00
|
|
|
return True
|
2019-05-10 16:34:28 +00:00
|
|
|
|
|
|
|
|
2019-08-04 22:06:36 +00:00
|
|
|
class GeniusBroker:
|
2019-05-10 16:34:28 +00:00
|
|
|
"""Container for geniushub client and data."""
|
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
def __init__(self, hass, client, hub_uid) -> None:
|
2019-05-10 16:34:28 +00:00
|
|
|
"""Initialize the geniushub client."""
|
2019-08-20 17:43:39 +00:00
|
|
|
self.hass = hass
|
2019-10-02 16:27:13 +00:00
|
|
|
self.client = client
|
|
|
|
self._hub_uid = hub_uid
|
|
|
|
|
|
|
|
@property
|
|
|
|
def hub_uid(self) -> int:
|
|
|
|
"""Return the Hub UID (MAC address)."""
|
|
|
|
# pylint: disable=no-member
|
|
|
|
return self._hub_uid if self._hub_uid is not None else self.client.uid
|
2019-05-10 16:34:28 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
async def async_update(self, now, **kwargs) -> None:
|
2019-05-10 16:34:28 +00:00
|
|
|
"""Update the geniushub client's data."""
|
|
|
|
try:
|
2019-08-20 17:43:39 +00:00
|
|
|
await self.client.update()
|
2019-08-04 22:06:36 +00:00
|
|
|
except aiohttp.ClientResponseError as err:
|
2019-10-02 16:27:13 +00:00
|
|
|
_LOGGER.warning("Update failed, message is: %s", err)
|
2019-05-10 16:34:28 +00:00
|
|
|
return
|
2019-08-04 22:06:36 +00:00
|
|
|
self.make_debug_log_entries()
|
2019-07-21 17:07:03 +00:00
|
|
|
|
2019-08-20 17:43:39 +00:00
|
|
|
async_dispatcher_send(self.hass, DOMAIN)
|
2019-08-04 22:06:36 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
def make_debug_log_entries(self) -> None:
|
2019-08-04 22:06:36 +00:00
|
|
|
"""Make any useful debug log entries."""
|
|
|
|
# pylint: disable=protected-access
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug(
|
2019-08-20 17:43:39 +00:00
|
|
|
"Raw JSON: \n\nclient._zones = %s \n\nclient._devices = %s",
|
|
|
|
self.client._zones,
|
|
|
|
self.client._devices,
|
2019-07-31 19:46:17 +00:00
|
|
|
)
|
2019-08-20 17:43:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
class GeniusEntity(Entity):
|
2019-10-02 16:27:13 +00:00
|
|
|
"""Base for all Genius Hub entities."""
|
2019-08-20 17:43:39 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
def __init__(self) -> None:
|
2019-08-20 17:43:39 +00:00
|
|
|
"""Initialize the entity."""
|
2019-10-02 16:27:13 +00:00
|
|
|
self._unique_id = self._name = None
|
2019-08-20 17:43:39 +00:00
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
async def async_added_to_hass(self) -> None:
|
2019-08-20 17:43:39 +00:00
|
|
|
"""Set up a listener when this entity is added to HA."""
|
|
|
|
async_dispatcher_connect(self.hass, DOMAIN, self._refresh)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _refresh(self) -> None:
|
|
|
|
self.async_schedule_update_ha_state(force_refresh=True)
|
|
|
|
|
2019-10-02 16:27:13 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self) -> Optional[str]:
|
|
|
|
"""Return a unique ID."""
|
|
|
|
return self._unique_id
|
|
|
|
|
2019-08-20 17:43:39 +00:00
|
|
|
@property
|
|
|
|
def name(self) -> str:
|
|
|
|
"""Return the name of the geniushub entity."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self) -> bool:
|
|
|
|
"""Return False as geniushub entities should not be polled."""
|
|
|
|
return False
|
2019-10-02 16:27:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
class GeniusDevice(GeniusEntity):
|
|
|
|
"""Base for all Genius Hub devices."""
|
|
|
|
|
|
|
|
def __init__(self, broker, device) -> None:
|
|
|
|
"""Initialize the Device."""
|
|
|
|
super().__init__()
|
|
|
|
|
|
|
|
self._device = device
|
|
|
|
self._unique_id = f"{broker.hub_uid}_device_{device.id}"
|
|
|
|
|
|
|
|
self._last_comms = self._state_attr = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self) -> Dict[str, Any]:
|
|
|
|
"""Return the device state attributes."""
|
|
|
|
|
|
|
|
attrs = {}
|
|
|
|
attrs["assigned_zone"] = self._device.data["assignedZones"][0]["name"]
|
|
|
|
if self._last_comms:
|
|
|
|
attrs["last_comms"] = self._last_comms.isoformat()
|
|
|
|
|
|
|
|
state = dict(self._device.data["state"])
|
|
|
|
if "_state" in self._device.data: # only for v3 API
|
|
|
|
state.update(self._device.data["_state"])
|
|
|
|
|
|
|
|
attrs["state"] = {
|
|
|
|
GH_DEVICE_ATTRS[k]: v for k, v in state.items() if k in GH_DEVICE_ATTRS
|
|
|
|
}
|
|
|
|
|
|
|
|
return attrs
|
|
|
|
|
|
|
|
async def async_update(self) -> None:
|
|
|
|
"""Update an entity's state data."""
|
|
|
|
if "_state" in self._device.data: # only for v3 API
|
|
|
|
self._last_comms = dt_util.utc_from_timestamp(
|
|
|
|
self._device.data["_state"]["lastComms"]
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class GeniusZone(GeniusEntity):
|
|
|
|
"""Base for all Genius Hub zones."""
|
|
|
|
|
|
|
|
def __init__(self, broker, zone) -> None:
|
|
|
|
"""Initialize the Zone."""
|
|
|
|
super().__init__()
|
|
|
|
|
|
|
|
self._zone = zone
|
|
|
|
self._unique_id = f"{broker.hub_uid}_device_{zone.id}"
|
|
|
|
|
|
|
|
self._max_temp = self._min_temp = self._supported_features = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self) -> str:
|
|
|
|
"""Return the name of the climate device."""
|
|
|
|
return self._zone.name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self) -> Dict[str, Any]:
|
|
|
|
"""Return the device state attributes."""
|
|
|
|
status = {k: v for k, v in self._zone.data.items() if k in GH_ZONE_ATTRS}
|
|
|
|
return {"status": status}
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_temperature(self) -> Optional[float]:
|
|
|
|
"""Return the current temperature."""
|
|
|
|
return self._zone.data.get("temperature")
|
|
|
|
|
|
|
|
@property
|
|
|
|
def target_temperature(self) -> float:
|
|
|
|
"""Return the temperature we try to reach."""
|
|
|
|
return self._zone.data["setpoint"]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def min_temp(self) -> float:
|
|
|
|
"""Return max valid temperature that can be set."""
|
|
|
|
return self._min_temp
|
|
|
|
|
|
|
|
@property
|
|
|
|
def max_temp(self) -> float:
|
|
|
|
"""Return max valid temperature that can be set."""
|
|
|
|
return self._max_temp
|
|
|
|
|
|
|
|
@property
|
|
|
|
def temperature_unit(self) -> str:
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return TEMP_CELSIUS
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self) -> int:
|
|
|
|
"""Return the bitmask of supported features."""
|
|
|
|
return self._supported_features
|
|
|
|
|
|
|
|
async def async_set_temperature(self, **kwargs) -> None:
|
|
|
|
"""Set a new target temperature for this zone."""
|
|
|
|
await self._zone.set_override(
|
|
|
|
kwargs[ATTR_TEMPERATURE], kwargs.get(ATTR_DURATION, 3600)
|
|
|
|
)
|