2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Telegram bots using webhooks."""
|
2017-05-10 04:42:17 +00:00
|
|
|
import datetime as dt
|
2021-09-22 18:59:52 +00:00
|
|
|
from http import HTTPStatus
|
2020-08-11 20:57:50 +00:00
|
|
|
from ipaddress import ip_address
|
2017-04-12 04:10:56 +00:00
|
|
|
import logging
|
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
from telegram import Update
|
2019-10-18 00:19:34 +00:00
|
|
|
from telegram.error import TimedOut
|
2022-04-03 03:39:14 +00:00
|
|
|
from telegram.ext import Dispatcher, TypeHandler
|
2019-10-18 00:19:34 +00:00
|
|
|
|
2017-04-12 04:10:56 +00:00
|
|
|
from homeassistant.components.http import HomeAssistantView
|
2021-09-22 18:59:52 +00:00
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_STOP
|
2020-05-08 19:53:28 +00:00
|
|
|
from homeassistant.helpers.network import get_url
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
from . import CONF_TRUSTED_NETWORKS, CONF_URL, BaseTelegramBotEntity
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2017-04-12 04:10:56 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
TELEGRAM_WEBHOOK_URL = "/api/telegram_webhooks"
|
|
|
|
REMOVE_WEBHOOK_URL = ""
|
2017-04-12 04:10:56 +00:00
|
|
|
|
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
async def async_setup_platform(hass, bot, config):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Set up the Telegram webhooks platform."""
|
2022-04-03 03:39:14 +00:00
|
|
|
pushbot = PushBot(hass, bot, config)
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
if not pushbot.webhook_url.startswith("https"):
|
|
|
|
_LOGGER.error("Invalid telegram webhook %s must be https", pushbot.webhook_url)
|
|
|
|
return False
|
2017-04-12 04:10:56 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
webhook_registered = await pushbot.register_webhook()
|
|
|
|
if not webhook_registered:
|
|
|
|
return False
|
2017-05-10 04:42:17 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, pushbot.deregister_webhook)
|
|
|
|
hass.http.register_view(
|
|
|
|
PushBotView(hass, bot, pushbot.dispatcher, config[CONF_TRUSTED_NETWORKS])
|
|
|
|
)
|
|
|
|
return True
|
2017-05-30 04:55:06 +00:00
|
|
|
|
2017-05-23 17:16:54 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
class PushBot(BaseTelegramBotEntity):
|
|
|
|
"""Handles all the push/webhook logic and passes telegram updates to `self.handle_update`."""
|
|
|
|
|
|
|
|
def __init__(self, hass, bot, config):
|
|
|
|
"""Create Dispatcher before calling super()."""
|
|
|
|
self.bot = bot
|
|
|
|
self.trusted_networks = config[CONF_TRUSTED_NETWORKS]
|
|
|
|
# Dumb dispatcher that just gets our updates to our handler callback (self.handle_update)
|
|
|
|
self.dispatcher = Dispatcher(bot, None)
|
|
|
|
self.dispatcher.add_handler(TypeHandler(Update, self.handle_update))
|
|
|
|
super().__init__(hass, config)
|
|
|
|
|
|
|
|
self.base_url = config.get(CONF_URL) or get_url(
|
|
|
|
hass, require_ssl=True, allow_internal=False
|
|
|
|
)
|
|
|
|
self.webhook_url = f"{self.base_url}{TELEGRAM_WEBHOOK_URL}"
|
|
|
|
|
|
|
|
def _try_to_set_webhook(self):
|
|
|
|
_LOGGER.debug("Registering webhook URL: %s", self.webhook_url)
|
2017-07-30 09:14:28 +00:00
|
|
|
retry_num = 0
|
|
|
|
while retry_num < 3:
|
|
|
|
try:
|
2022-04-03 03:39:14 +00:00
|
|
|
return self.bot.set_webhook(self.webhook_url, timeout=5)
|
2019-10-18 00:19:34 +00:00
|
|
|
except TimedOut:
|
2017-07-30 09:14:28 +00:00
|
|
|
retry_num += 1
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.warning("Timeout trying to set webhook (retry #%d)", retry_num)
|
2017-07-30 09:14:28 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
return False
|
2017-04-24 06:20:04 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
async def register_webhook(self):
|
|
|
|
"""Query telegram and register the URL for our webhook."""
|
|
|
|
current_status = await self.hass.async_add_executor_job(
|
|
|
|
self.bot.get_webhook_info
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2022-04-03 03:39:14 +00:00
|
|
|
# Some logging of Bot current status:
|
|
|
|
last_error_date = getattr(current_status, "last_error_date", None)
|
|
|
|
if (last_error_date is not None) and (isinstance(last_error_date, int)):
|
|
|
|
last_error_date = dt.datetime.fromtimestamp(last_error_date)
|
|
|
|
_LOGGER.debug(
|
|
|
|
"Telegram webhook last_error_date: %s. Status: %s",
|
|
|
|
last_error_date,
|
|
|
|
current_status,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
_LOGGER.debug("telegram webhook status: %s", current_status)
|
|
|
|
|
|
|
|
if current_status and current_status["url"] != self.webhook_url:
|
|
|
|
result = await self.hass.async_add_executor_job(self._try_to_set_webhook)
|
|
|
|
if result:
|
|
|
|
_LOGGER.info("Set new telegram webhook %s", self.webhook_url)
|
|
|
|
else:
|
|
|
|
_LOGGER.error("Set telegram webhook failed %s", self.webhook_url)
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
2017-04-12 04:10:56 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
def deregister_webhook(self, event=None):
|
|
|
|
"""Query telegram and deregister the URL for our webhook."""
|
|
|
|
_LOGGER.debug("Deregistering webhook URL")
|
|
|
|
return self.bot.delete_webhook()
|
2017-04-12 04:10:56 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
|
|
|
|
class PushBotView(HomeAssistantView):
|
|
|
|
"""View for handling webhook calls from Telegram."""
|
2017-04-12 04:10:56 +00:00
|
|
|
|
|
|
|
requires_auth = False
|
2022-04-03 03:39:14 +00:00
|
|
|
url = TELEGRAM_WEBHOOK_URL
|
2019-07-31 19:25:30 +00:00
|
|
|
name = "telegram_webhooks"
|
2017-04-12 04:10:56 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
def __init__(self, hass, bot, dispatcher, trusted_networks):
|
|
|
|
"""Initialize by storing stuff needed for setting up our webhook endpoint."""
|
|
|
|
self.hass = hass
|
|
|
|
self.bot = bot
|
|
|
|
self.dispatcher = dispatcher
|
2017-04-12 04:10:56 +00:00
|
|
|
self.trusted_networks = trusted_networks
|
|
|
|
|
2018-10-01 06:56:50 +00:00
|
|
|
async def post(self, request):
|
2017-04-12 04:10:56 +00:00
|
|
|
"""Accept the POST from telegram."""
|
2020-08-11 20:57:50 +00:00
|
|
|
real_ip = ip_address(request.remote)
|
2017-04-12 04:10:56 +00:00
|
|
|
if not any(real_ip in net for net in self.trusted_networks):
|
|
|
|
_LOGGER.warning("Access denied from %s", real_ip)
|
2021-09-22 18:59:52 +00:00
|
|
|
return self.json_message("Access denied", HTTPStatus.UNAUTHORIZED)
|
2017-04-12 04:10:56 +00:00
|
|
|
|
|
|
|
try:
|
2022-04-03 03:39:14 +00:00
|
|
|
update_data = await request.json()
|
2017-04-12 04:10:56 +00:00
|
|
|
except ValueError:
|
2021-09-22 18:59:52 +00:00
|
|
|
return self.json_message("Invalid JSON", HTTPStatus.BAD_REQUEST)
|
2017-04-12 04:10:56 +00:00
|
|
|
|
2022-04-03 03:39:14 +00:00
|
|
|
update = Update.de_json(update_data, self.bot)
|
|
|
|
_LOGGER.debug("Received Update on %s: %s", self.url, update)
|
|
|
|
await self.hass.async_add_executor_job(self.dispatcher.process_update, update)
|
|
|
|
|
2018-10-28 18:39:23 +00:00
|
|
|
return None
|