2
0
mirror of https://github.com/munin-monitoring/contrib.git synced 2018-11-08 00:59:34 +01:00

Merge pull request #35 from andrewjennings/gunicorn

Contribute gunicorn plugins
This commit is contained in:
Steve Schnepp 2012-02-17 10:28:32 -08:00
commit 7cab2fe50f
2 changed files with 187 additions and 0 deletions

View File

@ -0,0 +1,86 @@
#!/usr/bin/env python
#
# gunicorn_status - A munin plugin for Linux to monitor the memory
# usage of gunicorn processes
#
# Copyright (C) 2012 Azavea, Inc.
# Author: Andrew Jennings
#
# Like Munin, this plugin is licensed under the GNU GPL v2 license
# http://www.opensource.org/licenses/GPL-2.0
#
import sys, os
from subprocess import check_output
# set path to your gunicorn pid
try:
GUNICORN_PID_PATH = os.environ['gunicorn_pid_path']
except:
GUNICORN_PID_PATH = "/var/run/gunicorn.pid"
class GunicornStatus():
master_pid = ''
"""
The Gunicorn master process pid, as a string
"""
def __init__(self):
try:
self._get_master_pid()
except:
raise Exception("Couldn't read gunicorn pid information")
def print_total_memory(self):
print ('total_memory.value %d' % self._get_total_memory())
def _get_master_pid(self):
master_pid_file = open(GUNICORN_PID_PATH)
self.master_pid = master_pid_file.read().rstrip()
master_pid_file.close()
return True
def _get_total_memory(self):
master = self._get_master_memory()
total = master +self. _get_worker_memory()
total_in_mb = total / 1024
return total_in_mb
def _get_master_memory(self):
master = int(check_output(
['ps', '--pid', self.master_pid, '-o', 'rss', '--no-headers']))
return master
def _get_worker_memory(self):
worker_processes = check_output(
['ps', '--ppid', self.master_pid, '-o', 'rss', '--no-headers'])
process_memory_usage = [int(rss) for rss in worker_processes.splitlines()]
worker_memory_usage = sum(process_memory_usage)
return worker_memory_usage
def print_config():
print "graph_title Gunicorn - Memory Usage"
print "graph_args --base 1024 -l 0"
print "graph_vlabel Megabytes"
print "graph_category gunicorn"
print "total_memory.label Total Memory"
if __name__ == "__main__":
if len(sys.argv) == 2 and sys.argv[1] == 'config':
print_config()
elif len(sys.argv) == 2 and sys.argv[1] == 'autoconf':
try:
open(GUNICORN_PID_PATH).close()
print "yes"
except:
print "no"
# Some docs say it'll be called with fetch, some say no arg at all
elif len(sys.argv) == 1 or (len(sys.argv) == 2 and sys.argv[1] == 'fetch'):
try:
status = GunicornStatus()
status.print_total_memory()
except:
sys.exit("Couldn't retrieve gunicorn memory usage information")

101
plugins/other/gunicorn_status Executable file
View File

@ -0,0 +1,101 @@
#!/usr/bin/env python
#
# gunicorn_status - A munin plugin for Linux to monitor gunicorn processes
#
# Copyright (C) 2012 Azavea, Inc.
# Author: Andrew Jennings
#
# Like Munin, this plugin is licensed under the GNU GPL v2 license
# http://www.opensource.org/licenses/GPL-2.0
#
import sys, os
from subprocess import check_output
from time import sleep
# set path to your gunicorn pid
try:
GUNICORN_PID_PATH = os.environ['gunicorn_pid_path']
except:
GUNICORN_PID_PATH = "/var/run/gunicorn.pid"
class GunicornStatus():
master_pid = ''
"""
The gunicorn master process pid, as a string
"""
worker_pids = ''
"""
The list of gunicorn processes as strings
"""
def __init__(self):
try:
self._get_master_pid()
self._get_worker_pids(self.master_pid)
except:
sys.exit("Couldn't read gunicorn pid")
def print_total_workers(self):
print ('total_workers.value %d' % self._worker_count())
def print_idle_workers(self):
print ('idle_workers.value %d' % self._idle_worker_count())
def _get_master_pid(self):
master_pid_file = open(GUNICORN_PID_PATH)
self.master_pid = master_pid_file.read().rstrip()
master_pid_file.close()
def _get_worker_pids(self, master_pid):
children = check_output(
['ps', '--ppid', master_pid, '-o', 'pid', '--no-headers'])
self.worker_pids = [pid.strip() for pid in children.splitlines()]
def _worker_count(self):
return len(self.worker_pids)
def _idle_worker_count(self):
idle_workers = 0
for pid in self.worker_pids:
before = self._cpu_time(pid)
sleep(0.50)
after = self._cpu_time(pid)
if before == after:
idle_workers += 1
return idle_workers
def _cpu_time(self, pid):
proc_info = open('/proc/%s/stat' % pid).read()
proc_info = [field.rstrip() for field in proc_info.split()]
user_time = int(proc_info[13].rstrip())
kernel_time = int(proc_info[14].rstrip())
return user_time + kernel_time
def print_config():
print "graph_title Gunicorn - Status"
print "graph_args -l 0"
print "graph_vlabel Number of workers"
print "graph_category gunicorn"
print "total_workers.label Total Workers"
print "idle_workers.label Idle Workers"
if __name__ == "__main__":
if len(sys.argv) == 2 and sys.argv[1] == 'config':
print_config()
elif len(sys.argv) == 2 and sys.argv[1] == 'autoconf':
try:
open(GUNICORN_PID_PATH).close()
print "yes"
except:
print "no"
# Some docs say it'll be called with fetch, some say no arg at all
elif len(sys.argv) == 1 or (len(sys.argv) == 2 and sys.argv[1] == 'fetch'):
status = GunicornStatus()
try:
status.print_total_workers()
status.print_idle_workers()
except:
sys.exit("Couldn't retrieve gunicorn status")