2019-06-13 15:43:57 +00:00
|
|
|
"""Alexa message handlers."""
|
|
|
|
import logging
|
|
|
|
import math
|
|
|
|
|
|
|
|
from homeassistant import core as ha
|
2019-12-23 14:17:37 +00:00
|
|
|
from homeassistant.components import (
|
2020-03-28 04:19:11 +00:00
|
|
|
camera,
|
2019-12-23 14:17:37 +00:00
|
|
|
cover,
|
|
|
|
fan,
|
|
|
|
group,
|
|
|
|
input_number,
|
|
|
|
light,
|
|
|
|
media_player,
|
2020-01-10 22:11:50 +00:00
|
|
|
timer,
|
2020-01-15 17:15:31 +00:00
|
|
|
vacuum,
|
2019-12-23 14:17:37 +00:00
|
|
|
)
|
2019-07-11 15:35:46 +00:00
|
|
|
from homeassistant.components.climate import const as climate
|
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_ENTITY_ID,
|
|
|
|
ATTR_SUPPORTED_FEATURES,
|
|
|
|
ATTR_TEMPERATURE,
|
2019-10-04 15:41:47 +00:00
|
|
|
SERVICE_ALARM_ARM_AWAY,
|
|
|
|
SERVICE_ALARM_ARM_HOME,
|
|
|
|
SERVICE_ALARM_ARM_NIGHT,
|
|
|
|
SERVICE_ALARM_DISARM,
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_LOCK,
|
|
|
|
SERVICE_MEDIA_NEXT_TRACK,
|
|
|
|
SERVICE_MEDIA_PAUSE,
|
|
|
|
SERVICE_MEDIA_PLAY,
|
|
|
|
SERVICE_MEDIA_PREVIOUS_TRACK,
|
|
|
|
SERVICE_MEDIA_STOP,
|
|
|
|
SERVICE_SET_COVER_POSITION,
|
2019-12-19 11:44:17 +00:00
|
|
|
SERVICE_SET_COVER_TILT_POSITION,
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_TURN_OFF,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
SERVICE_UNLOCK,
|
|
|
|
SERVICE_VOLUME_DOWN,
|
|
|
|
SERVICE_VOLUME_MUTE,
|
|
|
|
SERVICE_VOLUME_SET,
|
|
|
|
SERVICE_VOLUME_UP,
|
2019-11-25 23:07:33 +00:00
|
|
|
STATE_ALARM_DISARMED,
|
2019-07-31 19:25:30 +00:00
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT,
|
|
|
|
)
|
2020-03-28 04:19:11 +00:00
|
|
|
from homeassistant.helpers import network
|
2019-07-11 15:35:46 +00:00
|
|
|
import homeassistant.util.color as color_util
|
|
|
|
from homeassistant.util.decorator import Registry
|
2019-12-08 13:56:42 +00:00
|
|
|
import homeassistant.util.dt as dt_util
|
2019-06-13 15:43:57 +00:00
|
|
|
from homeassistant.util.temperature import convert as convert_temperature
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
from .const import (
|
|
|
|
API_TEMP_UNITS,
|
|
|
|
API_THERMOSTAT_MODES,
|
2019-12-08 13:56:42 +00:00
|
|
|
API_THERMOSTAT_MODES_CUSTOM,
|
2019-10-23 05:01:03 +00:00
|
|
|
API_THERMOSTAT_PRESETS,
|
|
|
|
PERCENTAGE_FAN_MAP,
|
2019-12-08 13:56:42 +00:00
|
|
|
Cause,
|
|
|
|
Inputs,
|
2019-10-23 05:01:03 +00:00
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
from .entities import async_get_entities
|
|
|
|
from .errors import (
|
2019-10-23 05:01:03 +00:00
|
|
|
AlexaInvalidDirectiveError,
|
2019-07-31 19:25:30 +00:00
|
|
|
AlexaInvalidValueError,
|
2019-10-04 15:41:47 +00:00
|
|
|
AlexaSecurityPanelAuthorizationRequired,
|
|
|
|
AlexaSecurityPanelUnauthorizedError,
|
2019-07-31 19:25:30 +00:00
|
|
|
AlexaTempRangeError,
|
|
|
|
AlexaUnsupportedThermostatModeError,
|
2019-10-31 09:38:44 +00:00
|
|
|
AlexaVideoActionNotPermittedForContentError,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-07-11 15:35:46 +00:00
|
|
|
from .state_report import async_enable_proactive_mode
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
HANDLERS = Registry()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.Discovery", "Discover"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_discovery(hass, config, directive, context):
|
|
|
|
"""Create a API formatted discovery response.
|
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
|
|
|
discovery_endpoints = [
|
2019-06-19 08:06:29 +00:00
|
|
|
alexa_entity.serialize_discovery()
|
2019-06-13 15:43:57 +00:00
|
|
|
for alexa_entity in async_get_entities(hass, config)
|
|
|
|
if config.should_expose(alexa_entity.entity_id)
|
|
|
|
]
|
|
|
|
|
|
|
|
return directive.response(
|
2019-07-31 19:25:30 +00:00
|
|
|
name="Discover.Response",
|
|
|
|
namespace="Alexa.Discovery",
|
|
|
|
payload={"endpoints": discovery_endpoints},
|
2019-06-13 15:43:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.Authorization", "AcceptGrant"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_accept_grant(hass, config, directive, context):
|
|
|
|
"""Create a API formatted AcceptGrant response.
|
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
auth_code = directive.payload["grant"]["code"]
|
2019-06-13 15:43:57 +00:00
|
|
|
_LOGGER.debug("AcceptGrant code: %s", auth_code)
|
|
|
|
|
2019-06-13 18:58:08 +00:00
|
|
|
if config.supports_auth:
|
|
|
|
await config.async_accept_grant(auth_code)
|
2019-06-17 20:50:01 +00:00
|
|
|
|
|
|
|
if config.should_report_state:
|
|
|
|
await async_enable_proactive_mode(hass, config)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response(
|
2019-07-31 19:25:30 +00:00
|
|
|
name="AcceptGrant.Response", namespace="Alexa.Authorization", payload={}
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PowerController", "TurnOn"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_turn_on(hass, config, directive, context):
|
|
|
|
"""Process a turn on request."""
|
|
|
|
entity = directive.entity
|
|
|
|
domain = entity.domain
|
|
|
|
if domain == group.DOMAIN:
|
|
|
|
domain = ha.DOMAIN
|
|
|
|
|
|
|
|
service = SERVICE_TURN_ON
|
2020-01-29 17:03:20 +00:00
|
|
|
if domain == cover.DOMAIN:
|
|
|
|
service = cover.SERVICE_OPEN_COVER
|
2020-03-09 15:58:47 +00:00
|
|
|
elif domain == vacuum.DOMAIN:
|
|
|
|
supported = entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
if not supported & vacuum.SUPPORT_TURN_ON and supported & vacuum.SUPPORT_START:
|
|
|
|
service = vacuum.SERVICE_START
|
2020-03-10 04:59:06 +00:00
|
|
|
elif domain == timer.DOMAIN:
|
|
|
|
service = timer.SERVICE_START
|
2020-01-29 17:03:20 +00:00
|
|
|
elif domain == media_player.DOMAIN:
|
2019-07-11 15:35:46 +00:00
|
|
|
supported = entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
2019-07-31 19:25:30 +00:00
|
|
|
power_features = media_player.SUPPORT_TURN_ON | media_player.SUPPORT_TURN_OFF
|
2019-07-11 15:35:46 +00:00
|
|
|
if not supported & power_features:
|
|
|
|
service = media_player.SERVICE_MEDIA_PLAY
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
domain,
|
|
|
|
service,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PowerController", "TurnOff"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_turn_off(hass, config, directive, context):
|
|
|
|
"""Process a turn off request."""
|
|
|
|
entity = directive.entity
|
|
|
|
domain = entity.domain
|
|
|
|
if entity.domain == group.DOMAIN:
|
|
|
|
domain = ha.DOMAIN
|
|
|
|
|
|
|
|
service = SERVICE_TURN_OFF
|
2020-01-29 17:03:20 +00:00
|
|
|
if entity.domain == cover.DOMAIN:
|
|
|
|
service = cover.SERVICE_CLOSE_COVER
|
2020-03-09 15:58:47 +00:00
|
|
|
elif domain == vacuum.DOMAIN:
|
|
|
|
supported = entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
|
|
|
if (
|
|
|
|
not supported & vacuum.SUPPORT_TURN_OFF
|
|
|
|
and supported & vacuum.SUPPORT_RETURN_HOME
|
|
|
|
):
|
|
|
|
service = vacuum.SERVICE_RETURN_TO_BASE
|
2020-03-10 04:59:06 +00:00
|
|
|
elif domain == timer.DOMAIN:
|
|
|
|
service = timer.SERVICE_CANCEL
|
2020-01-29 17:03:20 +00:00
|
|
|
elif domain == media_player.DOMAIN:
|
2019-07-11 15:35:46 +00:00
|
|
|
supported = entity.attributes.get(ATTR_SUPPORTED_FEATURES, 0)
|
2019-07-31 19:25:30 +00:00
|
|
|
power_features = media_player.SUPPORT_TURN_ON | media_player.SUPPORT_TURN_OFF
|
2019-07-11 15:35:46 +00:00
|
|
|
if not supported & power_features:
|
|
|
|
service = media_player.SERVICE_MEDIA_STOP
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
domain,
|
|
|
|
service,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.BrightnessController", "SetBrightness"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_brightness(hass, config, directive, context):
|
|
|
|
"""Process a set brightness request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
brightness = int(directive.payload["brightness"])
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id, light.ATTR_BRIGHTNESS_PCT: brightness},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.BrightnessController", "AdjustBrightness"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_adjust_brightness(hass, config, directive, context):
|
|
|
|
"""Process an adjust brightness request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
brightness_delta = int(directive.payload["brightnessDelta"])
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
# read current state
|
|
|
|
try:
|
|
|
|
current = math.floor(
|
2019-07-31 19:25:30 +00:00
|
|
|
int(entity.attributes.get(light.ATTR_BRIGHTNESS)) / 255 * 100
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
except ZeroDivisionError:
|
|
|
|
current = 0
|
|
|
|
|
|
|
|
# set brightness
|
|
|
|
brightness = max(0, brightness_delta + current)
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id, light.ATTR_BRIGHTNESS_PCT: brightness},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.ColorController", "SetColor"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_color(hass, config, directive, context):
|
|
|
|
"""Process a set color request."""
|
|
|
|
entity = directive.entity
|
|
|
|
rgb = color_util.color_hsb_to_RGB(
|
2019-07-31 19:25:30 +00:00
|
|
|
float(directive.payload["color"]["hue"]),
|
|
|
|
float(directive.payload["color"]["saturation"]),
|
|
|
|
float(directive.payload["color"]["brightness"]),
|
2019-06-13 15:43:57 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id, light.ATTR_RGB_COLOR: rgb},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.ColorTemperatureController", "SetColorTemperature"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_color_temperature(hass, config, directive, context):
|
|
|
|
"""Process a set color temperature request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
kelvin = int(directive.payload["colorTemperatureInKelvin"])
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id, light.ATTR_KELVIN: kelvin},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.ColorTemperatureController", "DecreaseColorTemperature"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_decrease_color_temp(hass, config, directive, context):
|
|
|
|
"""Process a decrease color temperature request."""
|
|
|
|
entity = directive.entity
|
|
|
|
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)
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id, light.ATTR_COLOR_TEMP: value},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.ColorTemperatureController", "IncreaseColorTemperature"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_increase_color_temp(hass, config, directive, context):
|
|
|
|
"""Process an increase color temperature request."""
|
|
|
|
entity = directive.entity
|
|
|
|
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)
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id, light.ATTR_COLOR_TEMP: value},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.SceneController", "Activate"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_activate(hass, config, directive, context):
|
|
|
|
"""Process an activate request."""
|
|
|
|
entity = directive.entity
|
|
|
|
domain = entity.domain
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
domain,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
payload = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"cause": {"type": Cause.VOICE_INTERACTION},
|
2019-09-19 16:19:27 +00:00
|
|
|
"timestamp": f"{dt_util.utcnow().replace(tzinfo=None).isoformat()}Z",
|
2019-06-13 15:43:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return directive.response(
|
2019-07-31 19:25:30 +00:00
|
|
|
name="ActivationStarted", namespace="Alexa.SceneController", payload=payload
|
2019-06-13 15:43:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.SceneController", "Deactivate"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_deactivate(hass, config, directive, context):
|
|
|
|
"""Process a deactivate request."""
|
|
|
|
entity = directive.entity
|
|
|
|
domain = entity.domain
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
domain,
|
|
|
|
SERVICE_TURN_OFF,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
payload = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"cause": {"type": Cause.VOICE_INTERACTION},
|
2019-09-19 16:19:27 +00:00
|
|
|
"timestamp": f"{dt_util.utcnow().replace(tzinfo=None).isoformat()}Z",
|
2019-06-13 15:43:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return directive.response(
|
2019-07-31 19:25:30 +00:00
|
|
|
name="DeactivationStarted", namespace="Alexa.SceneController", payload=payload
|
2019-06-13 15:43:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PercentageController", "SetPercentage"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_percentage(hass, config, directive, context):
|
|
|
|
"""Process a set percentage request."""
|
|
|
|
entity = directive.entity
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
|
|
|
if entity.domain == fan.DOMAIN:
|
|
|
|
service = fan.SERVICE_SET_SPEED
|
|
|
|
speed = "off"
|
|
|
|
|
2020-04-04 21:20:48 +00:00
|
|
|
percentage = int(directive.payload["percentage"])
|
2019-06-13 15:43:57 +00:00
|
|
|
if percentage <= 33:
|
|
|
|
speed = "low"
|
|
|
|
elif percentage <= 66:
|
|
|
|
speed = "medium"
|
|
|
|
elif percentage <= 100:
|
|
|
|
speed = "high"
|
|
|
|
data[fan.ATTR_SPEED] = speed
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, service, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PercentageController", "AdjustPercentage"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_adjust_percentage(hass, config, directive, context):
|
|
|
|
"""Process an adjust percentage request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
percentage_delta = int(directive.payload["percentageDelta"])
|
2019-06-13 15:43:57 +00:00
|
|
|
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)
|
2019-10-23 05:01:03 +00:00
|
|
|
current = PERCENTAGE_FAN_MAP.get(speed, 100)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
# 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
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, service, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.LockController", "Lock"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_lock(hass, config, directive, context):
|
|
|
|
"""Process a lock request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_LOCK,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
response = directive.response()
|
2019-07-31 19:25:30 +00:00
|
|
|
response.add_context_property(
|
|
|
|
{"name": "lockState", "namespace": "Alexa.LockController", "value": "LOCKED"}
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
return response
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.LockController", "Unlock"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_unlock(hass, config, directive, context):
|
|
|
|
"""Process an unlock request."""
|
2020-01-03 20:23:22 +00:00
|
|
|
if config.locale not in {"de-DE", "en-US", "ja-JP"}:
|
|
|
|
msg = f"The unlock directive is not supported for the following locales: {config.locale}"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
SERVICE_UNLOCK,
|
|
|
|
{ATTR_ENTITY_ID: entity.entity_id},
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-10-14 21:19:05 +00:00
|
|
|
response = directive.response()
|
|
|
|
response.add_context_property(
|
|
|
|
{"namespace": "Alexa.LockController", "name": "lockState", "value": "UNLOCKED"}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.Speaker", "SetVolume"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_volume(hass, config, directive, context):
|
|
|
|
"""Process a set volume request."""
|
2019-07-31 19:25:30 +00:00
|
|
|
volume = round(float(directive.payload["volume"] / 100), 2)
|
2019-06-13 15:43:57 +00:00
|
|
|
entity = directive.entity
|
|
|
|
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.const.ATTR_MEDIA_VOLUME_LEVEL: volume,
|
|
|
|
}
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, SERVICE_VOLUME_SET, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.InputController", "SelectInput"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_select_input(hass, config, directive, context):
|
|
|
|
"""Process a set input request."""
|
2019-07-31 19:25:30 +00:00
|
|
|
media_input = directive.payload["input"]
|
2019-06-13 15:43:57 +00:00
|
|
|
entity = directive.entity
|
|
|
|
|
2019-11-25 23:17:12 +00:00
|
|
|
# Attempt to map the ALL UPPERCASE payload name to a source.
|
|
|
|
# Strips trailing 1 to match single input devices.
|
|
|
|
source_list = entity.attributes.get(media_player.const.ATTR_INPUT_SOURCE_LIST, [])
|
2019-06-13 15:43:57 +00:00
|
|
|
for source in source_list:
|
2019-11-25 23:17:12 +00:00
|
|
|
formatted_source = (
|
|
|
|
source.lower().replace("-", "").replace("_", "").replace(" ", "")
|
|
|
|
)
|
|
|
|
media_input = media_input.lower().replace(" ", "")
|
|
|
|
if (
|
|
|
|
formatted_source in Inputs.VALID_SOURCE_NAME_MAP.keys()
|
|
|
|
and formatted_source == media_input
|
|
|
|
) or (
|
|
|
|
media_input.endswith("1") and formatted_source == media_input.rstrip("1")
|
|
|
|
):
|
2019-06-13 15:43:57 +00:00
|
|
|
media_input = source
|
|
|
|
break
|
|
|
|
else:
|
2020-02-23 21:38:05 +00:00
|
|
|
msg = (
|
|
|
|
f"failed to map input {media_input} to a media source on {entity.entity_id}"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
raise AlexaInvalidValueError(msg)
|
|
|
|
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.const.ATTR_INPUT_SOURCE: media_input,
|
|
|
|
}
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain,
|
|
|
|
media_player.SERVICE_SELECT_SOURCE,
|
|
|
|
data,
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.Speaker", "AdjustVolume"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_adjust_volume(hass, config, directive, context):
|
|
|
|
"""Process an adjust volume request."""
|
2019-07-31 19:25:30 +00:00
|
|
|
volume_delta = int(directive.payload["volume"])
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
current_level = entity.attributes.get(media_player.const.ATTR_MEDIA_VOLUME_LEVEL)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
# 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.const.ATTR_MEDIA_VOLUME_LEVEL: volume,
|
|
|
|
}
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, SERVICE_VOLUME_SET, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.StepSpeaker", "AdjustVolume"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_adjust_volume_step(hass, config, directive, context):
|
|
|
|
"""Process an adjust volume step request."""
|
|
|
|
# media_player volume up/down service does not support specifying steps
|
|
|
|
# each component handles it differently e.g. via config.
|
2019-10-23 15:28:23 +00:00
|
|
|
# This workaround will simply call the volume up/Volume down the amount of steps asked for
|
|
|
|
# When no steps are called in the request, Alexa sends a default of 10 steps which for most
|
|
|
|
# purposes is too high. The default is set 1 in this case.
|
2019-06-13 15:43:57 +00:00
|
|
|
entity = directive.entity
|
2019-10-23 15:28:23 +00:00
|
|
|
volume_int = int(directive.payload["volumeSteps"])
|
|
|
|
is_default = bool(directive.payload["volumeStepsDefault"])
|
|
|
|
default_steps = 1
|
|
|
|
|
|
|
|
if volume_int < 0:
|
|
|
|
service_volume = SERVICE_VOLUME_DOWN
|
|
|
|
if is_default:
|
|
|
|
volume_int = -default_steps
|
|
|
|
else:
|
|
|
|
service_volume = SERVICE_VOLUME_UP
|
|
|
|
if is_default:
|
|
|
|
volume_int = default_steps
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2020-04-04 21:20:48 +00:00
|
|
|
for _ in range(abs(volume_int)):
|
2019-06-13 15:43:57 +00:00
|
|
|
await hass.services.async_call(
|
2019-10-23 15:28:23 +00:00
|
|
|
entity.domain, service_volume, data, blocking=False, context=context
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.StepSpeaker", "SetMute"))
|
|
|
|
@HANDLERS.register(("Alexa.Speaker", "SetMute"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_mute(hass, config, directive, context):
|
|
|
|
"""Process a set mute request."""
|
2019-07-31 19:25:30 +00:00
|
|
|
mute = bool(directive.payload["mute"])
|
2019-06-13 15:43:57 +00:00
|
|
|
entity = directive.entity
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.const.ATTR_MEDIA_VOLUME_MUTED: mute,
|
|
|
|
}
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, SERVICE_VOLUME_MUTE, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PlaybackController", "Play"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_play(hass, config, directive, context):
|
|
|
|
"""Process a play request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, SERVICE_MEDIA_PLAY, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PlaybackController", "Pause"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_pause(hass, config, directive, context):
|
|
|
|
"""Process a pause request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, SERVICE_MEDIA_PAUSE, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PlaybackController", "Stop"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_stop(hass, config, directive, context):
|
|
|
|
"""Process a stop request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, SERVICE_MEDIA_STOP, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PlaybackController", "Next"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_next(hass, config, directive, context):
|
|
|
|
"""Process a next request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain, SERVICE_MEDIA_NEXT_TRACK, data, blocking=False, context=context
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.PlaybackController", "Previous"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_previous(hass, config, directive, context):
|
|
|
|
"""Process a previous request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain,
|
|
|
|
SERVICE_MEDIA_PREVIOUS_TRACK,
|
|
|
|
data,
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
|
|
|
def temperature_from_object(hass, temp_obj, interval=False):
|
|
|
|
"""Get temperature from Temperature object in requested unit."""
|
|
|
|
to_unit = hass.config.units.temperature_unit
|
|
|
|
from_unit = TEMP_CELSIUS
|
2019-07-31 19:25:30 +00:00
|
|
|
temp = float(temp_obj["value"])
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if temp_obj["scale"] == "FAHRENHEIT":
|
2019-06-13 15:43:57 +00:00
|
|
|
from_unit = TEMP_FAHRENHEIT
|
2019-07-31 19:25:30 +00:00
|
|
|
elif temp_obj["scale"] == "KELVIN":
|
2019-06-13 15:43:57 +00:00
|
|
|
# convert to Celsius if absolute temperature
|
|
|
|
if not interval:
|
|
|
|
temp -= 273.15
|
|
|
|
|
|
|
|
return convert_temperature(temp, from_unit, to_unit, interval)
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.ThermostatController", "SetTargetTemperature"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_target_temp(hass, config, directive, context):
|
|
|
|
"""Process a set target temperature request."""
|
|
|
|
entity = directive.entity
|
|
|
|
min_temp = entity.attributes.get(climate.ATTR_MIN_TEMP)
|
|
|
|
max_temp = entity.attributes.get(climate.ATTR_MAX_TEMP)
|
|
|
|
unit = hass.config.units.temperature_unit
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
payload = directive.payload
|
|
|
|
response = directive.response()
|
2019-07-31 19:25:30 +00:00
|
|
|
if "targetSetpoint" in payload:
|
|
|
|
temp = temperature_from_object(hass, payload["targetSetpoint"])
|
2019-06-13 15:43:57 +00:00
|
|
|
if temp < min_temp or temp > max_temp:
|
|
|
|
raise AlexaTempRangeError(hass, temp, min_temp, max_temp)
|
|
|
|
data[ATTR_TEMPERATURE] = temp
|
2019-07-31 19:25:30 +00:00
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"name": "targetSetpoint",
|
|
|
|
"namespace": "Alexa.ThermostatController",
|
|
|
|
"value": {"value": temp, "scale": API_TEMP_UNITS[unit]},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
if "lowerSetpoint" in payload:
|
|
|
|
temp_low = temperature_from_object(hass, payload["lowerSetpoint"])
|
2019-06-13 15:43:57 +00:00
|
|
|
if temp_low < min_temp or temp_low > max_temp:
|
|
|
|
raise AlexaTempRangeError(hass, temp_low, min_temp, max_temp)
|
|
|
|
data[climate.ATTR_TARGET_TEMP_LOW] = temp_low
|
2019-07-31 19:25:30 +00:00
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"name": "lowerSetpoint",
|
|
|
|
"namespace": "Alexa.ThermostatController",
|
|
|
|
"value": {"value": temp_low, "scale": API_TEMP_UNITS[unit]},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
if "upperSetpoint" in payload:
|
|
|
|
temp_high = temperature_from_object(hass, payload["upperSetpoint"])
|
2019-06-13 15:43:57 +00:00
|
|
|
if temp_high < min_temp or temp_high > max_temp:
|
|
|
|
raise AlexaTempRangeError(hass, temp_high, min_temp, max_temp)
|
|
|
|
data[climate.ATTR_TARGET_TEMP_HIGH] = temp_high
|
2019-07-31 19:25:30 +00:00
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"name": "upperSetpoint",
|
|
|
|
"namespace": "Alexa.ThermostatController",
|
|
|
|
"value": {"value": temp_high, "scale": API_TEMP_UNITS[unit]},
|
|
|
|
}
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain,
|
|
|
|
climate.SERVICE_SET_TEMPERATURE,
|
|
|
|
data,
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.ThermostatController", "AdjustTargetTemperature"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_adjust_target_temp(hass, config, directive, context):
|
|
|
|
"""Process an adjust target temperature request."""
|
|
|
|
entity = directive.entity
|
|
|
|
min_temp = entity.attributes.get(climate.ATTR_MIN_TEMP)
|
|
|
|
max_temp = entity.attributes.get(climate.ATTR_MAX_TEMP)
|
|
|
|
unit = hass.config.units.temperature_unit
|
|
|
|
|
|
|
|
temp_delta = temperature_from_object(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, directive.payload["targetSetpointDelta"], interval=True
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
target_temp = float(entity.attributes.get(ATTR_TEMPERATURE)) + temp_delta
|
|
|
|
|
|
|
|
if target_temp < min_temp or target_temp > max_temp:
|
|
|
|
raise AlexaTempRangeError(hass, target_temp, min_temp, max_temp)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id, ATTR_TEMPERATURE: target_temp}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
response = directive.response()
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.domain,
|
|
|
|
climate.SERVICE_SET_TEMPERATURE,
|
|
|
|
data,
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"name": "targetSetpoint",
|
|
|
|
"namespace": "Alexa.ThermostatController",
|
|
|
|
"value": {"value": target_temp, "scale": API_TEMP_UNITS[unit]},
|
|
|
|
}
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa.ThermostatController", "SetThermostatMode"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_set_thermostat_mode(hass, config, directive, context):
|
|
|
|
"""Process a set thermostat mode request."""
|
|
|
|
entity = directive.entity
|
2019-07-31 19:25:30 +00:00
|
|
|
mode = directive.payload["thermostatMode"]
|
|
|
|
mode = mode if isinstance(mode, str) else mode["value"]
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-06-13 15:43:57 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ha_preset = next((k for k, v in API_THERMOSTAT_PRESETS.items() if v == mode), None)
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
|
|
|
|
if ha_preset:
|
|
|
|
presets = entity.attributes.get(climate.ATTR_PRESET_MODES, [])
|
|
|
|
|
|
|
|
if ha_preset not in presets:
|
2019-09-03 14:11:36 +00:00
|
|
|
msg = f"The requested thermostat mode {ha_preset} is not supported"
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
raise AlexaUnsupportedThermostatModeError(msg)
|
|
|
|
|
|
|
|
service = climate.SERVICE_SET_PRESET_MODE
|
2019-10-25 20:42:21 +00:00
|
|
|
data[climate.ATTR_PRESET_MODE] = ha_preset
|
|
|
|
|
|
|
|
elif mode == "CUSTOM":
|
|
|
|
operation_list = entity.attributes.get(climate.ATTR_HVAC_MODES)
|
|
|
|
custom_mode = directive.payload["thermostatMode"]["customName"]
|
|
|
|
custom_mode = next(
|
|
|
|
(k for k, v in API_THERMOSTAT_MODES_CUSTOM.items() if v == custom_mode),
|
|
|
|
None,
|
|
|
|
)
|
|
|
|
if custom_mode not in operation_list:
|
|
|
|
msg = (
|
|
|
|
f"The requested thermostat mode {mode}: {custom_mode} is not supported"
|
|
|
|
)
|
|
|
|
raise AlexaUnsupportedThermostatModeError(msg)
|
|
|
|
|
|
|
|
service = climate.SERVICE_SET_HVAC_MODE
|
|
|
|
data[climate.ATTR_HVAC_MODE] = custom_mode
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
|
|
|
|
else:
|
|
|
|
operation_list = entity.attributes.get(climate.ATTR_HVAC_MODES)
|
2019-10-25 20:42:21 +00:00
|
|
|
ha_modes = {k: v for k, v in API_THERMOSTAT_MODES.items() if v == mode}
|
|
|
|
ha_mode = next(iter(set(ha_modes).intersection(operation_list)), None)
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
if ha_mode not in operation_list:
|
2019-09-03 14:11:36 +00:00
|
|
|
msg = f"The requested thermostat mode {mode} is not supported"
|
Climate 1.0 (#23899)
* Climate 1.0 / part 1/2/3
* fix flake
* Lint
* Update Google Assistant
* ambiclimate to climate 1.0 (#24911)
* Fix Alexa
* Lint
* Migrate zhong_hong
* Migrate tuya
* Migrate honeywell to new climate schema (#24257)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* Fix PRESET can be None
* apply PR#23913 from dev
* remove EU component, etc.
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* apply PR#23913 from dev
* remove EU component, etc.
* ready to test now
* de-linted
* some tweaks
* de-lint
* better handling of edge cases
* delint
* fix set_mode typos
* delint, move debug code
* away preset now working
* code tidy-up
* code tidy-up 2
* code tidy-up 3
* address issues #18932, #15063
* address issues #18932, #15063 - 2/2
* refactor MODE_AUTO to MODE_HEAT_COOL and use F not C
* add low/high to set_temp
* add low/high to set_temp 2
* add low/high to set_temp - delint
* run HA scripts
* port changes from PR #24402
* manual rebase
* manual rebase 2
* delint
* minor change
* remove SUPPORT_HVAC_ACTION
* Migrate radiotherm
* Convert touchline
* Migrate flexit
* Migrate nuheat
* Migrate maxcube
* Fix names maxcube const
* Migrate proliphix
* Migrate heatmiser
* Migrate fritzbox
* Migrate opentherm_gw
* Migrate venstar
* Migrate daikin
* Migrate modbus
* Fix elif
* Migrate Homematic IP Cloud to climate-1.0 (#24913)
* hmip climate fix
* Update hvac_mode and preset_mode
* fix lint
* Fix lint
* Migrate generic_thermostat
* Migrate incomfort to new climate schema (#24915)
* initial commit
* Update climate.py
* Migrate eq3btsmart
* Lint
* cleanup PRESET_MANUAL
* Migrate ecobee
* No conditional features
* KNX: Migrate climate component to new climate platform (#24931)
* Migrate climate component
* Remove unused code
* Corrected line length
* Lint
* Lint
* fix tests
* Fix value
* Migrate geniushub to new climate schema (#24191)
* Update one
* Fix model climate v2
* Cleanup p4
* Add comfort hold mode
* Fix old code
* Update homeassistant/components/climate/__init__.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* Update homeassistant/components/climate/const.py
Co-Authored-By: Paulus Schoutsen <paulus@home-assistant.io>
* First renaming
* Rename operation to hvac for paulus
* Rename hold mode to preset mode
* Cleanup & update comments
* Remove on/off
* Fix supported feature count
* Update services
* Update demo
* Fix tests & use current_hvac
* Update comment
* Fix tests & add typing
* Add more typing
* Update modes
* Fix tests
* Cleanup low/high with range
* Update homematic part 1
* Finish homematic
* Fix lint
* fix hm mapping
* Support simple devices
* convert lcn
* migrate oem
* Fix xs1
* update hive
* update mil
* Update toon
* migrate deconz
* cleanup
* update tesla
* Fix lint
* Fix vera
* Migrate zwave
* Migrate velbus
* Cleanup humity feature
* Cleanup
* Migrate wink
* migrate dyson
* Fix current hvac
* Renaming
* Fix lint
* Migrate tfiac
* migrate tado
* delinted
* delinted
* use latest client
* clean up mappings
* clean up mappings
* add duration to set_temperature
* add duration to set_temperature
* manual rebase
* tweak
* fix regression
* small fix
* fix rebase mixup
* address comments
* finish refactor
* fix regression
* tweak type hints
* delint
* manual rebase
* WIP: Fixes for honeywell migration to climate-1.0 (#24938)
* add type hints
* code tidy-up
* Fixes for incomfort migration to climate-1.0 (#24936)
* delint type hints
* no async unless await
* revert: no async unless await
* revert: no async unless await 2
* delint
* fix typo
* Fix homekit_controller on climate-1.0 (#24948)
* Fix tests on climate-1.0 branch
* As part of climate-1.0, make state return the heating-cooling.current characteristic
* Fixes from review
* lint
* Fix imports
* Migrate stibel_eltron
* Fix lint
* Migrate coolmaster to climate 1.0 (#24967)
* Migrate coolmaster to climate 1.0
* fix lint errors
* More lint fixes
* Fix demo to work with UI
* Migrate spider
* Demo update
* Updated frontend to 20190705.0
* Fix boost mode (#24980)
* Prepare Netatmo for climate 1.0 (#24973)
* Migration Netatmo
* Address comments
* Update climate.py
* Migrate ephember
* Migrate Sensibo
* Implemented review comments (#24942)
* Migrate ESPHome
* Migrate MQTT
* Migrate Nest
* Migrate melissa
* Initial/partial migration of ST
* Migrate ST
* Remove Away mode (#24995)
* Migrate evohome, cache access tokens (#24491)
* add water_heater, add storage - initial commit
* add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
* Add Broker, Water Heater & Refactor
add missing code
desiderata
* update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
* bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
* support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
* store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
add water_heater, add storage - initial commit
delint
add missing code
desiderata
update honeywell client library & CODEOWNER
add auth_tokens code, refactor & delint
refactor for broker
delint
bugfix - loc_idx may not be 0
more refactor - ensure pure async
more refactoring
appears all r/o attributes are working
tweak precsion, DHW & delint
remove unused code
remove unused code 2
remove unused code, refactor _save_auth_tokens()
support RoundThermostat
bugfix opmode, switch to util.dt, add until=1h
revert breaking change
store at_expires as naive UTC
remove debug code
delint
tidy up exception handling
delint
* update CODEOWNERS
* fix regression
* fix requirements
* migrate to climate-1.0
* tweaking
* de-lint
* TCS working? & delint
* tweaking
* TCS code finalised
* remove available() logic
* refactor _switchpoints()
* tidy up switchpoint code
* tweak
* teaking device_state_attributes
* some refactoring
* move PRESET_CUSTOM back to evohome
* move CONF_ACCESS_TOKEN_EXPIRES CONF_REFRESH_TOKEN back to evohome
* refactor SP code and dt conversion
* delinted
* delinted
* remove water_heater
* fix regression
* Migrate homekit
* Cleanup away mode
* Fix tests
* add helpers
* fix tests melissa
* Fix nehueat
* fix zwave
* add more tests
* fix deconz
* Fix climate test emulate_hue
* fix tests
* fix dyson tests
* fix demo with new layout
* fix honeywell
* Switch homekit_controller to use HVAC_MODE_HEAT_COOL instead of HVAC_MODE_AUTO (#25009)
* Lint
* PyLint
* Pylint
* fix fritzbox tests
* Fix google
* Fix all tests
* Fix lint
* Fix auto for homekit like controler
* Fix lint
* fix lint
2019-07-08 12:00:24 +00:00
|
|
|
raise AlexaUnsupportedThermostatModeError(msg)
|
|
|
|
|
|
|
|
service = climate.SERVICE_SET_HVAC_MODE
|
|
|
|
data[climate.ATTR_HVAC_MODE] = ha_mode
|
|
|
|
|
2019-06-13 15:43:57 +00:00
|
|
|
response = directive.response()
|
|
|
|
await hass.services.async_call(
|
2019-07-31 19:25:30 +00:00
|
|
|
climate.DOMAIN, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"name": "thermostatMode",
|
|
|
|
"namespace": "Alexa.ThermostatController",
|
|
|
|
"value": mode,
|
|
|
|
}
|
|
|
|
)
|
2019-06-13 15:43:57 +00:00
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
@HANDLERS.register(("Alexa", "ReportState"))
|
2019-06-13 15:43:57 +00:00
|
|
|
async def async_api_reportstate(hass, config, directive, context):
|
|
|
|
"""Process a ReportState request."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return directive.response(name="StateReport")
|
2019-10-03 20:28:02 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.PowerLevelController", "SetPowerLevel"))
|
|
|
|
async def async_api_set_power_level(hass, config, directive, context):
|
|
|
|
"""Process a SetPowerLevel request."""
|
|
|
|
entity = directive.entity
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
|
|
|
if entity.domain == fan.DOMAIN:
|
|
|
|
service = fan.SERVICE_SET_SPEED
|
|
|
|
speed = "off"
|
|
|
|
|
2020-04-04 21:20:48 +00:00
|
|
|
percentage = int(directive.payload["powerLevel"])
|
2019-10-03 20:28:02 +00:00
|
|
|
if percentage <= 33:
|
|
|
|
speed = "low"
|
|
|
|
elif percentage <= 66:
|
|
|
|
speed = "medium"
|
|
|
|
else:
|
|
|
|
speed = "high"
|
|
|
|
|
|
|
|
data[fan.ATTR_SPEED] = speed
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.PowerLevelController", "AdjustPowerLevel"))
|
|
|
|
async def async_api_adjust_power_level(hass, config, directive, context):
|
|
|
|
"""Process an AdjustPowerLevel request."""
|
|
|
|
entity = directive.entity
|
|
|
|
percentage_delta = int(directive.payload["powerLevelDelta"])
|
|
|
|
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)
|
2019-10-23 05:01:03 +00:00
|
|
|
current = PERCENTAGE_FAN_MAP.get(speed, 100)
|
2019-10-03 20:28:02 +00:00
|
|
|
|
|
|
|
# set percentage
|
|
|
|
percentage = max(0, percentage_delta + current)
|
|
|
|
speed = "off"
|
|
|
|
|
|
|
|
if percentage <= 33:
|
|
|
|
speed = "low"
|
|
|
|
elif percentage <= 66:
|
|
|
|
speed = "medium"
|
|
|
|
else:
|
|
|
|
speed = "high"
|
|
|
|
|
|
|
|
data[fan.ATTR_SPEED] = speed
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
|
|
|
return directive.response()
|
2019-10-04 15:41:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.SecurityPanelController", "Arm"))
|
|
|
|
async def async_api_arm(hass, config, directive, context):
|
|
|
|
"""Process a Security Panel Arm request."""
|
|
|
|
entity = directive.entity
|
|
|
|
service = None
|
|
|
|
arm_state = directive.payload["armState"]
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
|
|
|
if entity.state != STATE_ALARM_DISARMED:
|
|
|
|
msg = "You must disarm the system before you can set the requested arm state."
|
|
|
|
raise AlexaSecurityPanelAuthorizationRequired(msg)
|
|
|
|
|
|
|
|
if arm_state == "ARMED_AWAY":
|
|
|
|
service = SERVICE_ALARM_ARM_AWAY
|
2020-04-04 21:20:48 +00:00
|
|
|
elif arm_state == "ARMED_NIGHT":
|
2019-10-04 15:41:47 +00:00
|
|
|
service = SERVICE_ALARM_ARM_NIGHT
|
2020-04-04 21:20:48 +00:00
|
|
|
elif arm_state == "ARMED_STAY":
|
|
|
|
service = SERVICE_ALARM_ARM_HOME
|
2019-10-04 15:41:47 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
2020-01-25 01:57:58 +00:00
|
|
|
# return 0 until alarm integration supports an exit delay
|
|
|
|
payload = {"exitDelayInSeconds": 0}
|
|
|
|
|
2019-10-04 15:41:47 +00:00
|
|
|
response = directive.response(
|
2020-01-25 01:57:58 +00:00
|
|
|
name="Arm.Response", namespace="Alexa.SecurityPanelController", payload=payload
|
2019-10-04 15:41:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"name": "armState",
|
|
|
|
"namespace": "Alexa.SecurityPanelController",
|
|
|
|
"value": arm_state,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.SecurityPanelController", "Disarm"))
|
|
|
|
async def async_api_disarm(hass, config, directive, context):
|
|
|
|
"""Process a Security Panel Disarm request."""
|
|
|
|
entity = directive.entity
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2020-01-25 01:57:58 +00:00
|
|
|
response = directive.response()
|
|
|
|
|
|
|
|
# Per Alexa Documentation: If you receive a Disarm directive, and the system is already disarmed,
|
|
|
|
# respond with a success response, not an error response.
|
|
|
|
if entity.state == STATE_ALARM_DISARMED:
|
|
|
|
return response
|
2019-10-04 15:41:47 +00:00
|
|
|
|
|
|
|
payload = directive.payload
|
|
|
|
if "authorization" in payload:
|
|
|
|
value = payload["authorization"]["value"]
|
|
|
|
if payload["authorization"]["type"] == "FOUR_DIGIT_PIN":
|
|
|
|
data["code"] = value
|
|
|
|
|
|
|
|
if not await hass.services.async_call(
|
|
|
|
entity.domain, SERVICE_ALARM_DISARM, data, blocking=True, context=context
|
|
|
|
):
|
|
|
|
msg = "Invalid Code"
|
|
|
|
raise AlexaSecurityPanelUnauthorizedError(msg)
|
|
|
|
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"name": "armState",
|
|
|
|
"namespace": "Alexa.SecurityPanelController",
|
|
|
|
"value": "DISARMED",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.ModeController", "SetMode"))
|
|
|
|
async def async_api_set_mode(hass, config, directive, context):
|
2019-12-19 11:44:17 +00:00
|
|
|
"""Process a SetMode directive."""
|
2019-10-23 05:01:03 +00:00
|
|
|
entity = directive.entity
|
|
|
|
instance = directive.instance
|
|
|
|
domain = entity.domain
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-12-19 11:44:17 +00:00
|
|
|
mode = directive.payload["mode"]
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Direction
|
2019-10-23 05:01:03 +00:00
|
|
|
if instance == f"{fan.DOMAIN}.{fan.ATTR_DIRECTION}":
|
2019-12-19 11:44:17 +00:00
|
|
|
_, direction = mode.split(".")
|
2019-11-25 23:07:33 +00:00
|
|
|
if direction in (fan.DIRECTION_REVERSE, fan.DIRECTION_FORWARD):
|
2019-10-23 05:01:03 +00:00
|
|
|
service = fan.SERVICE_SET_DIRECTION
|
|
|
|
data[fan.ATTR_DIRECTION] = direction
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position
|
|
|
|
elif instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
|
|
|
_, position = mode.split(".")
|
2019-11-25 23:07:33 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
if position == cover.STATE_CLOSED:
|
2019-11-25 23:07:33 +00:00
|
|
|
service = cover.SERVICE_CLOSE_COVER
|
2019-12-19 11:44:17 +00:00
|
|
|
elif position == cover.STATE_OPEN:
|
2019-11-25 23:07:33 +00:00
|
|
|
service = cover.SERVICE_OPEN_COVER
|
2019-12-19 11:44:17 +00:00
|
|
|
elif position == "custom":
|
|
|
|
service = cover.SERVICE_STOP_COVER
|
|
|
|
|
|
|
|
else:
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
2019-11-25 23:07:33 +00:00
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
2019-11-25 23:07:33 +00:00
|
|
|
response = directive.response()
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"namespace": "Alexa.ModeController",
|
|
|
|
"instance": instance,
|
|
|
|
"name": "mode",
|
2019-12-19 11:44:17 +00:00
|
|
|
"value": mode,
|
2019-11-25 23:07:33 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.ModeController", "AdjustMode"))
|
|
|
|
async def async_api_adjust_mode(hass, config, directive, context):
|
|
|
|
"""Process a AdjustMode request.
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
Requires capabilityResources supportedModes to be ordered.
|
|
|
|
Only supportedModes with ordered=True support the adjustMode directive.
|
2019-10-23 05:01:03 +00:00
|
|
|
"""
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Currently no supportedModes are configured with ordered=True to support this request.
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.ToggleController", "TurnOn"))
|
|
|
|
async def async_api_toggle_on(hass, config, directive, context):
|
|
|
|
"""Process a toggle on request."""
|
|
|
|
entity = directive.entity
|
|
|
|
instance = directive.instance
|
|
|
|
domain = entity.domain
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Oscillating
|
2019-10-23 05:01:03 +00:00
|
|
|
if instance == f"{fan.DOMAIN}.{fan.ATTR_OSCILLATING}":
|
|
|
|
service = fan.SERVICE_OSCILLATE
|
|
|
|
data[fan.ATTR_OSCILLATING] = True
|
2019-12-19 11:44:17 +00:00
|
|
|
else:
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
response = directive.response()
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"namespace": "Alexa.ToggleController",
|
|
|
|
"instance": instance,
|
|
|
|
"name": "toggleState",
|
|
|
|
"value": "ON",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.ToggleController", "TurnOff"))
|
|
|
|
async def async_api_toggle_off(hass, config, directive, context):
|
|
|
|
"""Process a toggle off request."""
|
|
|
|
entity = directive.entity
|
|
|
|
instance = directive.instance
|
|
|
|
domain = entity.domain
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Oscillating
|
2019-10-23 05:01:03 +00:00
|
|
|
if instance == f"{fan.DOMAIN}.{fan.ATTR_OSCILLATING}":
|
|
|
|
service = fan.SERVICE_OSCILLATE
|
|
|
|
data[fan.ATTR_OSCILLATING] = False
|
2019-12-19 11:44:17 +00:00
|
|
|
else:
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
response = directive.response()
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"namespace": "Alexa.ToggleController",
|
|
|
|
"instance": instance,
|
|
|
|
"name": "toggleState",
|
|
|
|
"value": "OFF",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.RangeController", "SetRangeValue"))
|
|
|
|
async def async_api_set_range(hass, config, directive, context):
|
|
|
|
"""Process a next request."""
|
|
|
|
entity = directive.entity
|
|
|
|
instance = directive.instance
|
|
|
|
domain = entity.domain
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-12-23 14:17:37 +00:00
|
|
|
range_value = directive.payload["rangeValue"]
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Speed
|
2019-10-23 05:01:03 +00:00
|
|
|
if instance == f"{fan.DOMAIN}.{fan.ATTR_SPEED}":
|
2020-01-11 01:26:37 +00:00
|
|
|
range_value = int(range_value)
|
2019-10-23 05:01:03 +00:00
|
|
|
service = fan.SERVICE_SET_SPEED
|
2020-01-11 01:26:37 +00:00
|
|
|
speed_list = entity.attributes[fan.ATTR_SPEED_LIST]
|
|
|
|
speed = next((v for i, v in enumerate(speed_list) if i == range_value), None)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
if not speed:
|
|
|
|
msg = "Entity does not support value"
|
|
|
|
raise AlexaInvalidValueError(msg)
|
|
|
|
|
|
|
|
if speed == fan.SPEED_OFF:
|
|
|
|
service = fan.SERVICE_TURN_OFF
|
|
|
|
|
|
|
|
data[fan.ATTR_SPEED] = speed
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Cover Position
|
|
|
|
elif instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
2019-12-23 14:17:37 +00:00
|
|
|
range_value = int(range_value)
|
2019-12-19 11:44:17 +00:00
|
|
|
if range_value == 0:
|
|
|
|
service = cover.SERVICE_CLOSE_COVER
|
|
|
|
elif range_value == 100:
|
|
|
|
service = cover.SERVICE_OPEN_COVER
|
|
|
|
else:
|
|
|
|
service = cover.SERVICE_SET_COVER_POSITION
|
|
|
|
data[cover.ATTR_POSITION] = range_value
|
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
# Cover Tilt
|
|
|
|
elif instance == f"{cover.DOMAIN}.tilt":
|
2019-12-23 14:17:37 +00:00
|
|
|
range_value = int(range_value)
|
2019-12-19 11:44:17 +00:00
|
|
|
if range_value == 0:
|
|
|
|
service = cover.SERVICE_CLOSE_COVER_TILT
|
|
|
|
elif range_value == 100:
|
|
|
|
service = cover.SERVICE_OPEN_COVER_TILT
|
|
|
|
else:
|
|
|
|
service = cover.SERVICE_SET_COVER_TILT_POSITION
|
2020-01-28 00:28:40 +00:00
|
|
|
data[cover.ATTR_TILT_POSITION] = range_value
|
2019-12-19 11:44:17 +00:00
|
|
|
|
2019-12-23 14:17:37 +00:00
|
|
|
# Input Number Value
|
|
|
|
elif instance == f"{input_number.DOMAIN}.{input_number.ATTR_VALUE}":
|
|
|
|
range_value = float(range_value)
|
|
|
|
service = input_number.SERVICE_SET_VALUE
|
|
|
|
min_value = float(entity.attributes[input_number.ATTR_MIN])
|
|
|
|
max_value = float(entity.attributes[input_number.ATTR_MAX])
|
|
|
|
data[input_number.ATTR_VALUE] = min(max_value, max(min_value, range_value))
|
|
|
|
|
2020-01-15 17:15:31 +00:00
|
|
|
# Vacuum Fan Speed
|
|
|
|
elif instance == f"{vacuum.DOMAIN}.{vacuum.ATTR_FAN_SPEED}":
|
|
|
|
service = vacuum.SERVICE_SET_FAN_SPEED
|
|
|
|
speed_list = entity.attributes[vacuum.ATTR_FAN_SPEED_LIST]
|
|
|
|
speed = next(
|
|
|
|
(v for i, v in enumerate(speed_list) if i == int(range_value)), None
|
|
|
|
)
|
|
|
|
|
|
|
|
if not speed:
|
|
|
|
msg = "Entity does not support value"
|
|
|
|
raise AlexaInvalidValueError(msg)
|
|
|
|
|
|
|
|
data[vacuum.ATTR_FAN_SPEED] = speed
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
else:
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
|
|
|
|
2019-10-23 05:01:03 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
response = directive.response()
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"namespace": "Alexa.RangeController",
|
|
|
|
"instance": instance,
|
|
|
|
"name": "rangeValue",
|
|
|
|
"value": range_value,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.RangeController", "AdjustRangeValue"))
|
|
|
|
async def async_api_adjust_range(hass, config, directive, context):
|
|
|
|
"""Process a next request."""
|
|
|
|
entity = directive.entity
|
|
|
|
instance = directive.instance
|
|
|
|
domain = entity.domain
|
|
|
|
service = None
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
2019-12-23 14:17:37 +00:00
|
|
|
range_delta = directive.payload["rangeValueDelta"]
|
2020-01-29 17:04:57 +00:00
|
|
|
range_delta_default = bool(directive.payload["rangeValueDeltaDefault"])
|
2019-12-19 11:44:17 +00:00
|
|
|
response_value = 0
|
2019-10-23 05:01:03 +00:00
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
# Fan Speed
|
2019-10-23 05:01:03 +00:00
|
|
|
if instance == f"{fan.DOMAIN}.{fan.ATTR_SPEED}":
|
2019-12-23 14:17:37 +00:00
|
|
|
range_delta = int(range_delta)
|
2019-10-23 05:01:03 +00:00
|
|
|
service = fan.SERVICE_SET_SPEED
|
2020-01-11 01:26:37 +00:00
|
|
|
speed_list = entity.attributes[fan.ATTR_SPEED_LIST]
|
|
|
|
current_speed = entity.attributes[fan.ATTR_SPEED]
|
|
|
|
current_speed_index = next(
|
|
|
|
(i for i, v in enumerate(speed_list) if v == current_speed), 0
|
|
|
|
)
|
|
|
|
new_speed_index = min(
|
|
|
|
len(speed_list) - 1, max(0, current_speed_index + range_delta)
|
|
|
|
)
|
|
|
|
speed = next(
|
|
|
|
(v for i, v in enumerate(speed_list) if i == new_speed_index), None
|
2019-12-19 11:44:17 +00:00
|
|
|
)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
if speed == fan.SPEED_OFF:
|
|
|
|
service = fan.SERVICE_TURN_OFF
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
data[fan.ATTR_SPEED] = response_value = speed
|
|
|
|
|
|
|
|
# Cover Position
|
|
|
|
elif instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}":
|
2020-01-29 17:04:57 +00:00
|
|
|
range_delta = int(range_delta * 20) if range_delta_default else int(range_delta)
|
2019-12-19 11:44:17 +00:00
|
|
|
service = SERVICE_SET_COVER_POSITION
|
|
|
|
current = entity.attributes.get(cover.ATTR_POSITION)
|
2020-01-29 17:04:57 +00:00
|
|
|
if not current:
|
2020-02-23 21:38:05 +00:00
|
|
|
msg = f"Unable to determine {entity.entity_id} current position"
|
2020-01-29 17:04:57 +00:00
|
|
|
raise AlexaInvalidValueError(msg)
|
2020-01-22 19:04:31 +00:00
|
|
|
position = response_value = min(100, max(0, range_delta + current))
|
|
|
|
if position == 100:
|
|
|
|
service = cover.SERVICE_OPEN_COVER
|
|
|
|
elif position == 0:
|
|
|
|
service = cover.SERVICE_CLOSE_COVER
|
|
|
|
else:
|
|
|
|
data[cover.ATTR_POSITION] = position
|
2019-12-19 11:44:17 +00:00
|
|
|
|
2020-01-17 23:04:46 +00:00
|
|
|
# Cover Tilt
|
|
|
|
elif instance == f"{cover.DOMAIN}.tilt":
|
2020-01-29 17:04:57 +00:00
|
|
|
range_delta = int(range_delta * 20) if range_delta_default else int(range_delta)
|
2019-12-19 11:44:17 +00:00
|
|
|
service = SERVICE_SET_COVER_TILT_POSITION
|
|
|
|
current = entity.attributes.get(cover.ATTR_TILT_POSITION)
|
2020-01-29 17:04:57 +00:00
|
|
|
if not current:
|
2020-02-23 21:38:05 +00:00
|
|
|
msg = f"Unable to determine {entity.entity_id} current tilt position"
|
2020-01-29 17:04:57 +00:00
|
|
|
raise AlexaInvalidValueError(msg)
|
2020-01-22 19:04:31 +00:00
|
|
|
tilt_position = response_value = min(100, max(0, range_delta + current))
|
|
|
|
if tilt_position == 100:
|
|
|
|
service = cover.SERVICE_OPEN_COVER_TILT
|
|
|
|
elif tilt_position == 0:
|
|
|
|
service = cover.SERVICE_CLOSE_COVER_TILT
|
|
|
|
else:
|
|
|
|
data[cover.ATTR_TILT_POSITION] = tilt_position
|
2019-12-19 11:44:17 +00:00
|
|
|
|
2019-12-23 14:17:37 +00:00
|
|
|
# Input Number Value
|
|
|
|
elif instance == f"{input_number.DOMAIN}.{input_number.ATTR_VALUE}":
|
|
|
|
range_delta = float(range_delta)
|
|
|
|
service = input_number.SERVICE_SET_VALUE
|
|
|
|
min_value = float(entity.attributes[input_number.ATTR_MIN])
|
|
|
|
max_value = float(entity.attributes[input_number.ATTR_MAX])
|
|
|
|
current = float(entity.state)
|
|
|
|
data[input_number.ATTR_VALUE] = response_value = min(
|
|
|
|
max_value, max(min_value, range_delta + current)
|
|
|
|
)
|
|
|
|
|
2020-01-15 17:15:31 +00:00
|
|
|
# Vacuum Fan Speed
|
|
|
|
elif instance == f"{vacuum.DOMAIN}.{vacuum.ATTR_FAN_SPEED}":
|
|
|
|
range_delta = int(range_delta)
|
|
|
|
service = vacuum.SERVICE_SET_FAN_SPEED
|
|
|
|
speed_list = entity.attributes[vacuum.ATTR_FAN_SPEED_LIST]
|
|
|
|
current_speed = entity.attributes[vacuum.ATTR_FAN_SPEED]
|
|
|
|
current_speed_index = next(
|
|
|
|
(i for i, v in enumerate(speed_list) if v == current_speed), 0
|
|
|
|
)
|
|
|
|
new_speed_index = min(
|
|
|
|
len(speed_list) - 1, max(0, current_speed_index + range_delta)
|
|
|
|
)
|
|
|
|
speed = next(
|
|
|
|
(v for i, v in enumerate(speed_list) if i == new_speed_index), None
|
|
|
|
)
|
|
|
|
|
|
|
|
data[vacuum.ATTR_FAN_SPEED] = response_value = speed
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
else:
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
2019-10-23 05:01:03 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
domain, service, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
2019-12-19 11:44:17 +00:00
|
|
|
response = directive.response()
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"namespace": "Alexa.RangeController",
|
|
|
|
"instance": instance,
|
|
|
|
"name": "rangeValue",
|
|
|
|
"value": response_value,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-10-23 15:28:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.ChannelController", "ChangeChannel"))
|
|
|
|
async def async_api_changechannel(hass, config, directive, context):
|
|
|
|
"""Process a change channel request."""
|
|
|
|
channel = "0"
|
|
|
|
entity = directive.entity
|
2019-12-03 00:10:44 +00:00
|
|
|
channel_payload = directive.payload["channel"]
|
|
|
|
metadata_payload = directive.payload["channelMetadata"]
|
2019-10-23 15:28:23 +00:00
|
|
|
payload_name = "number"
|
|
|
|
|
2019-12-03 00:10:44 +00:00
|
|
|
if "number" in channel_payload:
|
|
|
|
channel = channel_payload["number"]
|
2019-10-23 15:28:23 +00:00
|
|
|
payload_name = "number"
|
2019-12-03 00:10:44 +00:00
|
|
|
elif "callSign" in channel_payload:
|
|
|
|
channel = channel_payload["callSign"]
|
2019-10-23 15:28:23 +00:00
|
|
|
payload_name = "callSign"
|
2019-12-03 00:10:44 +00:00
|
|
|
elif "affiliateCallSign" in channel_payload:
|
|
|
|
channel = channel_payload["affiliateCallSign"]
|
2019-10-23 15:28:23 +00:00
|
|
|
payload_name = "affiliateCallSign"
|
2019-12-03 00:10:44 +00:00
|
|
|
elif "uri" in channel_payload:
|
|
|
|
channel = channel_payload["uri"]
|
2019-10-23 15:28:23 +00:00
|
|
|
payload_name = "uri"
|
2019-12-03 00:10:44 +00:00
|
|
|
elif "name" in metadata_payload:
|
|
|
|
channel = metadata_payload["name"]
|
|
|
|
payload_name = "callSign"
|
2019-10-23 15:28:23 +00:00
|
|
|
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.const.ATTR_MEDIA_CONTENT_ID: channel,
|
|
|
|
media_player.const.ATTR_MEDIA_CONTENT_TYPE: media_player.const.MEDIA_TYPE_CHANNEL,
|
|
|
|
}
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
media_player.const.SERVICE_PLAY_MEDIA,
|
|
|
|
data,
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
|
|
|
|
|
|
|
response = directive.response()
|
|
|
|
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"namespace": "Alexa.ChannelController",
|
|
|
|
"name": "channel",
|
|
|
|
"value": {payload_name: channel},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.ChannelController", "SkipChannels"))
|
|
|
|
async def async_api_skipchannel(hass, config, directive, context):
|
|
|
|
"""Process a skipchannel request."""
|
|
|
|
channel = int(directive.payload["channelCount"])
|
|
|
|
entity = directive.entity
|
|
|
|
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
|
|
|
if channel < 0:
|
|
|
|
service_media = SERVICE_MEDIA_PREVIOUS_TRACK
|
|
|
|
else:
|
|
|
|
service_media = SERVICE_MEDIA_NEXT_TRACK
|
|
|
|
|
2020-04-04 21:20:48 +00:00
|
|
|
for _ in range(abs(channel)):
|
2019-10-23 15:28:23 +00:00
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain, service_media, data, blocking=False, context=context
|
|
|
|
)
|
|
|
|
|
|
|
|
response = directive.response()
|
|
|
|
|
|
|
|
response.add_context_property(
|
|
|
|
{
|
|
|
|
"namespace": "Alexa.ChannelController",
|
|
|
|
"name": "channel",
|
|
|
|
"value": {"number": ""},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
return response
|
2019-10-31 09:38:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.SeekController", "AdjustSeekPosition"))
|
|
|
|
async def async_api_seek(hass, config, directive, context):
|
|
|
|
"""Process a seek request."""
|
|
|
|
entity = directive.entity
|
|
|
|
position_delta = int(directive.payload["deltaPositionMilliseconds"])
|
|
|
|
|
|
|
|
current_position = entity.attributes.get(media_player.ATTR_MEDIA_POSITION)
|
|
|
|
if not current_position:
|
|
|
|
msg = f"{entity} did not return the current media position."
|
|
|
|
raise AlexaVideoActionNotPermittedForContentError(msg)
|
|
|
|
|
|
|
|
seek_position = int(current_position) + int(position_delta / 1000)
|
|
|
|
|
|
|
|
if seek_position < 0:
|
|
|
|
seek_position = 0
|
|
|
|
|
|
|
|
media_duration = entity.attributes.get(media_player.ATTR_MEDIA_DURATION)
|
|
|
|
if media_duration and 0 < int(media_duration) < seek_position:
|
|
|
|
seek_position = media_duration
|
|
|
|
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity.entity_id,
|
|
|
|
media_player.ATTR_MEDIA_SEEK_POSITION: seek_position,
|
|
|
|
}
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
media_player.DOMAIN,
|
|
|
|
media_player.SERVICE_MEDIA_SEEK,
|
|
|
|
data,
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
|
|
|
|
|
|
|
# convert seconds to milliseconds for StateReport.
|
|
|
|
seek_position = int(seek_position * 1000)
|
|
|
|
|
|
|
|
payload = {"properties": [{"name": "positionMilliseconds", "value": seek_position}]}
|
|
|
|
return directive.response(
|
|
|
|
name="StateReport", namespace="Alexa.SeekController", payload=payload
|
|
|
|
)
|
2019-12-24 22:06:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.EqualizerController", "SetMode"))
|
|
|
|
async def async_api_set_eq_mode(hass, config, directive, context):
|
|
|
|
"""Process a SetMode request for EqualizerController."""
|
|
|
|
mode = directive.payload["mode"]
|
|
|
|
entity = directive.entity
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
|
|
|
sound_mode_list = entity.attributes.get(media_player.const.ATTR_SOUND_MODE_LIST)
|
|
|
|
if sound_mode_list and mode.lower() in sound_mode_list:
|
|
|
|
data[media_player.const.ATTR_SOUND_MODE] = mode.lower()
|
|
|
|
else:
|
2020-02-23 21:38:05 +00:00
|
|
|
msg = f"failed to map sound mode {mode} to a mode on {entity.entity_id}"
|
2019-12-24 22:06:39 +00:00
|
|
|
raise AlexaInvalidValueError(msg)
|
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
entity.domain,
|
|
|
|
media_player.SERVICE_SELECT_SOUND_MODE,
|
|
|
|
data,
|
|
|
|
blocking=False,
|
|
|
|
context=context,
|
|
|
|
)
|
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.EqualizerController", "AdjustBands"))
|
|
|
|
@HANDLERS.register(("Alexa.EqualizerController", "ResetBands"))
|
|
|
|
@HANDLERS.register(("Alexa.EqualizerController", "SetBands"))
|
|
|
|
async def async_api_bands_directive(hass, config, directive, context):
|
|
|
|
"""Handle an AdjustBands, ResetBands, SetBands request.
|
|
|
|
|
|
|
|
Only mode directives are currently supported for the EqualizerController.
|
|
|
|
"""
|
|
|
|
# Currently bands directives are not supported.
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
2020-01-10 22:11:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.TimeHoldController", "Hold"))
|
|
|
|
async def async_api_hold(hass, config, directive, context):
|
|
|
|
"""Process a TimeHoldController Hold request."""
|
|
|
|
entity = directive.entity
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
2020-01-15 17:15:31 +00:00
|
|
|
if entity.domain == timer.DOMAIN:
|
|
|
|
service = timer.SERVICE_PAUSE
|
|
|
|
|
|
|
|
elif entity.domain == vacuum.DOMAIN:
|
|
|
|
service = vacuum.SERVICE_START_PAUSE
|
|
|
|
|
|
|
|
else:
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
|
|
|
|
2020-01-10 22:11:50 +00:00
|
|
|
await hass.services.async_call(
|
2020-01-15 17:15:31 +00:00
|
|
|
entity.domain, service, data, blocking=False, context=context
|
2020-01-10 22:11:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
return directive.response()
|
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.TimeHoldController", "Resume"))
|
|
|
|
async def async_api_resume(hass, config, directive, context):
|
|
|
|
"""Process a TimeHoldController Resume request."""
|
|
|
|
entity = directive.entity
|
|
|
|
data = {ATTR_ENTITY_ID: entity.entity_id}
|
|
|
|
|
2020-01-15 17:15:31 +00:00
|
|
|
if entity.domain == timer.DOMAIN:
|
|
|
|
service = timer.SERVICE_START
|
|
|
|
|
|
|
|
elif entity.domain == vacuum.DOMAIN:
|
|
|
|
service = vacuum.SERVICE_START_PAUSE
|
|
|
|
|
|
|
|
else:
|
|
|
|
msg = "Entity does not support directive"
|
|
|
|
raise AlexaInvalidDirectiveError(msg)
|
|
|
|
|
2020-01-10 22:11:50 +00:00
|
|
|
await hass.services.async_call(
|
2020-01-15 17:15:31 +00:00
|
|
|
entity.domain, service, data, blocking=False, context=context
|
2020-01-10 22:11:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
return directive.response()
|
2020-03-28 04:19:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
@HANDLERS.register(("Alexa.CameraStreamController", "InitializeCameraStreams"))
|
|
|
|
async def async_api_initialize_camera_stream(hass, config, directive, context):
|
|
|
|
"""Process a InitializeCameraStreams request."""
|
|
|
|
entity = directive.entity
|
|
|
|
stream_source = await camera.async_request_stream(hass, entity.entity_id, fmt="hls")
|
|
|
|
camera_image = hass.states.get(entity.entity_id).attributes["entity_picture"]
|
|
|
|
external_url = network.async_get_external_url(hass)
|
|
|
|
payload = {
|
|
|
|
"cameraStreams": [
|
|
|
|
{
|
|
|
|
"uri": f"{external_url}{stream_source}",
|
|
|
|
"protocol": "HLS",
|
|
|
|
"resolution": {"width": 1280, "height": 720},
|
|
|
|
"authorizationType": "NONE",
|
|
|
|
"videoCodec": "H264",
|
|
|
|
"audioCodec": "AAC",
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"imageUri": f"{external_url}{camera_image}",
|
|
|
|
}
|
|
|
|
return directive.response(
|
|
|
|
name="Response", namespace="Alexa.CameraStreamController", payload=payload
|
|
|
|
)
|