#!/usr/bin/env python3 import argparse import configparser import os import subprocess import sys import time from collections import defaultdict from typing import Dict ZULIP_PATH = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) sys.path.append(ZULIP_PATH) from scripts.lib.check_rabbitmq_queue import normal_queues states = { 0: "OK", 1: "WARNING", 2: "CRITICAL", 3: "UNKNOWN", } if 'USER' in os.environ and not os.environ['USER'] in ['root', 'rabbitmq']: print("This script must be run as the root or rabbitmq user") usage = """Usage: check-rabbitmq-consumers --queue=[queue-name] --min-threshold=[min-threshold]""" parser = argparse.ArgumentParser(usage=usage) parser.add_argument('--min-threshold', dest='min_count', type=int, default=1, action='store') options = parser.parse_args() config_file = configparser.RawConfigParser() config_file.read("/etc/zulip/zulip.conf") def get_config(section: str, key: str, default_value: str) -> str: if config_file.has_option(section, key): return config_file.get(section, key) return default_value TORNADO_PROCESSES = int(get_config('application_server', 'tornado_processes', '1')) output = subprocess.check_output(['/usr/sbin/rabbitmqctl', 'list_consumers'], universal_newlines=True) consumers: Dict[str, int] = defaultdict(int) sys.path.append(os.path.join(os.path.dirname(os.path.dirname(os.path.dirname(__file__))))) queues = set(normal_queues).union({ # These queues may not be present if settings.TORNADO_PROCESSES > 1 'notify_tornado', }) for queue_name in queues: queue_name = queue_name.strip() consumers[queue_name] = 0 for line in output.split('\n'): parts = line.split('\t') if len(parts) >= 2: queue_name = parts[0] if queue_name.startswith("notify_tornado_"): queue_name = "notify_tornado" consumers[queue_name] += 1 now = int(time.time()) for queue_name in consumers.keys(): state_file_path = "/var/lib/nagios_state/check-rabbitmq-consumers-" + queue_name state_file_tmp = state_file_path + "-tmp" target_count = options.min_count if queue_name == "notify_tornado": target_count = TORNADO_PROCESSES if consumers[queue_name] < target_count: status = 2 else: status = 0 with open(state_file_tmp, "w") as f: f.write(f"{now}|{status}|{states[status]}|queue {queue_name} has {consumers[queue_name]} consumers, needs {target_count}\n") os.rename(state_file_tmp, state_file_path)