2017-04-12 06:10:56 +02:00
|
|
|
"""
|
|
|
|
Allows utilizing telegram webhooks.
|
|
|
|
|
|
|
|
See https://core.telegram.org/bots/webhooks for details
|
|
|
|
about webhooks.
|
|
|
|
|
|
|
|
"""
|
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
from ipaddress import ip_network
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2017-04-24 08:20:04 +02:00
|
|
|
EVENT_HOMEASSISTANT_STOP, HTTP_BAD_REQUEST, HTTP_UNAUTHORIZED)
|
2017-04-12 06:10:56 +02:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.components.http import HomeAssistantView
|
|
|
|
from homeassistant.components.telegram_bot import CONF_ALLOWED_CHAT_IDS, \
|
|
|
|
BaseTelegramBotEntity, PLATFORM_SCHEMA
|
|
|
|
from homeassistant.const import CONF_API_KEY
|
|
|
|
from homeassistant.components.http.util import get_real_ip
|
|
|
|
|
|
|
|
DEPENDENCIES = ['http']
|
2017-04-26 10:50:08 +02:00
|
|
|
REQUIREMENTS = ['python-telegram-bot==5.3.1']
|
2017-04-12 06:10:56 +02:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
TELEGRAM_HANDLER_URL = '/api/telegram_webhooks'
|
2017-04-24 08:20:04 +02:00
|
|
|
REMOVE_HANDLER_URL = ''
|
2017-04-12 06:10:56 +02:00
|
|
|
|
|
|
|
CONF_TRUSTED_NETWORKS = 'trusted_networks'
|
|
|
|
DEFAULT_TRUSTED_NETWORKS = [
|
|
|
|
ip_network('149.154.167.197/32'),
|
|
|
|
ip_network('149.154.167.198/31'),
|
|
|
|
ip_network('149.154.167.200/29'),
|
|
|
|
ip_network('149.154.167.208/28'),
|
|
|
|
ip_network('149.154.167.224/29'),
|
|
|
|
ip_network('149.154.167.232/31')
|
|
|
|
]
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Optional(CONF_TRUSTED_NETWORKS, default=DEFAULT_TRUSTED_NETWORKS):
|
|
|
|
vol.All(cv.ensure_list, [ip_network])
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
def setup_platform(hass, config, async_add_devices, discovery_info=None):
|
|
|
|
"""Setup the polling platform."""
|
|
|
|
import telegram
|
|
|
|
bot = telegram.Bot(config[CONF_API_KEY])
|
|
|
|
|
|
|
|
current_status = bot.getWebhookInfo()
|
2017-04-24 08:20:04 +02:00
|
|
|
handler_url = '{0}{1}'.format(
|
|
|
|
hass.config.api.base_url, TELEGRAM_HANDLER_URL)
|
2017-04-12 06:10:56 +02:00
|
|
|
if current_status and current_status['url'] != handler_url:
|
|
|
|
if bot.setWebhook(handler_url):
|
|
|
|
_LOGGER.info("set new telegram webhook %s", handler_url)
|
|
|
|
else:
|
|
|
|
_LOGGER.error("set telegram webhook failed %s", handler_url)
|
2017-04-24 08:20:04 +02:00
|
|
|
return False
|
|
|
|
|
|
|
|
hass.bus.listen_once(
|
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
|
|
lambda event: bot.setWebhook(REMOVE_HANDLER_URL))
|
|
|
|
hass.http.register_view(BotPushReceiver(
|
|
|
|
hass, config[CONF_ALLOWED_CHAT_IDS], config[CONF_TRUSTED_NETWORKS]))
|
|
|
|
return True
|
2017-04-12 06:10:56 +02:00
|
|
|
|
|
|
|
|
|
|
|
class BotPushReceiver(HomeAssistantView, BaseTelegramBotEntity):
|
|
|
|
"""Handle pushes from telegram."""
|
|
|
|
|
|
|
|
requires_auth = False
|
|
|
|
url = TELEGRAM_HANDLER_URL
|
2017-04-24 08:20:04 +02:00
|
|
|
name = 'telegram_webhooks'
|
2017-04-12 06:10:56 +02:00
|
|
|
|
|
|
|
def __init__(self, hass, allowed_chat_ids, trusted_networks):
|
|
|
|
"""Initialize the class."""
|
|
|
|
BaseTelegramBotEntity.__init__(self, hass, allowed_chat_ids)
|
|
|
|
self.trusted_networks = trusted_networks
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def post(self, request):
|
|
|
|
"""Accept the POST from telegram."""
|
|
|
|
real_ip = get_real_ip(request)
|
|
|
|
if not any(real_ip in net for net in self.trusted_networks):
|
|
|
|
_LOGGER.warning("Access denied from %s", real_ip)
|
|
|
|
return self.json_message('Access denied', HTTP_UNAUTHORIZED)
|
|
|
|
|
|
|
|
try:
|
|
|
|
data = yield from request.json()
|
|
|
|
except ValueError:
|
|
|
|
return self.json_message('Invalid JSON', HTTP_BAD_REQUEST)
|
|
|
|
|
|
|
|
if not self.process_message(data):
|
|
|
|
return self.json_message('Invalid message', HTTP_BAD_REQUEST)
|
|
|
|
else:
|
|
|
|
return self.json({})
|