alerta_alerta/alerta/webhooks/telegram.py

107 lines
4.4 KiB
Python

import os
from typing import Any, Dict, List # noqa
from flask import current_app, g, jsonify, request
from alerta.models.alert import Alert
from alerta.models.blackout import Blackout
from alerta.utils.audit import write_audit_trail
from . import WebhookBase
JSON = Dict[str, Any]
def send_message_reply(alert: Alert, action: str, user: str, data: JSON) -> None:
try:
import telepot # type: ignore
except ImportError:
current_app.logger.warning("You have configured Telegram but 'telepot' client is not installed", exc_info=True)
return
try:
bot_id = os.environ.get('TELEGRAM_TOKEN') or current_app.config.get('TELEGRAM_TOKEN')
dashboard_url = os.environ.get('DASHBOARD_URL') or current_app.config.get('DASHBOARD_URL')
chat_id = os.environ.get('TELEGRAM_CHAT_ID') or current_app.config.get('TELEGRAM_CHAT_ID')
bot = telepot.Bot(bot_id)
# message info
message_id = data['callback_query']['message']['message_id']
message_log = '\n'.join(data['callback_query']['message']['text'].split('\n')[1:])
# process buttons for reply text
inline_keyboard, reply = [], 'The status of alert {alert} is *{status}* now!' # type: List[List[JSON]], str
actions = ['watch', 'unwatch']
if action in actions:
reply = 'User `{user}` is _{status}ing_ alert {alert}'
next_action = actions[(actions.index(action) + 1) % len(actions)]
inline_keyboard = [
[
{'text': next_action.capitalize(), 'callback_data': f'/{next_action} {alert.id}'},
{'text': 'Ack', 'callback_data': f'/ack {alert.id}'},
{'text': 'Close', 'callback_data': f'/close {alert.id}'}
]
]
# format message response
alert_short_id = alert.get_id(short=True)
alert_url = f'{dashboard_url}/#/alert/{alert.id}'
reply = reply.format(alert=alert_short_id, status=action, user=user)
message = '{alert} *{level} - {event} on {resouce}*\n{log}\n{reply}'.format(
alert=f'[{alert_short_id}]({alert_url})', level=alert.severity.capitalize(),
event=alert.event, resouce=alert.resource, log=message_log, reply=reply)
# send message
bot.editMessageText(
msg_identifier=(chat_id, message_id), text=message,
parse_mode='Markdown', reply_markup={'inline_keyboard': inline_keyboard}
)
except Exception:
current_app.logger.warning('Error sending reply message', exc_info=True)
class TelegramWebhook(WebhookBase):
"""
Telegram Bot API
See https://core.telegram.org/bots/api
"""
def incoming(self, path, query_string, payload):
if 'callback_query' in payload:
author = payload['callback_query']['from']
user = f"{author.get('first_name')} {author.get('last_name')}"
command, alert_id = payload['callback_query']['data'].split(' ', 1)
customers = g.get('customers', None)
alert = Alert.find_by_id(alert_id, customers=customers)
action = command.lstrip('/')
if not alert and action != 'blackout':
return jsonify(status='error', message='alert not found for Telegram message')
if action in ['open', 'ack', 'close']:
alert.from_action(action, text='status change via Telegram')
elif action in ['watch', 'unwatch']:
alert.untag(tags=[f'{action}:{user}'])
elif action == 'blackout':
if alert:
# new style paremeters: only alert_id
environment = alert.environment
resource = alert.resource
event = alert.event
blackout = Blackout(environment, resource=resource, event=event)
blackout.create()
send_message_reply(alert, action, user, payload)
text = 'alert updated via telegram webhook'
write_audit_trail.send(current_app._get_current_object(), event='webhook-updated', message=text,
user=g.login, customers=g.customers, scopes=g.scopes, resource_id=alert.id,
type='alert', request=request)
return jsonify(status='ok')
else:
return jsonify(status='ok', message='no callback_query in Telegram message')