#!/usr/bin/env python3 import argparse import json import logging import os import requests from jinja2 import Environment, FileSystemLoader from jsonschema import validate logger = logging.getLogger(__name__) absolute_path = os.path.split(os.path.abspath(__file__))[0] class InvalidConfigException(Exception): pass def parse_arguments(): parser = argparse.ArgumentParser() # base arguments parser.add_argument('-v', '--verbose', dest='loglevel', action='store_const', const=logging.INFO, help='print more output') parser.add_argument('-d', '--debug', dest='loglevel', action='store_const', const=logging.DEBUG, default=logging.WARNING, help='print even more output') parser.add_argument('-c', '--config', help='configuration file location', required=True) parser.add_argument('-o', '--logfile', help='logging file location') subparsers = parser.add_subparsers(title='commands', dest='command', help='nagios notification type') # host notifications host_parser = subparsers.add_parser('host') host_parser.add_argument('--notification-type', help='nagios $NOTIFICATIONTYPE$', required=True) host_parser.add_argument('--service-desc', help='nagios $SERVICEDESC$', required=True) host_parser.add_argument('--host-name', help='nagios $HOSTNAME$', required=True) host_parser.add_argument('--host-state', help='nagios $HOSTSTATE$', required=True) host_parser.add_argument('--host-address', help='nagios $HOSTADDRESS$', required=True) host_parser.add_argument('--host-output', help='nagios $HOSTOUTPUT$', required=True) host_parser.add_argument('--long-date-time', help='nagios $LONGDATETIME$', required=True) # service notifications service_parser = subparsers.add_parser('service') service_parser.add_argument('--notification-type', help='nagios $NOTIFICATIONTYPE$', required=True) service_parser.add_argument('--service-desc', help='nagios $SERVICEDESC$', required=True) service_parser.add_argument('--host-alias', help='nagios $HOSTALIAS$', required=True) service_parser.add_argument('--host-address', help='nagios $HOSTADDRESS$', required=True) service_parser.add_argument('--service-state', help='nagios $SERVICESTATE$', required=True) service_parser.add_argument('--long-date-time', help='nagios $LONGDATETIME$', required=True) service_parser.add_argument('--service-output', help='nagios $SERVICEOUTPUT$', required=True) args = parser.parse_args() return args def read_config(filename=None): if filename and os.path.isfile(filename): with open(filename, 'r') as fd: return json.load(fd) else: return {} def validate_config(config): if config is None: raise InvalidConfigException('config is not a dict') with open(os.path.join(absolute_path, 'config.schema.json'), 'r') as fd: schema = json.loads(fd.read()) validate(instance=config, schema=schema) def setup_logging(args): log_format = '%(asctime)s %(levelname)s: %(message)s' if args.logfile else '%(levelname)s: %(message)s' logging.basicConfig(format=log_format, level=args.loglevel, filename=args.logfile) def markdown_escape(text): for special_char in ['\\', '`', '*', '_', '{', '}', '[', ']', '(', ')', '#', '+', '-', '.', '!', '=']: text = text.replace(special_char, fr'\{special_char}') return text def generate_payload(chat_id, message_type, message_variables, template_file_name=None): payload = {'chat_id': chat_id, 'parse_mode': 'MarkdownV2'} # define jinja template name if not template_file_name: template_name = 'host.md.j2' if message_type == 'host' else 'service.md.j2' template_file_name = os.path.join(absolute_path, 'templates', template_name) template_path = os.path.dirname(os.path.abspath(template_file_name)) template_name = os.path.basename(os.path.abspath(template_file_name)) # create a jinja file template loader = FileSystemLoader(template_path) env = Environment(loader=loader) template = env.get_template(template_name) # escape message variables template_variables = {} for key, value in message_variables.items(): template_variables[key] = markdown_escape(value) # render template and update payload text = template.render(**template_variables) payload['text'] = text return payload def generate_host_payload(chat_id, notification_type, host_name, host_state, host_address, host_output, long_date_time, template_file_name=None): message_variables = { 'notification_type': notification_type, 'host_name': host_name, 'host_state': host_state, 'host_address': host_address, 'host_output': host_output, 'long_date_time': long_date_time } return generate_payload(chat_id=chat_id, message_type='host', message_variables=message_variables, template_file_name=template_file_name) def generate_service_payload(chat_id, notification_type, service_desc, host_alias, host_address, service_state, long_date_time, service_output, template_file_name=None): message_variables = { 'notification_type': notification_type, 'service_desc': service_desc, 'host_alias': host_alias, 'host_address': host_address, 'service_state': service_state, 'long_date_time': long_date_time, 'service_output': service_output } return generate_payload(chat_id=chat_id, message_type='service', message_variables=message_variables, template_file_name=template_file_name) def send_message(auth_key, payload): r = requests.post(f'https://api.telegram.org/bot{auth_key}/sendMessage', json=payload) if r.status_code != requests.codes.ok: description = r.json().get('description') logger.error(f'{r.status_code}: {description}') logger.debug(payload) def main(): try: args = parse_arguments() setup_logging(args) logger.info(f'reading configuration file {args.config}') config = read_config(args.config) logger.info('validating configuration') validate_config(config) logger.info('generating payload') if args.command == 'host': payload = generate_host_payload(chat_id=config['chat_id'], notification_type=args.notification_type, host_name=args.host_name, host_state=args.host_state, host_address=args.host_address, host_output=args.host_output, long_date_time=args.long_date_time, template_file_name=config.get('host_template')) elif args.command == 'service': payload = generate_service_payload(chat_id=config['chat_id'], notification_type=args.notification_type, service_desc=args.service_desc, host_alias=args.host_alias, host_address=args.host_address, service_state=args.service_state, long_date_time=args.long_date_time, service_output=args.service_output, template_file_name=config.get('service_template')) else: raise NotImplementedError(f'command {args.command} not supported') logger.info('sending message to telegram api') send_message(auth_key=config['auth_key'], payload=payload) except Exception: logger.exception('cannot execute program') if __name__ == '__main__': main()