242 lines
7.2 KiB
Python
242 lines
7.2 KiB
Python
"""
|
|
Support for Eight smart mattress covers and mattresses.
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
https://home-assistant.io/components/eight_sleep/
|
|
"""
|
|
import asyncio
|
|
import logging
|
|
import os
|
|
from datetime import timedelta
|
|
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.core import callback
|
|
from homeassistant.config import load_yaml_config_file
|
|
from homeassistant.const import (
|
|
CONF_USERNAME, CONF_PASSWORD, CONF_SENSORS, CONF_BINARY_SENSORS,
|
|
ATTR_ENTITY_ID, EVENT_HOMEASSISTANT_STOP)
|
|
from homeassistant.helpers import discovery
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.dispatcher import (
|
|
async_dispatcher_send, async_dispatcher_connect)
|
|
from homeassistant.helpers.entity import Entity
|
|
from homeassistant.helpers.event import async_track_point_in_utc_time
|
|
from homeassistant.util.dt import utcnow
|
|
|
|
REQUIREMENTS = ['pyeight==0.0.4']
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
CONF_PARTNER = 'partner'
|
|
|
|
DATA_EIGHT = 'eight_sleep'
|
|
DEFAULT_PARTNER = False
|
|
DOMAIN = 'eight_sleep'
|
|
|
|
HEAT_ENTITY = 'heat'
|
|
USER_ENTITY = 'user'
|
|
|
|
HEAT_SCAN_INTERVAL = timedelta(seconds=60)
|
|
USER_SCAN_INTERVAL = timedelta(seconds=300)
|
|
|
|
SIGNAL_UPDATE_HEAT = 'eight_heat_update'
|
|
SIGNAL_UPDATE_USER = 'eight_user_update'
|
|
|
|
NAME_MAP = {
|
|
'left_current_sleep': 'Left Sleep Session',
|
|
'left_last_sleep': 'Left Previous Sleep Session',
|
|
'left_bed_state': 'Left Bed State',
|
|
'left_presence': 'Left Bed Presence',
|
|
'left_bed_temp': 'Left Bed Temperature',
|
|
'left_sleep_stage': 'Left Sleep Stage',
|
|
'right_current_sleep': 'Right Sleep Session',
|
|
'right_last_sleep': 'Right Previous Sleep Session',
|
|
'right_bed_state': 'Right Bed State',
|
|
'right_presence': 'Right Bed Presence',
|
|
'right_bed_temp': 'Right Bed Temperature',
|
|
'right_sleep_stage': 'Right Sleep Stage',
|
|
'room_temp': 'Room Temperature',
|
|
}
|
|
|
|
SENSORS = ['current_sleep',
|
|
'last_sleep',
|
|
'bed_state',
|
|
'bed_temp',
|
|
'sleep_stage']
|
|
|
|
SERVICE_HEAT_SET = 'heat_set'
|
|
|
|
ATTR_TARGET_HEAT = 'target'
|
|
ATTR_HEAT_DURATION = 'duration'
|
|
|
|
VALID_TARGET_HEAT = vol.All(vol.Coerce(int), vol.Clamp(min=0, max=100))
|
|
VALID_DURATION = vol.All(vol.Coerce(int), vol.Clamp(min=0, max=28800))
|
|
|
|
SERVICE_EIGHT_SCHEMA = vol.Schema({
|
|
ATTR_ENTITY_ID: cv.entity_ids,
|
|
ATTR_TARGET_HEAT: VALID_TARGET_HEAT,
|
|
ATTR_HEAT_DURATION: VALID_DURATION,
|
|
})
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
DOMAIN: vol.Schema({
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
|
vol.Optional(CONF_PARTNER, default=DEFAULT_PARTNER): cv.boolean,
|
|
}),
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
@asyncio.coroutine
|
|
def async_setup(hass, config):
|
|
"""Set up the Eight Sleep component."""
|
|
from pyeight.eight import EightSleep
|
|
|
|
conf = config.get(DOMAIN)
|
|
user = conf.get(CONF_USERNAME)
|
|
password = conf.get(CONF_PASSWORD)
|
|
partner = conf.get(CONF_PARTNER)
|
|
|
|
if hass.config.time_zone is None:
|
|
_LOGGER.error('Timezone is not set in Home Assistant.')
|
|
return False
|
|
|
|
timezone = hass.config.time_zone
|
|
|
|
eight = EightSleep(user, password, timezone, partner, None, hass.loop)
|
|
|
|
hass.data[DATA_EIGHT] = eight
|
|
|
|
# Authenticate, build sensors
|
|
success = yield from eight.start()
|
|
if not success:
|
|
# Authentication failed, cannot continue
|
|
return False
|
|
|
|
@asyncio.coroutine
|
|
def async_update_heat_data(now):
|
|
"""Update heat data from eight in HEAT_SCAN_INTERVAL."""
|
|
yield from eight.update_device_data()
|
|
async_dispatcher_send(hass, SIGNAL_UPDATE_HEAT)
|
|
|
|
async_track_point_in_utc_time(
|
|
hass, async_update_heat_data, utcnow() + HEAT_SCAN_INTERVAL)
|
|
|
|
@asyncio.coroutine
|
|
def async_update_user_data(now):
|
|
"""Update user data from eight in USER_SCAN_INTERVAL."""
|
|
yield from eight.update_user_data()
|
|
async_dispatcher_send(hass, SIGNAL_UPDATE_USER)
|
|
|
|
async_track_point_in_utc_time(
|
|
hass, async_update_user_data, utcnow() + USER_SCAN_INTERVAL)
|
|
|
|
yield from async_update_heat_data(None)
|
|
yield from async_update_user_data(None)
|
|
|
|
# Load sub components
|
|
sensors = []
|
|
binary_sensors = []
|
|
if eight.users:
|
|
for user in eight.users:
|
|
obj = eight.users[user]
|
|
for sensor in SENSORS:
|
|
sensors.append('{}_{}'.format(obj.side, sensor))
|
|
binary_sensors.append('{}_presence'.format(obj.side))
|
|
sensors.append('room_temp')
|
|
|
|
hass.async_add_job(discovery.async_load_platform(
|
|
hass, 'sensor', DOMAIN, {
|
|
CONF_SENSORS: sensors,
|
|
}, config))
|
|
|
|
hass.async_add_job(discovery.async_load_platform(
|
|
hass, 'binary_sensor', DOMAIN, {
|
|
CONF_BINARY_SENSORS: binary_sensors,
|
|
}, config))
|
|
|
|
descriptions = yield from hass.loop.run_in_executor(
|
|
None, load_yaml_config_file,
|
|
os.path.join(os.path.dirname(__file__), 'services.yaml'))
|
|
|
|
@asyncio.coroutine
|
|
def async_service_handler(service):
|
|
"""Handle eight sleep service calls."""
|
|
params = service.data.copy()
|
|
|
|
sensor = params.pop(ATTR_ENTITY_ID, None)
|
|
target = params.pop(ATTR_TARGET_HEAT, None)
|
|
duration = params.pop(ATTR_HEAT_DURATION, 0)
|
|
|
|
for sens in sensor:
|
|
side = sens.split('_')[1]
|
|
userid = eight.fetch_userid(side)
|
|
usrobj = eight.users[userid]
|
|
yield from usrobj.set_heating_level(target, duration)
|
|
|
|
async_dispatcher_send(hass, SIGNAL_UPDATE_HEAT)
|
|
|
|
# Register services
|
|
hass.services.async_register(
|
|
DOMAIN, SERVICE_HEAT_SET, async_service_handler,
|
|
descriptions[DOMAIN].get(SERVICE_HEAT_SET),
|
|
schema=SERVICE_EIGHT_SCHEMA)
|
|
|
|
@asyncio.coroutine
|
|
def stop_eight(event):
|
|
"""Handle stopping eight api session."""
|
|
yield from eight.stop()
|
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, stop_eight)
|
|
|
|
return True
|
|
|
|
|
|
class EightSleepUserEntity(Entity):
|
|
"""The Eight Sleep device entity."""
|
|
|
|
def __init__(self, eight):
|
|
"""Initialize the data oject."""
|
|
self._eight = eight
|
|
|
|
@asyncio.coroutine
|
|
def async_added_to_hass(self):
|
|
"""Register update dispatcher."""
|
|
@callback
|
|
def async_eight_user_update():
|
|
"""Update callback."""
|
|
self.hass.async_add_job(self.async_update_ha_state(True))
|
|
|
|
async_dispatcher_connect(
|
|
self.hass, SIGNAL_UPDATE_USER, async_eight_user_update)
|
|
|
|
@property
|
|
def should_poll(self):
|
|
"""Return True if entity has to be polled for state."""
|
|
return False
|
|
|
|
|
|
class EightSleepHeatEntity(Entity):
|
|
"""The Eight Sleep device entity."""
|
|
|
|
def __init__(self, eight):
|
|
"""Initialize the data oject."""
|
|
self._eight = eight
|
|
|
|
@asyncio.coroutine
|
|
def async_added_to_hass(self):
|
|
"""Register update dispatcher."""
|
|
@callback
|
|
def async_eight_heat_update():
|
|
"""Update callback."""
|
|
self.hass.async_add_job(self.async_update_ha_state(True))
|
|
|
|
async_dispatcher_connect(
|
|
self.hass, SIGNAL_UPDATE_HEAT, async_eight_heat_update)
|
|
|
|
@property
|
|
def should_poll(self):
|
|
"""Return True if entity has to be polled for state."""
|
|
return False
|