2019-02-14 04:35:12 +00:00
|
|
|
"""Support for Mailgun."""
|
2018-10-30 11:12:41 +00:00
|
|
|
import hashlib
|
|
|
|
import hmac
|
|
|
|
import json
|
|
|
|
import logging
|
2018-10-23 09:14:46 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import CONF_API_KEY, CONF_DOMAIN, CONF_WEBHOOK_ID
|
|
|
|
from homeassistant.helpers import config_entry_flow
|
2019-12-05 05:14:39 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-10-23 09:14:46 +00:00
|
|
|
|
2019-05-13 08:16:55 +00:00
|
|
|
from .const import DOMAIN
|
|
|
|
|
2018-10-30 11:12:41 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2019-02-14 04:35:12 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_SANDBOX = "sandbox"
|
2019-02-14 04:35:12 +00:00
|
|
|
|
2018-10-23 09:14:46 +00:00
|
|
|
DEFAULT_SANDBOX = False
|
2019-02-14 04:35:12 +00:00
|
|
|
|
2019-09-03 19:14:00 +00:00
|
|
|
MESSAGE_RECEIVED = f"{DOMAIN}_message_received"
|
2018-10-23 09:14:46 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(DOMAIN): vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_API_KEY): cv.string,
|
|
|
|
vol.Required(CONF_DOMAIN): cv.string,
|
|
|
|
vol.Optional(CONF_SANDBOX, default=DEFAULT_SANDBOX): cv.boolean,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2018-10-23 09:14:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def async_setup(hass, config):
|
|
|
|
"""Set up the Mailgun component."""
|
|
|
|
if DOMAIN not in config:
|
|
|
|
return True
|
|
|
|
|
|
|
|
hass.data[DOMAIN] = config[DOMAIN]
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def handle_webhook(hass, webhook_id, request):
|
|
|
|
"""Handle incoming webhook with Mailgun inbound messages."""
|
2018-10-30 11:12:41 +00:00
|
|
|
body = await request.text()
|
|
|
|
try:
|
|
|
|
data = json.loads(body) if body else {}
|
|
|
|
except ValueError:
|
|
|
|
return None
|
|
|
|
|
2021-03-27 09:03:15 +00:00
|
|
|
if (
|
|
|
|
isinstance(data, dict)
|
|
|
|
and "signature" in data
|
|
|
|
and await verify_webhook(hass, **data["signature"])
|
|
|
|
):
|
|
|
|
data["webhook_id"] = webhook_id
|
|
|
|
hass.bus.async_fire(MESSAGE_RECEIVED, data)
|
|
|
|
return
|
2018-10-30 11:12:41 +00:00
|
|
|
|
|
|
|
_LOGGER.warning(
|
2019-07-31 19:25:30 +00:00
|
|
|
"Mailgun webhook received an unauthenticated message - webhook_id: %s",
|
|
|
|
webhook_id,
|
2018-10-30 11:12:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def verify_webhook(hass, token=None, timestamp=None, signature=None):
|
|
|
|
"""Verify webhook was signed by Mailgun."""
|
|
|
|
if DOMAIN not in hass.data:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.warning("Cannot validate Mailgun webhook, missing API Key")
|
2018-10-30 11:12:41 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
if not (token and timestamp and signature):
|
|
|
|
return False
|
|
|
|
|
|
|
|
hmac_digest = hmac.new(
|
2019-07-31 19:25:30 +00:00
|
|
|
key=bytes(hass.data[DOMAIN][CONF_API_KEY], "utf-8"),
|
2019-09-03 19:14:00 +00:00
|
|
|
msg=bytes(f"{timestamp}{token}", "utf-8"),
|
2019-07-31 19:25:30 +00:00
|
|
|
digestmod=hashlib.sha256,
|
2018-10-30 11:12:41 +00:00
|
|
|
).hexdigest()
|
|
|
|
|
|
|
|
return hmac.compare_digest(signature, hmac_digest)
|
2018-10-23 09:14:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(hass, entry):
|
|
|
|
"""Configure based on config entry."""
|
|
|
|
hass.components.webhook.async_register(
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN, "Mailgun", entry.data[CONF_WEBHOOK_ID], handle_webhook
|
|
|
|
)
|
2018-10-23 09:14:46 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_unload_entry(hass, entry):
|
|
|
|
"""Unload a config entry."""
|
|
|
|
hass.components.webhook.async_unregister(entry.data[CONF_WEBHOOK_ID])
|
|
|
|
return True
|
|
|
|
|
2019-04-01 12:07:12 +00:00
|
|
|
|
|
|
|
async_remove_entry = config_entry_flow.webhook_async_remove_entry
|