2011-11-18 00:44:28 +00:00
|
|
|
|
#!/usr/bin/env python
|
|
|
|
|
# -*- coding: utf-8 -*-
|
2012-01-27 10:38:27 +00:00
|
|
|
|
import sys
|
2011-11-18 00:56:23 +00:00
|
|
|
|
import os
|
|
|
|
|
import time
|
2011-11-28 12:56:38 +00:00
|
|
|
|
import argparse
|
2011-11-28 12:58:13 +00:00
|
|
|
|
import redis
|
|
|
|
|
from redis.exceptions import ConnectionError
|
2012-03-23 14:15:29 +00:00
|
|
|
|
from rq import use_connection, Queue, Worker
|
2011-11-21 09:52:23 +00:00
|
|
|
|
from rq.utils import gettermsize, make_colorizer
|
|
|
|
|
|
2011-11-21 12:21:23 +00:00
|
|
|
|
red = make_colorizer('darkred')
|
2011-11-21 09:52:23 +00:00
|
|
|
|
green = make_colorizer('darkgreen')
|
2011-11-21 12:21:23 +00:00
|
|
|
|
yellow = make_colorizer('darkyellow')
|
2011-11-18 00:56:23 +00:00
|
|
|
|
|
|
|
|
|
|
2011-11-21 12:21:23 +00:00
|
|
|
|
def pad(s, pad_to_length):
|
|
|
|
|
"""Pads the given string to the given length."""
|
|
|
|
|
return ('%-' + '%ds' % pad_to_length) % (s,)
|
|
|
|
|
|
2011-11-18 00:56:23 +00:00
|
|
|
|
def get_scale(x):
|
|
|
|
|
"""Finds the lowest scale where x <= scale."""
|
|
|
|
|
scales = [20, 50, 100, 200, 400, 600, 800, 1000]
|
|
|
|
|
for scale in scales:
|
|
|
|
|
if x <= scale:
|
|
|
|
|
return scale
|
|
|
|
|
return x
|
2011-11-18 00:44:28 +00:00
|
|
|
|
|
2011-11-28 10:40:45 +00:00
|
|
|
|
def state_symbol(state):
|
|
|
|
|
symbols = {
|
2012-01-27 10:38:55 +00:00
|
|
|
|
'busy': red('busy'),
|
|
|
|
|
'idle': green('idle'),
|
2011-11-28 10:40:45 +00:00
|
|
|
|
}
|
|
|
|
|
try:
|
|
|
|
|
return symbols[state]
|
|
|
|
|
except KeyError:
|
|
|
|
|
return state
|
|
|
|
|
|
2011-11-18 00:44:28 +00:00
|
|
|
|
|
2011-11-28 12:56:38 +00:00
|
|
|
|
def show_queues(args):
|
2012-03-27 09:22:22 +00:00
|
|
|
|
if len(args.queues):
|
|
|
|
|
qs = map(Queue, args.queues)
|
|
|
|
|
else:
|
|
|
|
|
qs = Queue.all()
|
2011-11-18 00:56:23 +00:00
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
num_jobs = 0
|
|
|
|
|
termwidth, _ = gettermsize()
|
|
|
|
|
chartwidth = min(20, termwidth - 20)
|
|
|
|
|
|
|
|
|
|
max_count = 0
|
|
|
|
|
counts = dict()
|
|
|
|
|
for q in qs:
|
|
|
|
|
count = q.count
|
|
|
|
|
counts[q] = count
|
|
|
|
|
max_count = max(max_count, count)
|
|
|
|
|
scale = get_scale(max_count)
|
|
|
|
|
ratio = chartwidth * 1.0 / scale
|
|
|
|
|
|
|
|
|
|
for q in qs:
|
|
|
|
|
count = counts[q]
|
|
|
|
|
if not args.raw:
|
|
|
|
|
chart = green('|' + '█' * int(ratio * count))
|
|
|
|
|
line = '%-12s %s %d' % (q.name, chart, count)
|
|
|
|
|
else:
|
|
|
|
|
line = 'queue %s %d' % (q.name, count)
|
|
|
|
|
print(line)
|
2011-11-18 00:56:23 +00:00
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
num_jobs += count
|
2011-11-18 00:56:23 +00:00
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
# Print summary when not in raw mode
|
|
|
|
|
if not args.raw:
|
|
|
|
|
print('%d queues, %d jobs total' % (len(qs), num_jobs))
|
2011-11-28 12:56:38 +00:00
|
|
|
|
|
2011-11-18 00:56:23 +00:00
|
|
|
|
|
2011-11-28 12:56:38 +00:00
|
|
|
|
def show_workers(args):
|
2012-03-27 09:22:22 +00:00
|
|
|
|
if len(args.queues):
|
|
|
|
|
qs = map(Queue, args.queues)
|
|
|
|
|
|
|
|
|
|
def any_matching_queue(worker):
|
|
|
|
|
def queue_matches(q):
|
|
|
|
|
return q in qs
|
|
|
|
|
return any(map(queue_matches, worker.queues))
|
|
|
|
|
|
|
|
|
|
# Filter out workers that don't match the queue filter
|
|
|
|
|
ws = [w for w in Worker.all() if any_matching_queue(w)]
|
2012-03-27 10:17:39 +00:00
|
|
|
|
|
|
|
|
|
def filter_queues(queue_names):
|
|
|
|
|
return [qname for qname in queue_names if Queue(qname) in qs]
|
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
else:
|
2011-11-21 12:21:23 +00:00
|
|
|
|
qs = Queue.all()
|
|
|
|
|
ws = Worker.all()
|
2012-03-27 10:17:39 +00:00
|
|
|
|
filter_queues = lambda x: x
|
2011-11-21 12:21:23 +00:00
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
if not args.by_queue:
|
|
|
|
|
for w in ws:
|
2012-03-27 10:17:39 +00:00
|
|
|
|
worker_queues = filter_queues(w.queue_names())
|
2012-03-27 09:22:22 +00:00
|
|
|
|
if not args.raw:
|
2012-03-27 10:17:39 +00:00
|
|
|
|
print '%s %s: %s' % (w.name, state_symbol(w.state), ', '.join(worker_queues))
|
2012-03-27 09:22:22 +00:00
|
|
|
|
else:
|
2012-03-27 10:17:39 +00:00
|
|
|
|
print 'worker %s %s %s' % (w.name, w.state, ','.join(worker_queues))
|
2012-03-27 09:22:22 +00:00
|
|
|
|
else:
|
|
|
|
|
# Create reverse lookup table
|
|
|
|
|
queues = {q: [] for q in qs}
|
2011-11-21 12:21:23 +00:00
|
|
|
|
for w in ws:
|
|
|
|
|
for q in w.queues:
|
|
|
|
|
if not q in queues:
|
2012-03-27 09:22:22 +00:00
|
|
|
|
continue
|
2011-11-21 12:21:23 +00:00
|
|
|
|
queues[q].append(w)
|
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
max_qname = max(map(lambda q: len(q.name), queues.keys())) if queues else 0
|
|
|
|
|
for q in queues:
|
|
|
|
|
if queues[q]:
|
|
|
|
|
queues_str = ", ".join(sorted(map(lambda w: '%s (%s)' % (w.name, state_symbol(w.state)), queues[q])))
|
|
|
|
|
else:
|
|
|
|
|
queues_str = '–'
|
|
|
|
|
print '%s %s' % (pad(q.name + ':', max_qname + 1), queues_str)
|
|
|
|
|
|
|
|
|
|
if not args.raw:
|
|
|
|
|
print '%d workers, %d queues' % (len(ws), len(qs))
|
2011-11-21 12:21:23 +00:00
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
|
|
|
|
|
def show_both(args):
|
|
|
|
|
show_queues(args)
|
|
|
|
|
if not args.raw:
|
|
|
|
|
print ''
|
|
|
|
|
show_workers(args)
|
|
|
|
|
if not args.raw:
|
|
|
|
|
print ''
|
|
|
|
|
import datetime
|
|
|
|
|
print 'Updated: %s' % datetime.datetime.now()
|
2011-11-21 12:21:23 +00:00
|
|
|
|
|
2011-11-18 00:56:23 +00:00
|
|
|
|
|
2011-11-28 12:56:38 +00:00
|
|
|
|
def parse_args():
|
2011-11-28 14:03:49 +00:00
|
|
|
|
parser = argparse.ArgumentParser(description='RQ command-line monitor.')
|
2011-11-28 12:58:13 +00:00
|
|
|
|
parser.add_argument('--host', '-H', default='localhost', help='The Redis hostname (default: localhost)')
|
|
|
|
|
parser.add_argument('--port', '-p', type=int, default=6379, help='The Redis portnumber (default: 6379)')
|
2011-11-28 20:02:42 +00:00
|
|
|
|
parser.add_argument('--db', '-d', type=int, default=0, help='The Redis database (default: 0)')
|
2011-12-13 16:09:58 +00:00
|
|
|
|
parser.add_argument('--path', '-P', default='.', help='Specify the import path.')
|
2012-03-27 09:22:22 +00:00
|
|
|
|
parser.add_argument('--interval', '-i', metavar='N', type=float, default=2.5, help='Updates stats every N seconds (default: don\'t poll)')
|
|
|
|
|
parser.add_argument('--raw', '-r', action='store_true', default=False, help='Print only the raw numbers, no bar charts')
|
|
|
|
|
parser.add_argument('--only-queues', '-Q', dest='only_queues', default=False, action='store_true', help='Show only queue info')
|
|
|
|
|
parser.add_argument('--only-workers', '-W', dest='only_workers', default=False, action='store_true', help='Show only worker info')
|
|
|
|
|
parser.add_argument('--by-queue', '-R', dest='by_queue', default=False, action='store_true', help='Shows workers by queue')
|
|
|
|
|
parser.add_argument('queues', nargs='*', help='The queues to poll')
|
|
|
|
|
return parser.parse_args()
|
2011-11-18 00:44:28 +00:00
|
|
|
|
|
2011-11-28 12:56:38 +00:00
|
|
|
|
|
2012-03-27 09:22:22 +00:00
|
|
|
|
def interval(val, func, args):
|
|
|
|
|
while True:
|
2012-03-27 10:23:56 +00:00
|
|
|
|
if val and sys.stdout.isatty():
|
2012-03-27 09:22:22 +00:00
|
|
|
|
os.system('clear')
|
|
|
|
|
func(args)
|
2012-03-27 10:23:56 +00:00
|
|
|
|
if val and sys.stdout.isatty():
|
2012-03-27 09:22:22 +00:00
|
|
|
|
time.sleep(val)
|
|
|
|
|
else:
|
|
|
|
|
break
|
2011-11-28 12:56:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
|
args = parse_args()
|
2011-11-28 12:58:13 +00:00
|
|
|
|
|
2011-12-13 16:09:58 +00:00
|
|
|
|
if args.path:
|
|
|
|
|
sys.path = args.path.split(':') + sys.path
|
|
|
|
|
|
2011-11-28 12:58:13 +00:00
|
|
|
|
# Setup connection to Redis
|
2011-11-28 20:02:42 +00:00
|
|
|
|
redis_conn = redis.Redis(host=args.host, port=args.port, db=args.db)
|
2012-03-23 14:15:29 +00:00
|
|
|
|
use_connection(redis_conn)
|
2011-11-28 12:58:13 +00:00
|
|
|
|
try:
|
2012-03-27 09:22:22 +00:00
|
|
|
|
if args.only_queues:
|
|
|
|
|
func = show_queues
|
|
|
|
|
elif args.only_workers:
|
|
|
|
|
func = show_workers
|
|
|
|
|
else:
|
|
|
|
|
func = show_both
|
|
|
|
|
|
|
|
|
|
interval(args.interval, func, args)
|
2011-11-28 12:58:13 +00:00
|
|
|
|
except ConnectionError as e:
|
|
|
|
|
print(e)
|
2011-11-18 00:44:28 +00:00
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
|
main()
|