mirror of https://github.com/home-assistant/core
119 lines
3.3 KiB
Python
119 lines
3.3 KiB
Python
"""Offer webhook triggered automation rules."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from dataclasses import dataclass
|
|
import logging
|
|
from typing import Any
|
|
|
|
from aiohttp import hdrs, web
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.const import CONF_PLATFORM, CONF_WEBHOOK_ID
|
|
from homeassistant.core import CALLBACK_TYPE, HassJob, HomeAssistant, callback
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.trigger import TriggerActionType, TriggerInfo
|
|
from homeassistant.helpers.typing import ConfigType
|
|
|
|
from . import (
|
|
DEFAULT_METHODS,
|
|
DOMAIN,
|
|
SUPPORTED_METHODS,
|
|
async_register,
|
|
async_unregister,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
DEPENDENCIES = ("webhook",)
|
|
|
|
CONF_ALLOWED_METHODS = "allowed_methods"
|
|
CONF_LOCAL_ONLY = "local_only"
|
|
|
|
TRIGGER_SCHEMA = cv.TRIGGER_BASE_SCHEMA.extend(
|
|
{
|
|
vol.Required(CONF_PLATFORM): "webhook",
|
|
vol.Required(CONF_WEBHOOK_ID): cv.string,
|
|
vol.Optional(CONF_ALLOWED_METHODS): vol.All(
|
|
cv.ensure_list,
|
|
[vol.All(vol.Upper, vol.In(SUPPORTED_METHODS))],
|
|
vol.Unique(),
|
|
),
|
|
vol.Optional(CONF_LOCAL_ONLY): bool,
|
|
}
|
|
)
|
|
|
|
WEBHOOK_TRIGGERS = f"{DOMAIN}_triggers"
|
|
|
|
|
|
@dataclass(slots=True)
|
|
class TriggerInstance:
|
|
"""Attached trigger settings."""
|
|
|
|
trigger_info: TriggerInfo
|
|
job: HassJob
|
|
|
|
|
|
async def _handle_webhook(
|
|
hass: HomeAssistant, webhook_id: str, request: web.Request
|
|
) -> None:
|
|
"""Handle incoming webhook."""
|
|
base_result: dict[str, Any] = {"platform": "webhook", "webhook_id": webhook_id}
|
|
|
|
if "json" in request.headers.get(hdrs.CONTENT_TYPE, ""):
|
|
base_result["json"] = await request.json()
|
|
else:
|
|
base_result["data"] = await request.post()
|
|
|
|
base_result["query"] = request.query
|
|
base_result["description"] = "webhook"
|
|
|
|
triggers: dict[str, list[TriggerInstance]] = hass.data.setdefault(
|
|
WEBHOOK_TRIGGERS, {}
|
|
)
|
|
for trigger in triggers[webhook_id]:
|
|
result = {**base_result, **trigger.trigger_info["trigger_data"]}
|
|
hass.async_run_hass_job(trigger.job, {"trigger": result})
|
|
|
|
|
|
async def async_attach_trigger(
|
|
hass: HomeAssistant,
|
|
config: ConfigType,
|
|
action: TriggerActionType,
|
|
trigger_info: TriggerInfo,
|
|
) -> CALLBACK_TYPE:
|
|
"""Trigger based on incoming webhooks."""
|
|
webhook_id: str = config[CONF_WEBHOOK_ID]
|
|
local_only = config.get(CONF_LOCAL_ONLY, True)
|
|
allowed_methods = config.get(CONF_ALLOWED_METHODS, DEFAULT_METHODS)
|
|
job = HassJob(action)
|
|
|
|
triggers: dict[str, list[TriggerInstance]] = hass.data.setdefault(
|
|
WEBHOOK_TRIGGERS, {}
|
|
)
|
|
|
|
if webhook_id not in triggers:
|
|
async_register(
|
|
hass,
|
|
trigger_info["domain"],
|
|
trigger_info["name"],
|
|
webhook_id,
|
|
_handle_webhook,
|
|
local_only=local_only,
|
|
allowed_methods=allowed_methods,
|
|
)
|
|
triggers[webhook_id] = []
|
|
|
|
trigger_instance = TriggerInstance(trigger_info, job)
|
|
triggers[webhook_id].append(trigger_instance)
|
|
|
|
@callback
|
|
def unregister() -> None:
|
|
"""Unregister webhook."""
|
|
triggers[webhook_id].remove(trigger_instance)
|
|
if not triggers[webhook_id]:
|
|
async_unregister(hass, webhook_id)
|
|
triggers.pop(webhook_id)
|
|
|
|
return unregister
|