summaryrefslogtreecommitdiffstats
path: root/Monitoring/plugins/vmware_discovery_runner.py
diff options
context:
space:
mode:
Diffstat (limited to 'Monitoring/plugins/vmware_discovery_runner.py')
-rwxr-xr-xMonitoring/plugins/vmware_discovery_runner.py225
1 files changed, 225 insertions, 0 deletions
diff --git a/Monitoring/plugins/vmware_discovery_runner.py b/Monitoring/plugins/vmware_discovery_runner.py
new file mode 100755
index 00000000..3a17c765
--- /dev/null
+++ b/Monitoring/plugins/vmware_discovery_runner.py
@@ -0,0 +1,225 @@
+#!/usr/bin/env python
+#Copyright (C) 2009-2010 :
+# Gabes Jean, naparuba@gmail.com
+# Gerhard Lausser, Gerhard.Lausser@consol.de
+# Gregory Starck, g.starck@gmail.com
+# Hartmut Goebel <h.goebel@goebel-consult.de>
+#
+#This file is part of Shinken.
+#
+#Shinken is free software: you can redistribute it and/or modify
+#it under the terms of the GNU Affero General Public License as published by
+#the Free Software Foundation, either version 3 of the License, or
+#(at your option) any later version.
+#
+#Shinken 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 Affero General Public License for more details.
+#
+#You should have received a copy of the GNU Affero General Public License
+#along with Shinken. If not, see <http://www.gnu.org/licenses/>.
+
+import os
+import sys
+import shlex
+import shutil
+import optparse
+from subprocess import Popen, PIPE
+
+# Try to load json (2.5 and higer) or simplejson if failed (python2.4)
+try:
+ import json
+except ImportError:
+ # For old Python version, load
+ # simple json (it can be hard json?! It's 2 functions guy!)
+ try:
+ import simplejson as json
+ except ImportError:
+ sys.exit("Error : you need the json or simplejson module for this script")
+
+VERSION = '0.1'
+
+# Search if we can findthe check_esx3.pl file somewhere
+def search_for_check_esx3():
+ me = os.path.abspath( __file__ )
+ my_dir = os.path.dirname(me)
+ possible_paths = [os.path.join(my_dir, 'check_esx3.pl'),
+ '/var/lib/nagios/check_esx3.pl',
+ '/var/lib/plugins/nagios/check_esx3.pl',
+ '/var/lib/shinken/check_esx3.pl',
+ '/usr/local/nagios/libexec/check_esx3.pl',
+ '/usr/local/shinken/libexec/check_esx3.pl',
+ 'c:\\shinken\\libexec\\check_esx3.pl']
+
+ for p in possible_paths:
+ print "Look for", p
+ if os.path.exists(p):
+ print "Found a check_esx3.pl at", p
+ return p
+ return None
+
+
+# Split and clean the rules from a string to a list
+def _split_rules(rules):
+ return [r.strip() for r in rules.split('|')]
+
+# Apply all rules on the objects names
+def _apply_rules(name, rules):
+ if 'nofqdn' in rules:
+ name = name.split('.', 1)[0]
+ if 'lower' in rules:
+ name = name.lower()
+ return name
+
+# Get all vmware hosts from a VCenter and return the list
+def get_vmware_hosts(check_esx_path, vcenter, user, password):
+ list_host_cmd = [check_esx_path, '-D', vcenter, '-u', user, '-p', password,
+ '-l', 'runtime', '-s', 'listhost']
+
+ print "Got host list"
+ print ' '.join(list_host_cmd)
+ p = Popen(list_host_cmd, stdout=PIPE, stderr=PIPE)
+ output = p.communicate()
+
+ print "Exit status", p.returncode
+ if p.returncode == 2:
+ print "Error : the check_esx3.pl return in error :", output
+ sys.exit(2)
+
+ parts = output[0].split(':')
+ hsts_raw = parts[1].split('|')[0]
+ hsts_raw_lst = hsts_raw.split(',')
+
+ hosts = []
+ for hst_raw in hsts_raw_lst:
+ hst_raw = hst_raw.strip()
+ # look as server4.mydomain(UP)
+ elts = hst_raw.split('(')
+ hst = elts[0]
+ hosts.append(hst)
+
+ return hosts
+
+
+# For a specific host, ask all VM on it to the VCenter
+def get_vm_of_host(check_esx_path, vcenter, host, user, password):
+ print "Listing host", host
+ list_vm_cmd = [check_esx_path, '-D', vcenter, '-H', host,
+ '-u', user, '-p', password,
+ '-l', 'runtime', '-s', 'list']
+ print ' '.join(list_vm_cmd)
+ p = Popen(list_vm_cmd, stdout=PIPE)
+ output = p.communicate()
+
+ print "Exit status", p.returncode
+ if p.returncode == 2:
+ print "Error : the check_esx3.pl return in error :", output
+ sys.exit(2)
+
+ parts = output[0].split(':')
+ # Maybe we got a 'CRITICAL - There are no VMs.' message,
+ # if so, we bypass this host
+ if len(parts) < 2:
+ return None
+
+ vms_raw = parts[1].split('|')[0]
+ vms_raw_lst = vms_raw.split(',')
+
+ lst = []
+ for vm_raw in vms_raw_lst:
+ vm_raw = vm_raw.strip()
+ # look as MYVM(UP)
+ elts = vm_raw.split('(')
+ vm = elts[0]
+ lst.append(vm)
+ return lst
+
+
+# Create all tuples of the links for the hosts
+def print_all_links(res, rules):
+ r = []
+ for host in res:
+ host_name = _apply_rules(host, rules)
+ print "%s::isesxhost=1" % host_name
+ for vm in res[host]:
+ # First we apply rules on the names
+ vm_name = _apply_rules(vm, rules)
+ #v = (('host', host_name),('host', vm_name))
+ print "%s::isesxvm=1" % vm_name
+ print "%s::esxhost=%s" % (vm_name, host_name)
+ #r.append(v)
+ return r
+
+
+def write_output(r, path):
+ try:
+ f = open(path+'.tmp', 'wb')
+ buf = json.dumps(r)
+ f.write(buf)
+ f.close()
+ shutil.move(path+'.tmp', path)
+ print "File %s wrote" % path
+ except IOError, exp:
+ sys.exit("Error writing the file %s : %s" % (path, exp))
+
+
+def main(check_esx_path, vcenter, user, password, rules):
+ rules = _split_rules(rules)
+ res = {}
+ hosts = get_vmware_hosts(check_esx_path, vcenter, user, password)
+
+ for host in hosts:
+ lst = get_vm_of_host(check_esx_path, vcenter, host, user, password)
+ if lst:
+ res[host] = lst
+
+
+ print_all_links(res, rules)
+
+ #write_output(r, output)
+ print "Finished!"
+
+
+# Here we go!
+if __name__ == "__main__":
+ # Manage the options
+ parser = optparse.OptionParser(
+ version="Shinken VMware links dumping script version %s" % VERSION)
+ parser.add_option("-x", "--esx3-path", dest='check_esx_path',
+ default='/usr/local/nagios/libexec/check_esx3.pl',
+ help="Full path of the check_esx3.pl script (default: %default)")
+ parser.add_option("-V", "--vcenter", '--Vcenter',
+ help="tThe IP/DNS address of your Vcenter host.")
+ parser.add_option("-u", "--user",
+ help="User name to connect to this Vcenter")
+ parser.add_option("-p", "--password",
+ help="The password of this user")
+ parser.add_option('-r', '--rules', default='',
+ help="Rules of name transformation. Valid names are: "
+ "`lower`: to lower names, "
+ "`nofqdn`: keep only the first name (server.mydomain.com -> server)."
+ "You can use several rules like `lower|nofqdn`")
+
+ opts, args = parser.parse_args()
+ if args:
+ parser.error("does not take any positional arguments")
+
+ if opts.vcenter is None:
+ parser.error("missing -V or --Vcenter option for the vcenter IP/DNS address")
+ if opts.user is None:
+ parser.error("missing -u or --user option for the vcenter username")
+ if opts.password is None:
+ error = True
+ parser.error("missing -p or --password option for the vcenter password")
+ if not os.path.exists(opts.check_esx_path):
+ parser.error("the path %s for the check_esx3.pl script is wrong, missing file" % opts.check_esx_path)
+ else:
+ # Not given, try to find one
+ p = search_for_check_esx3()
+ if p is None:
+ parser.error("Sorry, I cannot find check_esx3.pl, please specify it with -x")
+ #else set it :)
+ opts.check_esx_path = p
+
+ main(**opts.__dict__)