2016-11-23 14:52:14 +00:00
|
|
|
"""
|
|
|
|
Support Hook, available at hooksmarthome.com.
|
|
|
|
|
2016-11-23 23:21:48 +00:00
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/switch.hook/
|
2016-11-23 14:52:14 +00:00
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
import asyncio
|
2016-11-23 23:21:48 +00:00
|
|
|
|
2016-11-23 14:52:14 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
import async_timeout
|
|
|
|
import aiohttp
|
|
|
|
|
2016-11-23 23:21:48 +00:00
|
|
|
from homeassistant.components.switch import (SwitchDevice, PLATFORM_SCHEMA)
|
2017-04-04 08:55:43 +00:00
|
|
|
from homeassistant.const import CONF_PASSWORD, CONF_USERNAME, CONF_TOKEN
|
2016-11-28 00:26:46 +00:00
|
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
2016-11-23 14:52:14 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-11-23 23:21:48 +00:00
|
|
|
HOOK_ENDPOINT = 'https://api.gethook.io/v1/'
|
2016-11-23 14:52:14 +00:00
|
|
|
TIMEOUT = 10
|
|
|
|
|
2016-11-23 23:21:48 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
2017-04-04 08:55:43 +00:00
|
|
|
vol.Exclusive(CONF_PASSWORD, 'hook_secret', msg='hook: provide ' +
|
|
|
|
'username/password OR token'): cv.string,
|
|
|
|
vol.Exclusive(CONF_TOKEN, 'hook_secret', msg='hook: provide ' +
|
|
|
|
'username/password OR token'): cv.string,
|
|
|
|
vol.Inclusive(CONF_USERNAME, 'hook_auth'): cv.string,
|
|
|
|
vol.Inclusive(CONF_PASSWORD, 'hook_auth'): cv.string,
|
2016-11-23 14:52:14 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup_platform(hass, config, async_add_devices, discovery_info=None):
|
2016-11-23 23:21:48 +00:00
|
|
|
"""Set up Hook by getting the access token and list of actions."""
|
2016-11-23 14:52:14 +00:00
|
|
|
username = config.get(CONF_USERNAME)
|
|
|
|
password = config.get(CONF_PASSWORD)
|
2017-04-04 08:55:43 +00:00
|
|
|
token = config.get(CONF_TOKEN)
|
2016-11-28 00:26:46 +00:00
|
|
|
websession = async_get_clientsession(hass)
|
2017-04-04 08:55:43 +00:00
|
|
|
# If password is set in config, prefer it over token
|
|
|
|
if username is not None and password is not None:
|
|
|
|
try:
|
|
|
|
with async_timeout.timeout(TIMEOUT, loop=hass.loop):
|
|
|
|
response = yield from websession.post(
|
|
|
|
'{}{}'.format(HOOK_ENDPOINT, 'user/login'),
|
|
|
|
data={
|
|
|
|
'username': username,
|
|
|
|
'password': password})
|
2017-04-24 06:58:17 +00:00
|
|
|
# The Hook API returns JSON but calls it 'text/html'. Setting
|
|
|
|
# content_type=None disables aiohttp's content-type validation.
|
|
|
|
data = yield from response.json(content_type=None)
|
2017-04-04 08:55:43 +00:00
|
|
|
except (asyncio.TimeoutError, aiohttp.ClientError) as error:
|
|
|
|
_LOGGER.error("Failed authentication API call: %s", error)
|
|
|
|
return False
|
2016-11-23 14:52:14 +00:00
|
|
|
|
2017-04-04 08:55:43 +00:00
|
|
|
try:
|
|
|
|
token = data['data']['token']
|
|
|
|
except KeyError:
|
|
|
|
_LOGGER.error("No token. Check username and password")
|
|
|
|
return False
|
2016-11-23 14:52:14 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
with async_timeout.timeout(TIMEOUT, loop=hass.loop):
|
2016-11-28 00:26:46 +00:00
|
|
|
response = yield from websession.get(
|
2016-11-23 23:21:48 +00:00
|
|
|
'{}{}'.format(HOOK_ENDPOINT, 'device'),
|
2017-04-04 08:55:43 +00:00
|
|
|
params={"token": token})
|
2017-04-24 06:58:17 +00:00
|
|
|
data = yield from response.json(content_type=None)
|
2017-03-30 07:50:53 +00:00
|
|
|
except (asyncio.TimeoutError, aiohttp.ClientError) as error:
|
2016-11-23 14:52:14 +00:00
|
|
|
_LOGGER.error("Failed getting devices: %s", error)
|
|
|
|
return False
|
|
|
|
|
2017-03-01 04:33:19 +00:00
|
|
|
async_add_devices(
|
2016-11-23 14:52:14 +00:00
|
|
|
HookSmartHome(
|
|
|
|
hass,
|
|
|
|
token,
|
|
|
|
d['device_id'],
|
|
|
|
d['device_name'])
|
|
|
|
for lst in data['data']
|
|
|
|
for d in lst)
|
|
|
|
|
|
|
|
|
|
|
|
class HookSmartHome(SwitchDevice):
|
|
|
|
"""Representation of a Hook device, allowing on and off commands."""
|
|
|
|
|
|
|
|
def __init__(self, hass, token, device_id, device_name):
|
|
|
|
"""Initialize the switch."""
|
2016-11-28 00:26:46 +00:00
|
|
|
self.hass = hass
|
2016-11-23 14:52:14 +00:00
|
|
|
self._token = token
|
|
|
|
self._state = False
|
|
|
|
self._id = device_id
|
|
|
|
self._name = device_name
|
|
|
|
_LOGGER.debug(
|
2016-11-23 23:21:48 +00:00
|
|
|
"Creating Hook object: ID: %s Name: %s", self._id, self._name)
|
2016-11-23 14:52:14 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the switch."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if device is on."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _send(self, url):
|
|
|
|
"""Send the url to the Hook API."""
|
|
|
|
try:
|
|
|
|
_LOGGER.debug("Sending: %s", url)
|
2016-11-28 00:26:46 +00:00
|
|
|
websession = async_get_clientsession(self.hass)
|
|
|
|
with async_timeout.timeout(TIMEOUT, loop=self.hass.loop):
|
|
|
|
response = yield from websession.get(
|
2016-11-23 23:21:48 +00:00
|
|
|
url, params={"token": self._token})
|
2017-04-24 06:58:17 +00:00
|
|
|
data = yield from response.json(content_type=None)
|
2016-11-30 21:05:58 +00:00
|
|
|
|
2017-03-30 07:50:53 +00:00
|
|
|
except (asyncio.TimeoutError, aiohttp.ClientError) as error:
|
2016-11-23 14:52:14 +00:00
|
|
|
_LOGGER.error("Failed setting state: %s", error)
|
|
|
|
return False
|
2016-11-30 21:05:58 +00:00
|
|
|
|
2016-11-23 14:52:14 +00:00
|
|
|
_LOGGER.debug("Got: %s", data)
|
|
|
|
return data['return_value'] == '1'
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
2017-07-06 03:02:16 +00:00
|
|
|
def async_turn_on(self, **kwargs):
|
2016-11-23 14:52:14 +00:00
|
|
|
"""Turn the device on asynchronously."""
|
|
|
|
_LOGGER.debug("Turning on: %s", self._name)
|
2016-11-23 23:21:48 +00:00
|
|
|
url = '{}{}{}{}'.format(
|
|
|
|
HOOK_ENDPOINT, 'device/trigger/', self._id, '/On')
|
|
|
|
success = yield from self._send(url)
|
2016-11-23 14:52:14 +00:00
|
|
|
self._state = success
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
2017-07-06 03:02:16 +00:00
|
|
|
def async_turn_off(self, **kwargs):
|
2016-11-23 14:52:14 +00:00
|
|
|
"""Turn the device off asynchronously."""
|
|
|
|
_LOGGER.debug("Turning off: %s", self._name)
|
2016-11-23 23:21:48 +00:00
|
|
|
url = '{}{}{}{}'.format(
|
|
|
|
HOOK_ENDPOINT, 'device/trigger/', self._id, '/Off')
|
|
|
|
success = yield from self._send(url)
|
2016-11-23 14:52:14 +00:00
|
|
|
# If it wasn't successful, keep state as true
|
|
|
|
self._state = not success
|