2009-02-09 19:05:42 +01:00
|
|
|
#!/usr/bin/python
|
|
|
|
|
|
|
|
# Copyright (C) 2009 Andreas Thienemann <andreas@bawue.net>
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Library General Public License as published by
|
|
|
|
# the Free Software Foundation; version 2 only
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU Library General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Library General Public License
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
#
|
|
|
|
|
|
|
|
#
|
|
|
|
# Munin Plugin to get storage device throughput for Bacula by parsing the bconsole
|
|
|
|
# output.
|
|
|
|
#
|
|
|
|
# Parameters:
|
|
|
|
#
|
|
|
|
# config (required)
|
|
|
|
# autoconf (optional - only used by munin-config)
|
|
|
|
#
|
|
|
|
|
|
|
|
# Magic markers (optional - only used by munin-config and some
|
|
|
|
# installation scripts):
|
|
|
|
#
|
|
|
|
#%# family=contrib
|
|
|
|
#%# capabilities=autoconf
|
|
|
|
|
|
|
|
import subprocess
|
|
|
|
import time
|
|
|
|
import sys
|
|
|
|
import re
|
|
|
|
import os
|
|
|
|
|
|
|
|
def parse_devices():
|
|
|
|
""" Parse the bconsole output once to get the device names """
|
|
|
|
|
|
|
|
bconsole = subprocess.Popen("bconsole", stdin=subprocess.PIPE, stdout=subprocess.PIPE)
|
|
|
|
stdout, stderr = bconsole.communicate("status\n2")
|
|
|
|
|
|
|
|
devs = []
|
|
|
|
|
|
|
|
# Hold the line numbers for devices
|
|
|
|
dev_line = []
|
|
|
|
input = stdout.split("\n")
|
|
|
|
|
|
|
|
for line, i in zip(input, range(0, len(input))):
|
|
|
|
if line.startswith("Connecting to Storage daemon "):
|
|
|
|
hostname = line.split()[-1].split(":")[0]
|
|
|
|
if line.startswith("Device \""):
|
|
|
|
dev_line.append(i)
|
|
|
|
|
|
|
|
for pos in dev_line:
|
|
|
|
# Get the device name
|
|
|
|
dev_name = input[pos].split()[1][1:-1]
|
|
|
|
dev_dev = input[pos].split()[2][1:-1]
|
|
|
|
dev_dev_clean = re.sub("^[^A-Za-z_]", "_", dev_dev, 1)
|
|
|
|
dev_dev_clean = re.sub("[^A-Za-z0-9_]", "_", dev_dev_clean, 0)
|
|
|
|
devs.append([dev_name, dev_dev, dev_dev_clean])
|
|
|
|
|
|
|
|
return hostname, devs
|
|
|
|
|
|
|
|
|
|
|
|
def parse():
|
|
|
|
""" Parse the bconsole output """
|
|
|
|
|
|
|
|
bconsole = subprocess.Popen("bconsole", stdin=subprocess.PIPE, stdout=subprocess.PIPE)
|
|
|
|
stdout, stderr = bconsole.communicate("status\n2")
|
|
|
|
|
|
|
|
devstats = []
|
|
|
|
|
|
|
|
# Hold the line numbers for devices
|
|
|
|
dev_line = []
|
|
|
|
input = stdout.split("\n")
|
|
|
|
|
|
|
|
for line, i in zip(input, range(0, len(input))):
|
|
|
|
if line.startswith("Device \""):
|
|
|
|
dev_line.append(i)
|
|
|
|
|
|
|
|
for pos in dev_line:
|
|
|
|
# Get the device name
|
|
|
|
dev_dev = input[pos].split()[2][1:-1]
|
|
|
|
dev_dev_clean = re.sub("^[^A-Za-z_]", "_", dev_dev, 1)
|
|
|
|
dev_dev_clean = re.sub("[^A-Za-z0-9_]", "_", dev_dev_clean, 0)
|
|
|
|
|
|
|
|
# Get the current bytes
|
|
|
|
if input[pos].endswith("is mounted with:"):
|
|
|
|
bytes = long(input[pos+5].split()[1].split("=")[1].replace(",", ""))
|
|
|
|
devstats.append([dev_dev, dev_dev_clean, bytes])
|
|
|
|
else:
|
|
|
|
devstats.append([dev_dev, dev_dev_clean, 0])
|
|
|
|
|
|
|
|
return devstats
|
|
|
|
|
|
|
|
|
|
|
|
def print_config():
|
|
|
|
hostname, devstats = parse_devices()
|
|
|
|
print "graph_title Bacula Storage Daemon throughput"
|
|
|
|
print "graph_vlabel bytes per ${graph_period}"
|
|
|
|
print "graph_args --base 1024 -l 0"
|
|
|
|
print "graph_scale yes"
|
|
|
|
print "graph_info Bacula Storage Daemon througput measurement based on written bytes. This may be somewhat inacurate whenever a tape is changed."
|
2017-02-20 22:14:23 +01:00
|
|
|
print "graph_category backup"
|
2009-02-09 19:05:42 +01:00
|
|
|
print "graph_order",
|
|
|
|
for dev in devstats:
|
|
|
|
print dev[2],
|
|
|
|
print
|
|
|
|
if os.getenv("report_hostname") is not None and \
|
|
|
|
os.getenv("report_hostname").upper() in ["YES", "TRUE", "1", "Y"]:
|
|
|
|
print "host_name", hostname
|
|
|
|
for dev in devstats:
|
|
|
|
print "%s.label %s" % (dev[2], dev[1])
|
|
|
|
print "%s.type DERIVE" % (dev[2])
|
|
|
|
print "%s.min 0" % (dev[2])
|
|
|
|
# print "%s.max %s" % (dev[2], str(1024*1024*1024*16))
|
|
|
|
# print "%s.cdef up,8,*" (dev[2])
|
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
|
|
|
|
if "config" in sys.argv[1:]:
|
|
|
|
print_config()
|
|
|
|
elif "autoconf" in sys.argv[1:]:
|
|
|
|
for dir in os.getenv("PATH").split(":"):
|
|
|
|
for root, dirs, files in os.walk(dir):
|
|
|
|
if "bconsole" in files:
|
|
|
|
print "yes"
|
|
|
|
sys.exit(0)
|
|
|
|
print "no"
|
|
|
|
sys.exit(1)
|
|
|
|
elif "suggest" in sys.argv[1:]:
|
|
|
|
sys.exit(1)
|
|
|
|
else:
|
|
|
|
for dev in parse():
|
|
|
|
print "%s.value %s" % (dev[1], dev[2])
|