2017-09-16 19:35:28 +00:00
|
|
|
"""Support for alexa Smart Home Skill API."""
|
|
|
|
import asyncio
|
|
|
|
import logging
|
2017-11-01 03:28:17 +00:00
|
|
|
import math
|
2018-01-23 08:01:18 +00:00
|
|
|
from datetime import datetime
|
2017-09-16 19:35:28 +00:00
|
|
|
from uuid import uuid4
|
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
from homeassistant.components import (
|
|
|
|
alert, automation, cover, fan, group, input_boolean, light, lock,
|
2018-01-23 18:45:28 +00:00
|
|
|
media_player, scene, script, switch, http)
|
|
|
|
import homeassistant.core as ha
|
|
|
|
import homeassistant.util.color as color_util
|
|
|
|
from homeassistant.util.decorator import Registry
|
2017-09-16 19:35:28 +00:00
|
|
|
from homeassistant.const import (
|
2018-01-21 06:35:38 +00:00
|
|
|
ATTR_ENTITY_ID, ATTR_SUPPORTED_FEATURES, CONF_NAME, SERVICE_LOCK,
|
2017-11-17 17:14:22 +00:00
|
|
|
SERVICE_MEDIA_NEXT_TRACK, SERVICE_MEDIA_PAUSE, SERVICE_MEDIA_PLAY,
|
|
|
|
SERVICE_MEDIA_PREVIOUS_TRACK, SERVICE_MEDIA_STOP,
|
|
|
|
SERVICE_SET_COVER_POSITION, SERVICE_TURN_OFF, SERVICE_TURN_ON,
|
|
|
|
SERVICE_UNLOCK, SERVICE_VOLUME_SET)
|
2018-01-23 18:45:28 +00:00
|
|
|
from .const import CONF_FILTER, CONF_ENTITY_CONFIG
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
API_DIRECTIVE = 'directive'
|
2017-11-17 17:14:22 +00:00
|
|
|
API_ENDPOINT = 'endpoint'
|
2017-10-07 20:31:57 +00:00
|
|
|
API_EVENT = 'event'
|
|
|
|
API_HEADER = 'header'
|
|
|
|
API_PAYLOAD = 'payload'
|
2017-11-17 17:14:22 +00:00
|
|
|
|
2018-01-23 18:45:28 +00:00
|
|
|
SMART_HOME_HTTP_ENDPOINT = '/api/alexa/smart_home'
|
|
|
|
|
2018-01-05 20:33:22 +00:00
|
|
|
CONF_DESCRIPTION = 'description'
|
|
|
|
CONF_DISPLAY_CATEGORIES = 'display_categories'
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
HANDLERS = Registry()
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
MAPPING_COMPONENT = {
|
2017-11-17 17:14:22 +00:00
|
|
|
alert.DOMAIN: ['OTHER', ('Alexa.PowerController',), None],
|
|
|
|
automation.DOMAIN: ['OTHER', ('Alexa.PowerController',), None],
|
|
|
|
cover.DOMAIN: [
|
|
|
|
'DOOR', ('Alexa.PowerController',), {
|
|
|
|
cover.SUPPORT_SET_POSITION: 'Alexa.PercentageController',
|
|
|
|
}
|
|
|
|
],
|
|
|
|
fan.DOMAIN: [
|
|
|
|
'OTHER', ('Alexa.PowerController',), {
|
|
|
|
fan.SUPPORT_SET_SPEED: 'Alexa.PercentageController',
|
|
|
|
}
|
|
|
|
],
|
|
|
|
group.DOMAIN: ['OTHER', ('Alexa.PowerController',), None],
|
|
|
|
input_boolean.DOMAIN: ['OTHER', ('Alexa.PowerController',), None],
|
2017-09-16 19:35:28 +00:00
|
|
|
light.DOMAIN: [
|
2017-10-07 20:31:57 +00:00
|
|
|
'LIGHT', ('Alexa.PowerController',), {
|
2017-11-01 03:28:17 +00:00
|
|
|
light.SUPPORT_BRIGHTNESS: 'Alexa.BrightnessController',
|
|
|
|
light.SUPPORT_RGB_COLOR: 'Alexa.ColorController',
|
2017-11-01 11:16:05 +00:00
|
|
|
light.SUPPORT_XY_COLOR: 'Alexa.ColorController',
|
2017-11-01 03:28:17 +00:00
|
|
|
light.SUPPORT_COLOR_TEMP: 'Alexa.ColorTemperatureController',
|
2017-09-16 19:35:28 +00:00
|
|
|
}
|
|
|
|
],
|
2017-11-17 17:14:22 +00:00
|
|
|
lock.DOMAIN: ['SMARTLOCK', ('Alexa.LockController',), None],
|
|
|
|
media_player.DOMAIN: [
|
|
|
|
'TV', ('Alexa.PowerController',), {
|
|
|
|
media_player.SUPPORT_VOLUME_SET: 'Alexa.Speaker',
|
|
|
|
media_player.SUPPORT_PLAY: 'Alexa.PlaybackController',
|
|
|
|
media_player.SUPPORT_PAUSE: 'Alexa.PlaybackController',
|
|
|
|
media_player.SUPPORT_STOP: 'Alexa.PlaybackController',
|
|
|
|
media_player.SUPPORT_NEXT_TRACK: 'Alexa.PlaybackController',
|
|
|
|
media_player.SUPPORT_PREVIOUS_TRACK: 'Alexa.PlaybackController',
|
|
|
|
}
|
|
|
|
],
|
|
|
|
scene.DOMAIN: ['ACTIVITY_TRIGGER', ('Alexa.SceneController',), None],
|
|
|
|
script.DOMAIN: ['OTHER', ('Alexa.PowerController',), None],
|
|
|
|
switch.DOMAIN: ['SWITCH', ('Alexa.PowerController',), None],
|
2017-09-16 19:35:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-23 08:01:18 +00:00
|
|
|
class _Cause(object):
|
|
|
|
"""Possible causes for property changes.
|
|
|
|
|
|
|
|
https://developer.amazon.com/docs/smarthome/state-reporting-for-a-smart-home-skill.html#cause-object
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Indicates that the event was caused by a customer interaction with an
|
|
|
|
# application. For example, a customer switches on a light, or locks a door
|
|
|
|
# using the Alexa app or an app provided by a device vendor.
|
|
|
|
APP_INTERACTION = 'APP_INTERACTION'
|
|
|
|
|
|
|
|
# Indicates that the event was caused by a physical interaction with an
|
|
|
|
# endpoint. For example manually switching on a light or manually locking a
|
|
|
|
# door lock
|
|
|
|
PHYSICAL_INTERACTION = 'PHYSICAL_INTERACTION'
|
|
|
|
|
|
|
|
# Indicates that the event was caused by the periodic poll of an appliance,
|
|
|
|
# which found a change in value. For example, you might poll a temperature
|
|
|
|
# sensor every hour, and send the updated temperature to Alexa.
|
|
|
|
PERIODIC_POLL = 'PERIODIC_POLL'
|
|
|
|
|
|
|
|
# Indicates that the event was caused by the application of a device rule.
|
|
|
|
# For example, a customer configures a rule to switch on a light if a
|
|
|
|
# motion sensor detects motion. In this case, Alexa receives an event from
|
|
|
|
# the motion sensor, and another event from the light to indicate that its
|
|
|
|
# state change was caused by the rule.
|
|
|
|
RULE_TRIGGER = 'RULE_TRIGGER'
|
|
|
|
|
|
|
|
# Indicates that the event was caused by a voice interaction with Alexa.
|
|
|
|
# For example a user speaking to their Echo device.
|
|
|
|
VOICE_INTERACTION = 'VOICE_INTERACTION'
|
|
|
|
|
|
|
|
|
2018-01-05 20:33:22 +00:00
|
|
|
class Config:
|
|
|
|
"""Hold the configuration for Alexa."""
|
|
|
|
|
|
|
|
def __init__(self, should_expose, entity_config=None):
|
|
|
|
"""Initialize the configuration."""
|
|
|
|
self.should_expose = should_expose
|
|
|
|
self.entity_config = entity_config or {}
|
2017-11-18 05:10:24 +00:00
|
|
|
|
|
|
|
|
2018-01-23 18:45:28 +00:00
|
|
|
@ha.callback
|
|
|
|
def async_setup(hass, config):
|
|
|
|
"""Activate Smart Home functionality of Alexa component.
|
|
|
|
|
|
|
|
This is optional, triggered by having a `smart_home:` sub-section in the
|
|
|
|
alexa configuration.
|
|
|
|
|
|
|
|
Even if that's disabled, the functionality in this module may still be used
|
|
|
|
by the cloud component which will call async_handle_message directly.
|
|
|
|
"""
|
|
|
|
smart_home_config = Config(
|
|
|
|
should_expose=config[CONF_FILTER],
|
|
|
|
entity_config=config.get(CONF_ENTITY_CONFIG),
|
|
|
|
)
|
|
|
|
hass.http.register_view(SmartHomeView(smart_home_config))
|
|
|
|
|
|
|
|
|
|
|
|
class SmartHomeView(http.HomeAssistantView):
|
|
|
|
"""Expose Smart Home v3 payload interface via HTTP POST."""
|
|
|
|
|
|
|
|
url = SMART_HOME_HTTP_ENDPOINT
|
|
|
|
name = 'api:alexa:smart_home'
|
|
|
|
|
|
|
|
def __init__(self, smart_home_config):
|
|
|
|
"""Initialize."""
|
|
|
|
self.smart_home_config = smart_home_config
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def post(self, request):
|
|
|
|
"""Handle Alexa Smart Home requests.
|
|
|
|
|
|
|
|
The Smart Home API requires the endpoint to be implemented in AWS
|
|
|
|
Lambda, which will need to forward the requests to here and pass back
|
|
|
|
the response.
|
|
|
|
"""
|
|
|
|
hass = request.app['hass']
|
|
|
|
message = yield from request.json()
|
|
|
|
|
|
|
|
_LOGGER.debug("Received Alexa Smart Home request: %s", message)
|
|
|
|
|
|
|
|
response = yield from async_handle_message(
|
|
|
|
hass, self.smart_home_config, message)
|
|
|
|
return b'' if response is None else self.json(response)
|
|
|
|
|
|
|
|
|
2017-09-16 19:35:28 +00:00
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_handle_message(hass, config, message):
|
2017-09-23 15:15:46 +00:00
|
|
|
"""Handle incoming API messages."""
|
2017-10-07 20:31:57 +00:00
|
|
|
assert message[API_DIRECTIVE][API_HEADER]['payloadVersion'] == '3'
|
|
|
|
|
|
|
|
# Read head data
|
|
|
|
message = message[API_DIRECTIVE]
|
|
|
|
namespace = message[API_HEADER]['namespace']
|
|
|
|
name = message[API_HEADER]['name']
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
# Do we support this API request?
|
2017-10-07 20:31:57 +00:00
|
|
|
funct_ref = HANDLERS.get((namespace, name))
|
2017-09-16 19:35:28 +00:00
|
|
|
if not funct_ref:
|
|
|
|
_LOGGER.warning(
|
2017-10-07 20:31:57 +00:00
|
|
|
"Unsupported API request %s/%s", namespace, name)
|
2017-09-16 19:35:28 +00:00
|
|
|
return api_error(message)
|
|
|
|
|
2017-11-18 05:10:24 +00:00
|
|
|
return (yield from funct_ref(hass, config, message))
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
def api_message(request, name='Response', namespace='Alexa', payload=None):
|
2017-09-23 15:15:46 +00:00
|
|
|
"""Create a API formatted response message.
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
|
|
|
payload = payload or {}
|
2017-10-07 20:31:57 +00:00
|
|
|
|
|
|
|
response = {
|
|
|
|
API_EVENT: {
|
|
|
|
API_HEADER: {
|
|
|
|
'namespace': namespace,
|
|
|
|
'name': name,
|
|
|
|
'messageId': str(uuid4()),
|
|
|
|
'payloadVersion': '3',
|
|
|
|
},
|
|
|
|
API_PAYLOAD: payload,
|
|
|
|
}
|
2017-09-16 19:35:28 +00:00
|
|
|
}
|
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
# If a correlation token exsits, add it to header / Need by Async requests
|
|
|
|
token = request[API_HEADER].get('correlationToken')
|
|
|
|
if token:
|
|
|
|
response[API_EVENT][API_HEADER]['correlationToken'] = token
|
|
|
|
|
|
|
|
# Extend event with endpoint object / Need by Async requests
|
|
|
|
if API_ENDPOINT in request:
|
|
|
|
response[API_EVENT][API_ENDPOINT] = request[API_ENDPOINT].copy()
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
|
def api_error(request, error_type='INTERNAL_ERROR', error_message=""):
|
2017-09-23 15:15:46 +00:00
|
|
|
"""Create a API formatted error response.
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
2017-10-07 20:31:57 +00:00
|
|
|
payload = {
|
|
|
|
'type': error_type,
|
|
|
|
'message': error_message,
|
|
|
|
}
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
return api_message(request, name='ErrorResponse', payload=payload)
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.Discovery', 'Discover'))
|
2017-09-16 19:35:28 +00:00
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_discovery(hass, config, request):
|
2017-09-23 15:15:46 +00:00
|
|
|
"""Create a API formatted discovery response.
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
2017-10-07 20:31:57 +00:00
|
|
|
discovery_endpoints = []
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
for entity in hass.states.async_all():
|
2018-01-05 20:33:22 +00:00
|
|
|
if not config.should_expose(entity.entity_id):
|
2017-11-18 05:10:24 +00:00
|
|
|
_LOGGER.debug("Not exposing %s because filtered by config",
|
|
|
|
entity.entity_id)
|
|
|
|
continue
|
|
|
|
|
2017-09-16 19:35:28 +00:00
|
|
|
class_data = MAPPING_COMPONENT.get(entity.domain)
|
|
|
|
|
|
|
|
if not class_data:
|
|
|
|
continue
|
|
|
|
|
2018-01-05 20:33:22 +00:00
|
|
|
entity_conf = config.entity_config.get(entity.entity_id, {})
|
|
|
|
|
|
|
|
friendly_name = entity_conf.get(CONF_NAME, entity.name)
|
|
|
|
description = entity_conf.get(CONF_DESCRIPTION, entity.entity_id)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
# Required description as per Amazon Scene docs
|
|
|
|
if entity.domain == scene.DOMAIN:
|
2017-11-24 23:52:59 +00:00
|
|
|
scene_fmt = '{} (Scene connected via Home Assistant)'
|
2017-11-17 17:14:22 +00:00
|
|
|
description = scene_fmt.format(description)
|
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
display_categories = entity_conf.get(
|
|
|
|
CONF_DISPLAY_CATEGORIES, class_data[0])
|
2017-11-17 17:14:22 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
endpoint = {
|
2017-11-17 17:14:22 +00:00
|
|
|
'displayCategories': [display_categories],
|
2017-09-16 19:35:28 +00:00
|
|
|
'additionalApplianceDetails': {},
|
2017-10-07 20:31:57 +00:00
|
|
|
'endpointId': entity.entity_id.replace('.', '#'),
|
2017-11-17 17:14:22 +00:00
|
|
|
'friendlyName': friendly_name,
|
|
|
|
'description': description,
|
|
|
|
'manufacturerName': 'Home Assistant',
|
2017-09-16 19:35:28 +00:00
|
|
|
}
|
2017-10-07 20:31:57 +00:00
|
|
|
actions = set()
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
# static actions
|
|
|
|
if class_data[1]:
|
2017-10-07 20:31:57 +00:00
|
|
|
actions |= set(class_data[1])
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
# dynamic actions
|
|
|
|
if class_data[2]:
|
|
|
|
supported = entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
for feature, action_name in class_data[2].items():
|
|
|
|
if feature & supported > 0:
|
2017-10-07 20:31:57 +00:00
|
|
|
actions.add(action_name)
|
|
|
|
|
|
|
|
# Write action into capabilities
|
|
|
|
capabilities = []
|
|
|
|
for action in actions:
|
|
|
|
capabilities.append({
|
|
|
|
'type': 'AlexaInterface',
|
|
|
|
'interface': action,
|
|
|
|
'version': 3,
|
|
|
|
})
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
endpoint['capabilities'] = capabilities
|
|
|
|
discovery_endpoints.append(endpoint)
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
return api_message(
|
2017-10-07 20:31:57 +00:00
|
|
|
request, name='Discover.Response', namespace='Alexa.Discovery',
|
|
|
|
payload={'endpoints': discovery_endpoints})
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
def extract_entity(funct):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Decorate for extract entity object from request."""
|
2017-09-16 19:35:28 +00:00
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_entity_wrapper(hass, config, request):
|
2017-09-16 19:35:28 +00:00
|
|
|
"""Process a turn on request."""
|
2017-10-07 20:31:57 +00:00
|
|
|
entity_id = request[API_ENDPOINT]['endpointId'].replace('#', '.')
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
# extract state object
|
|
|
|
entity = hass.states.get(entity_id)
|
|
|
|
if not entity:
|
|
|
|
_LOGGER.error("Can't process %s for %s",
|
2017-10-07 20:31:57 +00:00
|
|
|
request[API_HEADER]['name'], entity_id)
|
|
|
|
return api_error(request, error_type='NO_SUCH_ENDPOINT')
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-11-18 05:10:24 +00:00
|
|
|
return (yield from funct(hass, config, request, entity))
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
return async_api_entity_wrapper
|
|
|
|
|
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
@HANDLERS.register(('Alexa.PowerController', 'TurnOn'))
|
2017-09-16 19:35:28 +00:00
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_turn_on(hass, config, request, entity):
|
2017-09-16 19:35:28 +00:00
|
|
|
"""Process a turn on request."""
|
2017-11-18 05:10:24 +00:00
|
|
|
domain = entity.domain
|
|
|
|
if entity.domain == group.DOMAIN:
|
|
|
|
domain = ha.DOMAIN
|
|
|
|
|
2017-12-24 23:05:56 +00:00
|
|
|
service = SERVICE_TURN_ON
|
|
|
|
if entity.domain == cover.DOMAIN:
|
|
|
|
service = cover.SERVICE_OPEN_COVER
|
|
|
|
|
|
|
|
yield from hass.services.async_call(domain, service, {
|
2017-09-16 19:35:28 +00:00
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
return api_message(request)
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
@HANDLERS.register(('Alexa.PowerController', 'TurnOff'))
|
2017-09-16 19:35:28 +00:00
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_turn_off(hass, config, request, entity):
|
2017-09-16 19:35:28 +00:00
|
|
|
"""Process a turn off request."""
|
2017-11-18 05:10:24 +00:00
|
|
|
domain = entity.domain
|
|
|
|
if entity.domain == group.DOMAIN:
|
|
|
|
domain = ha.DOMAIN
|
|
|
|
|
2017-12-24 23:05:56 +00:00
|
|
|
service = SERVICE_TURN_OFF
|
|
|
|
if entity.domain == cover.DOMAIN:
|
|
|
|
service = cover.SERVICE_CLOSE_COVER
|
|
|
|
|
|
|
|
yield from hass.services.async_call(domain, service, {
|
2017-09-16 19:35:28 +00:00
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
return api_message(request)
|
2017-09-16 19:35:28 +00:00
|
|
|
|
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
@HANDLERS.register(('Alexa.BrightnessController', 'SetBrightness'))
|
2017-09-16 19:35:28 +00:00
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_set_brightness(hass, config, request, entity):
|
2017-10-07 20:31:57 +00:00
|
|
|
"""Process a set brightness request."""
|
2017-11-01 03:28:17 +00:00
|
|
|
brightness = int(request[API_PAYLOAD]['brightness'])
|
2017-09-16 19:35:28 +00:00
|
|
|
|
2017-10-07 20:31:57 +00:00
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
2017-11-01 03:28:17 +00:00
|
|
|
light.ATTR_BRIGHTNESS_PCT: brightness,
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-01 03:28:17 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.BrightnessController', 'AdjustBrightness'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_adjust_brightness(hass, config, request, entity):
|
2017-11-01 03:28:17 +00:00
|
|
|
"""Process a adjust brightness request."""
|
|
|
|
brightness_delta = int(request[API_PAYLOAD]['brightnessDelta'])
|
|
|
|
|
|
|
|
# read current state
|
|
|
|
try:
|
|
|
|
current = math.floor(
|
|
|
|
int(entity.attributes.get(light.ATTR_BRIGHTNESS)) / 255 * 100)
|
|
|
|
except ZeroDivisionError:
|
2017-11-01 11:16:05 +00:00
|
|
|
current = 0
|
2017-11-01 03:28:17 +00:00
|
|
|
|
|
|
|
# set brightness
|
2017-11-01 11:16:05 +00:00
|
|
|
brightness = max(0, brightness_delta + current)
|
2017-11-01 03:28:17 +00:00
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
light.ATTR_BRIGHTNESS_PCT: brightness,
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-01 03:28:17 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.ColorController', 'SetColor'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_set_color(hass, config, request, entity):
|
2017-11-01 03:28:17 +00:00
|
|
|
"""Process a set color request."""
|
2017-11-01 11:16:05 +00:00
|
|
|
supported = entity.attributes.get(ATTR_SUPPORTED_FEATURES)
|
|
|
|
rgb = color_util.color_hsb_to_RGB(
|
|
|
|
float(request[API_PAYLOAD]['color']['hue']),
|
|
|
|
float(request[API_PAYLOAD]['color']['saturation']),
|
|
|
|
float(request[API_PAYLOAD]['color']['brightness'])
|
|
|
|
)
|
|
|
|
|
|
|
|
if supported & light.SUPPORT_RGB_COLOR > 0:
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
light.ATTR_RGB_COLOR: rgb,
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-01 11:16:05 +00:00
|
|
|
else:
|
|
|
|
xyz = color_util.color_RGB_to_xy(*rgb)
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
light.ATTR_XY_COLOR: (xyz[0], xyz[1]),
|
|
|
|
light.ATTR_BRIGHTNESS: xyz[2],
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-01 03:28:17 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.ColorTemperatureController', 'SetColorTemperature'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_set_color_temperature(hass, config, request, entity):
|
2017-11-01 03:28:17 +00:00
|
|
|
"""Process a set color temperature request."""
|
|
|
|
kelvin = int(request[API_PAYLOAD]['colorTemperatureInKelvin'])
|
|
|
|
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
light.ATTR_KELVIN: kelvin,
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-01 03:28:17 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(
|
|
|
|
('Alexa.ColorTemperatureController', 'DecreaseColorTemperature'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_decrease_color_temp(hass, config, request, entity):
|
2017-11-01 03:28:17 +00:00
|
|
|
"""Process a decrease color temperature request."""
|
|
|
|
current = int(entity.attributes.get(light.ATTR_COLOR_TEMP))
|
|
|
|
max_mireds = int(entity.attributes.get(light.ATTR_MAX_MIREDS))
|
|
|
|
|
|
|
|
value = min(max_mireds, current + 50)
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
light.ATTR_COLOR_TEMP: value,
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-01 03:28:17 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(
|
|
|
|
('Alexa.ColorTemperatureController', 'IncreaseColorTemperature'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_increase_color_temp(hass, config, request, entity):
|
2017-11-01 03:28:17 +00:00
|
|
|
"""Process a increase color temperature request."""
|
|
|
|
current = int(entity.attributes.get(light.ATTR_COLOR_TEMP))
|
|
|
|
min_mireds = int(entity.attributes.get(light.ATTR_MIN_MIREDS))
|
|
|
|
|
|
|
|
value = max(min_mireds, current - 50)
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
light.ATTR_COLOR_TEMP: value,
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-10-07 20:31:57 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.SceneController', 'Activate'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_activate(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a activate request."""
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_TURN_ON, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
2018-01-23 08:01:18 +00:00
|
|
|
payload = {
|
|
|
|
'cause': {'type': _Cause.VOICE_INTERACTION},
|
|
|
|
'timestamp': '%sZ' % (datetime.utcnow().isoformat(),)
|
|
|
|
}
|
|
|
|
|
|
|
|
return api_message(
|
|
|
|
request,
|
|
|
|
name='ActivationStarted',
|
|
|
|
namespace='Alexa.SceneController',
|
|
|
|
payload=payload,
|
|
|
|
)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.PercentageController', 'SetPercentage'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_set_percentage(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a set percentage request."""
|
|
|
|
percentage = int(request[API_PAYLOAD]['percentage'])
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
|
|
|
if entity.domain == fan.DOMAIN:
|
|
|
|
service = fan.SERVICE_SET_SPEED
|
|
|
|
speed = "off"
|
|
|
|
|
|
|
|
if percentage <= 33:
|
|
|
|
speed = "low"
|
|
|
|
elif percentage <= 66:
|
|
|
|
speed = "medium"
|
|
|
|
elif percentage <= 100:
|
|
|
|
speed = "high"
|
|
|
|
data[fan.ATTR_SPEED] = speed
|
|
|
|
|
|
|
|
elif entity.domain == cover.DOMAIN:
|
|
|
|
service = SERVICE_SET_COVER_POSITION
|
|
|
|
data[cover.ATTR_POSITION] = percentage
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, service, data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.PercentageController', 'AdjustPercentage'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_adjust_percentage(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a adjust percentage request."""
|
|
|
|
percentage_delta = int(request[API_PAYLOAD]['percentageDelta'])
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
|
|
|
if entity.domain == fan.DOMAIN:
|
|
|
|
service = fan.SERVICE_SET_SPEED
|
|
|
|
speed = entity.attributes.get(fan.ATTR_SPEED)
|
|
|
|
|
|
|
|
if speed == "off":
|
|
|
|
current = 0
|
|
|
|
elif speed == "low":
|
|
|
|
current = 33
|
|
|
|
elif speed == "medium":
|
|
|
|
current = 66
|
|
|
|
elif speed == "high":
|
|
|
|
current = 100
|
|
|
|
|
|
|
|
# set percentage
|
|
|
|
percentage = max(0, percentage_delta + current)
|
|
|
|
speed = "off"
|
|
|
|
|
|
|
|
if percentage <= 33:
|
|
|
|
speed = "low"
|
|
|
|
elif percentage <= 66:
|
|
|
|
speed = "medium"
|
|
|
|
elif percentage <= 100:
|
|
|
|
speed = "high"
|
|
|
|
|
|
|
|
data[fan.ATTR_SPEED] = speed
|
|
|
|
|
|
|
|
elif entity.domain == cover.DOMAIN:
|
|
|
|
service = SERVICE_SET_COVER_POSITION
|
|
|
|
|
|
|
|
current = entity.attributes.get(cover.ATTR_POSITION)
|
|
|
|
|
|
|
|
data[cover.ATTR_POSITION] = max(0, percentage_delta + current)
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, service, data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.LockController', 'Lock'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_lock(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a lock request."""
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_LOCK, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
# Not supported by Alexa yet
|
|
|
|
@HANDLERS.register(('Alexa.LockController', 'Unlock'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_unlock(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a unlock request."""
|
|
|
|
yield from hass.services.async_call(entity.domain, SERVICE_UNLOCK, {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
2017-12-29 17:44:06 +00:00
|
|
|
}, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.Speaker', 'SetVolume'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_set_volume(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a set volume request."""
|
|
|
|
volume = round(float(request[API_PAYLOAD]['volume'] / 100), 2)
|
|
|
|
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.ATTR_MEDIA_VOLUME_LEVEL: volume,
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, SERVICE_VOLUME_SET,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.Speaker', 'AdjustVolume'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_adjust_volume(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a adjust volume request."""
|
|
|
|
volume_delta = int(request[API_PAYLOAD]['volume'])
|
|
|
|
|
|
|
|
current_level = entity.attributes.get(media_player.ATTR_MEDIA_VOLUME_LEVEL)
|
|
|
|
|
|
|
|
# read current state
|
|
|
|
try:
|
|
|
|
current = math.floor(int(current_level * 100))
|
|
|
|
except ZeroDivisionError:
|
|
|
|
current = 0
|
|
|
|
|
|
|
|
volume = float(max(0, volume_delta + current) / 100)
|
|
|
|
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.ATTR_MEDIA_VOLUME_LEVEL: volume,
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, media_player.SERVICE_VOLUME_SET,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.Speaker', 'SetMute'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_set_mute(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a set mute request."""
|
|
|
|
mute = bool(request[API_PAYLOAD]['mute'])
|
|
|
|
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.ATTR_MEDIA_VOLUME_MUTED: mute,
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, media_player.SERVICE_VOLUME_MUTE,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.PlaybackController', 'Play'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_play(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a play request."""
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, SERVICE_MEDIA_PLAY,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.PlaybackController', 'Pause'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_pause(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a pause request."""
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, SERVICE_MEDIA_PAUSE,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.PlaybackController', 'Stop'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_stop(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a stop request."""
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, SERVICE_MEDIA_STOP,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.PlaybackController', 'Next'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_next(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a next request."""
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, SERVICE_MEDIA_NEXT_TRACK,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(('Alexa.PlaybackController', 'Previous'))
|
|
|
|
@extract_entity
|
|
|
|
@asyncio.coroutine
|
2017-11-18 05:10:24 +00:00
|
|
|
def async_api_previous(hass, config, request, entity):
|
2017-11-17 17:14:22 +00:00
|
|
|
"""Process a previous request."""
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:44:06 +00:00
|
|
|
yield from hass.services.async_call(
|
|
|
|
entity.domain, SERVICE_MEDIA_PREVIOUS_TRACK,
|
|
|
|
data, blocking=False)
|
2017-11-17 17:14:22 +00:00
|
|
|
|
|
|
|
return api_message(request)
|