2019-12-01 22:12:57 +00:00
|
|
|
"""Intents for the light integration."""
|
2022-12-13 22:46:40 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2023-01-19 23:15:01 +00:00
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
from typing import Any
|
|
|
|
|
2019-12-01 22:12:57 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2022-01-05 17:21:20 +00:00
|
|
|
from homeassistant.const import ATTR_ENTITY_ID, SERVICE_TURN_ON
|
2023-01-19 23:15:01 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2023-03-01 02:58:19 +00:00
|
|
|
from homeassistant.helpers import area_registry as ar, config_validation as cv, intent
|
2019-12-08 17:16:23 +00:00
|
|
|
import homeassistant.util.color as color_util
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
from . import (
|
2019-12-08 17:16:23 +00:00
|
|
|
ATTR_BRIGHTNESS_PCT,
|
2019-12-01 22:12:57 +00:00
|
|
|
ATTR_RGB_COLOR,
|
2021-05-14 21:23:29 +00:00
|
|
|
ATTR_SUPPORTED_COLOR_MODES,
|
2019-12-01 22:12:57 +00:00
|
|
|
DOMAIN,
|
2021-05-14 21:23:29 +00:00
|
|
|
brightness_supported,
|
|
|
|
color_supported,
|
2019-12-01 22:12:57 +00:00
|
|
|
)
|
|
|
|
|
2023-01-19 23:15:01 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-12-01 22:12:57 +00:00
|
|
|
INTENT_SET = "HassLightSet"
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_intents(hass: HomeAssistant) -> None:
|
|
|
|
"""Set up the light intents."""
|
2022-05-17 18:33:51 +00:00
|
|
|
intent.async_register(hass, SetIntentHandler())
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
class SetIntentHandler(intent.IntentHandler):
|
|
|
|
"""Handle set color intents."""
|
|
|
|
|
|
|
|
intent_type = INTENT_SET
|
|
|
|
slot_schema = {
|
2023-01-19 23:15:01 +00:00
|
|
|
vol.Any("name", "area"): cv.string,
|
|
|
|
vol.Optional("domain"): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
vol.Optional("device_class"): vol.All(cv.ensure_list, [cv.string]),
|
2019-12-01 22:12:57 +00:00
|
|
|
vol.Optional("color"): color_util.color_name_to_rgb,
|
|
|
|
vol.Optional("brightness"): vol.All(vol.Coerce(int), vol.Range(0, 100)),
|
|
|
|
}
|
|
|
|
|
|
|
|
async def async_handle(self, intent_obj: intent.Intent) -> intent.IntentResponse:
|
|
|
|
"""Handle the hass intent."""
|
|
|
|
hass = intent_obj.hass
|
2023-01-19 23:15:01 +00:00
|
|
|
service_data: dict[str, Any] = {}
|
2019-12-01 22:12:57 +00:00
|
|
|
slots = self.async_validate_slots(intent_obj.slots)
|
2023-01-19 23:15:01 +00:00
|
|
|
|
|
|
|
name: str | None = slots.get("name", {}).get("value")
|
|
|
|
if name == "all":
|
|
|
|
# Don't match on name if targeting all entities
|
|
|
|
name = None
|
|
|
|
|
|
|
|
# Look up area first to fail early
|
|
|
|
area_name = slots.get("area", {}).get("value")
|
2023-03-01 02:58:19 +00:00
|
|
|
area: ar.AreaEntry | None = None
|
2023-01-19 23:15:01 +00:00
|
|
|
if area_name is not None:
|
2023-03-01 02:58:19 +00:00
|
|
|
areas = ar.async_get(hass)
|
2023-01-19 23:15:01 +00:00
|
|
|
area = areas.async_get_area(area_name) or areas.async_get_area_by_name(
|
|
|
|
area_name
|
|
|
|
)
|
|
|
|
if area is None:
|
|
|
|
raise intent.IntentHandleError(f"No area named {area_name}")
|
|
|
|
|
|
|
|
# Optional domain/device class filters.
|
|
|
|
# Convert to sets for speed.
|
|
|
|
domains: set[str] | None = None
|
|
|
|
device_classes: set[str] | None = None
|
|
|
|
|
|
|
|
if "domain" in slots:
|
|
|
|
domains = set(slots["domain"]["value"])
|
|
|
|
|
|
|
|
if "device_class" in slots:
|
|
|
|
device_classes = set(slots["device_class"]["value"])
|
|
|
|
|
|
|
|
states = list(
|
|
|
|
intent.async_match_states(
|
|
|
|
hass,
|
|
|
|
name=name,
|
|
|
|
area=area,
|
|
|
|
domains=domains,
|
|
|
|
device_classes=device_classes,
|
|
|
|
)
|
2019-12-01 22:12:57 +00:00
|
|
|
)
|
|
|
|
|
2023-01-19 23:15:01 +00:00
|
|
|
if not states:
|
|
|
|
raise intent.IntentHandleError("No entities matched")
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
if "color" in slots:
|
|
|
|
service_data[ATTR_RGB_COLOR] = slots["color"]["value"]
|
|
|
|
|
|
|
|
if "brightness" in slots:
|
|
|
|
service_data[ATTR_BRIGHTNESS_PCT] = slots["brightness"]["value"]
|
|
|
|
|
|
|
|
response = intent_obj.create_response()
|
2023-01-19 23:15:01 +00:00
|
|
|
needs_brightness = ATTR_BRIGHTNESS_PCT in service_data
|
|
|
|
needs_color = ATTR_RGB_COLOR in service_data
|
|
|
|
|
|
|
|
success_results: list[intent.IntentResponseTarget] = []
|
|
|
|
failed_results: list[intent.IntentResponseTarget] = []
|
|
|
|
service_coros = []
|
|
|
|
|
|
|
|
if area is not None:
|
|
|
|
success_results.append(
|
|
|
|
intent.IntentResponseTarget(
|
|
|
|
type=intent.IntentResponseTargetType.AREA,
|
|
|
|
name=area.name,
|
|
|
|
id=area.id,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
for state in states:
|
|
|
|
target = intent.IntentResponseTarget(
|
|
|
|
type=intent.IntentResponseTargetType.ENTITY,
|
|
|
|
name=state.name,
|
|
|
|
id=state.entity_id,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Test brightness/color
|
|
|
|
supported_color_modes = state.attributes.get(ATTR_SUPPORTED_COLOR_MODES)
|
|
|
|
if (needs_color and not color_supported(supported_color_modes)) or (
|
|
|
|
needs_brightness and not brightness_supported(supported_color_modes)
|
|
|
|
):
|
|
|
|
failed_results.append(target)
|
|
|
|
continue
|
|
|
|
|
|
|
|
service_coros.append(
|
|
|
|
hass.services.async_call(
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{**service_data, ATTR_ENTITY_ID: state.entity_id},
|
|
|
|
context=intent_obj.context,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
success_results.append(target)
|
|
|
|
|
|
|
|
# Handle service calls in parallel.
|
|
|
|
await asyncio.gather(*service_coros)
|
|
|
|
|
|
|
|
response.async_set_results(
|
|
|
|
success_results=success_results, failed_results=failed_results
|
|
|
|
)
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
return response
|