root / plugins / bacula / bacula_job @ 430d68ff
Historique | Voir | Annoter | Télécharger (4,66 ko)
| 1 | 9176c668 | Andreas Thienemann | #!/usr/bin/python |
|---|---|---|---|
| 2 | |||
| 3 | # Copyright (C) 2009 Andreas Thienemann <andreas@bawue.net> |
||
| 4 | # |
||
| 5 | # This program is free software; you can redistribute it and/or modify |
||
| 6 | # it under the terms of the GNU Library General Public License as published by |
||
| 7 | # the Free Software Foundation; version 2 only |
||
| 8 | # |
||
| 9 | # This program is distributed in the hope that it will be useful, |
||
| 10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
||
| 11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||
| 12 | # GNU Library General Public License for more details. |
||
| 13 | # |
||
| 14 | # You should have received a copy of the GNU Library General Public License |
||
| 15 | # along with this program; if not, write to the Free Software |
||
| 16 | # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. |
||
| 17 | # |
||
| 18 | |||
| 19 | # |
||
| 20 | # Munin Plugin to get job throughput for Bacula by parsing the bconsole |
||
| 21 | # output. |
||
| 22 | # |
||
| 23 | # Parameters: |
||
| 24 | # |
||
| 25 | # config (required) |
||
| 26 | # autoconf (optional - only used by munin-config) |
||
| 27 | # |
||
| 28 | |||
| 29 | # Magic markers (optional - only used by munin-config and some |
||
| 30 | # installation scripts): |
||
| 31 | # |
||
| 32 | #%# family=contrib |
||
| 33 | #%# capabilities=autoconf |
||
| 34 | |||
| 35 | import subprocess |
||
| 36 | import time |
||
| 37 | import sys |
||
| 38 | import re |
||
| 39 | import os |
||
| 40 | |||
| 41 | def parse_running_jobs(): |
||
| 42 | """ Parse the bconsole output once to get the running jobs """ |
||
| 43 | |||
| 44 | bconsole = subprocess.Popen("bconsole", stdin=subprocess.PIPE, stdout=subprocess.PIPE)
|
||
| 45 | stdout, stderr = bconsole.communicate("status\n1\nstatus\n3\n.")
|
||
| 46 | |||
| 47 | jobs = [] |
||
| 48 | clients = [] |
||
| 49 | clientlist = False |
||
| 50 | |||
| 51 | # Hold the line numbers for devices |
||
| 52 | dev_line = [] |
||
| 53 | input = stdout.split("\n")
|
||
| 54 | |||
| 55 | for line, i in zip(input, range(0, len(input))): |
||
| 56 | if line.startswith("Connecting to Director "):
|
||
| 57 | hostname = line.split()[-1].split(":")[0]
|
||
| 58 | |||
| 59 | if line.endswith(" is running"):
|
||
| 60 | jobs.append(line.split()[2].split(".")[0])
|
||
| 61 | |||
| 62 | # Parse the clientlist, warning, order of statements is important |
||
| 63 | if line.startswith("Select Client (File daemon) resource"):
|
||
| 64 | clientlist = False |
||
| 65 | |||
| 66 | if clientlist is True: |
||
| 67 | client_id, client_name = line.split() |
||
| 68 | client_clean = re.sub("^[^A-Za-z_]", "_", client_name, 1)
|
||
| 69 | client_clean = re.sub("[^A-Za-z0-9_]", "_", client_clean, 0)
|
||
| 70 | clients.append((client_name, client_clean, client_id[:-1])) |
||
| 71 | |||
| 72 | if line.startswith("The defined Client resources are:"):
|
||
| 73 | clientlist = True |
||
| 74 | |||
| 75 | return hostname, jobs, clients |
||
| 76 | |||
| 77 | |||
| 78 | def parse(clients): |
||
| 79 | """ Parse the bconsole output """ |
||
| 80 | |||
| 81 | query_str = "" |
||
| 82 | for client in clients: |
||
| 83 | query_str = query_str + "status\n3\n" + client[1] + "\n" |
||
| 84 | query_str = query_str + "quit" |
||
| 85 | |||
| 86 | bconsole = subprocess.Popen("bconsole", stdin=subprocess.PIPE, stdout=subprocess.PIPE)
|
||
| 87 | stdout, stderr = bconsole.communicate(query_str) |
||
| 88 | |||
| 89 | input = stdout.split("\n")
|
||
| 90 | |||
| 91 | jobstats = [] |
||
| 92 | |||
| 93 | for line, pos in zip(input, range(0, len(input))): |
||
| 94 | |||
| 95 | # Get the client name |
||
| 96 | if line.startswith("Connecting to Client "):
|
||
| 97 | # client_name = input[pos].split()[3].split(".")[0]
|
||
| 98 | client_name = line.split()[3] |
||
| 99 | client_clean = re.sub("^[^A-Za-z_]", "_", client_name, 1)
|
||
| 100 | client_clean = re.sub("[^A-Za-z0-9_]", "_", client_clean, 0)
|
||
| 101 | |||
| 102 | # Get the current bytes |
||
| 103 | if line.endswith(" is running."):
|
||
| 104 | bytes = long(input[pos+2].split()[1].split("=")[1].replace(",", ""))
|
||
| 105 | jobstats.append([client_name, client_clean, bytes]) |
||
| 106 | |||
| 107 | job_dict = {}
|
||
| 108 | for job in jobstats: |
||
| 109 | job_dict[job[0].split("-")[0]] = job
|
||
| 110 | |||
| 111 | return job_dict |
||
| 112 | |||
| 113 | |||
| 114 | def print_config(): |
||
| 115 | hostname, jobs, clients = parse_running_jobs() |
||
| 116 | print "graph_title Bacula Job throughput" |
||
| 117 | print "graph_vlabel bytes per ${graph_period}"
|
||
| 118 | print "graph_args --base 1024 -l 0" |
||
| 119 | print "graph_scale yes" |
||
| 120 | print "graph_info Bacula Job measurement." |
||
| 121 | print "graph_category Bacula" |
||
| 122 | print "graph_order", |
||
| 123 | for fd in clients: |
||
| 124 | print fd[1], |
||
| 125 | |||
| 126 | if os.getenv("report_hostname") is not None and \
|
||
| 127 | os.getenv("report_hostname").upper() in ["YES", "TRUE", "1", "Y"]:
|
||
| 128 | print "host_name", hostname |
||
| 129 | for client in clients: |
||
| 130 | print "%s.label %s" % (client[1], client[0]) |
||
| 131 | print "%s.type DERIVE" % (client[1]) |
||
| 132 | print "%s.min 0" % (client[1]) |
||
| 133 | # print "%s.max %s" % (client[1], str(1024*1024*1024*16)) |
||
| 134 | # print "%s.cdef up,8,*" (client[1]) |
||
| 135 | sys.exit(0) |
||
| 136 | |||
| 137 | |||
| 138 | if "config" in sys.argv[1:]: |
||
| 139 | print_config() |
||
| 140 | elif "autoconf" in sys.argv[1:]: |
||
| 141 | for dir in os.getenv("PATH").split(":"):
|
||
| 142 | for root, dirs, files in os.walk(dir): |
||
| 143 | if "bconsole" in files: |
||
| 144 | print "yes" |
||
| 145 | sys.exit(0) |
||
| 146 | print "no" |
||
| 147 | sys.exit(1) |
||
| 148 | elif "suggest" in sys.argv[1:]: |
||
| 149 | sys.exit(1) |
||
| 150 | else: |
||
| 151 | hostname, jobs, clients = parse_running_jobs() |
||
| 152 | str = [] |
||
| 153 | for client in clients: |
||
| 154 | if client[0].split("-")[0] in jobs:
|
||
| 155 | str.append((client[0], client[2])) |
||
| 156 | |||
| 157 | client_values = parse(str) |
||
| 158 | |||
| 159 | for client in clients: |
||
| 160 | client_name_short = client[0].split("-")[0]
|
||
| 161 | if client_name_short in client_values: |
||
| 162 | print "%s.value %s" % (client_values[client_name_short][1], client_values[client_name_short][2]) |
||
| 163 | else: |
||
| 164 | print "%s.value %s" % (client[1], "0") |
||
| 165 | |||
| 166 | sys.exit(0) |
