monitoringplugin.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726
  1. #!/usr/bin/env python
  2. # -*- encoding: utf-8 -*-
  3. #####################################################################
  4. # (c) 2010-2011 by Sven Velt and team(ix) GmbH, Nuernberg, Germany #
  5. # sv@teamix.net #
  6. # (c) 2016 by Sven Velt, Germany #
  7. # sven-mymonplugins@velt.biz #
  8. # #
  9. # This file is part of "velt.biz - My Monitoring Plugins" #
  10. # a fork of "team(ix) Monitoring Plugins" in 2015 #
  11. # URL: https://gogs.velt.biz/velt.biz/MyMonPlugins/ #
  12. # #
  13. # This file is free software: you can redistribute it and/or modify #
  14. # it under the terms of the GNU General Public License as published #
  15. # by the Free Software Foundation, either version 2 of the License, #
  16. # or (at your option) any later version. #
  17. # #
  18. # This file is distributed in the hope that it will be useful, but #
  19. # WITHOUT ANY WARRANTY; without even the implied warranty of #
  20. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
  21. # GNU General Public License for more details. #
  22. # #
  23. # You should have received a copy of the GNU General Public License #
  24. # along with this file. If not, see <http://www.gnu.org/licenses/>. #
  25. #####################################################################
  26. __version__ = '0.0.161124'
  27. __all__ = ['MonitoringPlugin', 'SNMPMonitoringPlugin']
  28. import datetime, optparse, os, re, sys
  29. try:
  30. import netsnmp
  31. except ImportError:
  32. pass
  33. class MonitoringPlugin(object):
  34. RETURNSTRINGS = { 0: "OK", 1: "WARNING", 2: "CRITICAL", 3: "UNKNOWN", 127: "UNKNOWN" }
  35. RETURNCODE = { 'OK': 0, 'WARNING': 1, 'CRITICAL': 2, 'UNKNOWN': 3 }
  36. returncode_priority = [2, 1, 3, 0]
  37. powers_binary = ['', 'Ki', 'Mi', 'Gi', 'Ti', 'Pi', 'Ei', 'Zi']
  38. powers_binary_lower = [ p.lower() for p in powers_binary]
  39. powers_si = ['', 'k', 'M', 'G', 'T', 'P', 'E', 'Z']
  40. powers_si_lower = [ p.lower() for p in powers_si]
  41. def __init__(self, *args, **kwargs):
  42. self.__pluginname = kwargs.get('pluginname') or ''
  43. self.__version = kwargs.get('version') or None
  44. self.__tagforstatusline = kwargs.get('tagforstatusline') or ''
  45. self.__tagforstatusline = self.__tagforstatusline.replace('|', ' ')
  46. self.__description = kwargs.get('description') or None
  47. self.__output = []
  48. self.__multilineoutput = []
  49. self.__performancedata = []
  50. self.__returncode = []
  51. self.__brain_checks = []
  52. self.__brain_perfdata = []
  53. self.__brain_perfdatalabels = []
  54. self.__optparser = optparse.OptionParser(version=self.__version, description=self.__description)
  55. self._cmdlineoptions_parsed = False
  56. def add_cmdlineoption(self, shortoption, longoption, dest, help, **kwargs):
  57. if help == None:
  58. help = optparse.SUPPRESS_HELP
  59. self.__optparser.add_option(shortoption, longoption, dest=dest, help=help, **kwargs)
  60. def parse_cmdlineoptions(self):
  61. if self._cmdlineoptions_parsed:
  62. return
  63. # self.__optparser.add_option('-V', '--version', action='version', help='show version number and exit')
  64. self.__optparser.add_option('-v', '--verbose', dest='verbose', help='Verbosity, more for more ;-)', action='count')
  65. (self.options, self.args) = self.__optparser.parse_args()
  66. self._cmdlineoptions_parsed = True
  67. def range_to_limits(self, range):
  68. # Check if we must negate result
  69. if len(range) > 0 and range[0] == '@':
  70. negate = True
  71. range = range[1:]
  72. else:
  73. negate = False
  74. # Look for a ':'...
  75. if range.find(':') >= 0:
  76. # ... this is a range
  77. (low, high) = range.split(':')
  78. if not low:
  79. low = float(0.0)
  80. elif low[0] == '~':
  81. low = float('-infinity')
  82. else:
  83. low = float(low)
  84. if high:
  85. high = float(high)
  86. else:
  87. high = float('infinity')
  88. elif len(range) == 0:
  89. low = float('-infinity')
  90. high = float('infinity')
  91. else:
  92. # ... this is just a number
  93. low = float(0.0)
  94. high = float(range)
  95. return (low, high, negate)
  96. def value_in_range(self, value, range):
  97. if range not in ['', None]:
  98. (low, high, negate) = self.range_to_limits(range)
  99. else:
  100. return True
  101. if value < low or value > high:
  102. result = False
  103. else:
  104. result = True
  105. if negate:
  106. result = not result
  107. return result
  108. def value_wc_to_returncode(self, value, range_warn, range_crit):
  109. if not self.value_in_range(value, range_crit):
  110. return 2
  111. elif not self.value_in_range(value, range_warn):
  112. return 1
  113. return 0
  114. def is_float(self, string):
  115. try:
  116. float(string)
  117. return True
  118. except ValueError:
  119. return False
  120. def special_value_wc_to_returncode(self, value, warn, crit):
  121. # Special add on: WARN > CRIT
  122. if self.is_float(warn) and self.is_float(crit) and float(warn) > float(crit):
  123. # Test if value is *smaller* than thresholds
  124. warn = '@0:' + warn
  125. crit = '@0:' + crit
  126. return self.value_wc_to_returncode(value, warn, crit)
  127. def add_output(self, value):
  128. self.__output.append(value)
  129. def add_multilineoutput(self, value):
  130. self.__multilineoutput.append(value)
  131. def format_performancedata(self, label, value, unit, *args, **kwargs):
  132. label = label.lstrip().rstrip()
  133. if re.search('[=\' ]', label):
  134. label = '\'' + label + '\''
  135. perfdata = label + '=' + str(value)
  136. if unit:
  137. perfdata += str(unit).lstrip().rstrip()
  138. for key in ['warn', 'crit', 'min', 'max']:
  139. perfdata += ';'
  140. if key in kwargs and kwargs[key]!=None:
  141. perfdata += str(kwargs[key])
  142. return perfdata
  143. def add_performancedata(self, perfdata):
  144. self.__performancedata.append(perfdata)
  145. def format_add_performancedata(self, label, value, unit, *args, **kwargs):
  146. self.add_performancedata(self.format_performancedata(label, value, unit, *args, **kwargs))
  147. def add_returncode(self, value):
  148. self.__returncode.append(value)
  149. def tagtarget(self, tag, target):
  150. if target:
  151. return str(tag) + ':' + str(target)
  152. else:
  153. return str(tag)
  154. def remember_check(self, tag, returncode, output, multilineoutput=None, perfdata=None, target=None):
  155. check = {}
  156. check['tag'] = tag
  157. check['returncode'] = returncode
  158. check['output'] = output
  159. check['multilineoutput'] = multilineoutput
  160. check['perfdata'] = perfdata
  161. check['target'] = target
  162. self.remember_perfdata(perfdata)
  163. self.__brain_checks.append(check)
  164. return check
  165. def remember_perfdata(self, perfdata=None):
  166. if perfdata:
  167. for pd in perfdata:
  168. if pd['label'] in self.__brain_perfdatalabels:
  169. pdidx = self.__brain_perfdatalabels.index(pd['label'])
  170. self.__brain_perfdata[pdidx] = pd
  171. else:
  172. self.__brain_perfdata.append(pd)
  173. self.__brain_perfdatalabels.append(pd['label'])
  174. def dump_brain(self):
  175. return (self.__brain_checks, self.__brain_perfdata)
  176. def brain2output(self):
  177. if len(self.__brain_checks) == 1:
  178. check = self.__brain_checks[0]
  179. self.add_output(check.get('output'))
  180. if check.get('multilineoutput'):
  181. self.add_multilineoutput(check.get('multilineoutput'))
  182. self.add_returncode(check.get('returncode') or 0)
  183. else:
  184. out = [[], [], [], []]
  185. for check in self.__brain_checks:
  186. tagtarget = self.tagtarget(check['tag'], check.get('target'))
  187. returncode = check.get('returncode') or 0
  188. self.add_returncode(returncode)
  189. out[returncode].append(tagtarget)
  190. self.add_multilineoutput(self.RETURNSTRINGS[returncode] + ' ' + tagtarget + ' - ' + check.get('output'))
  191. if check.get('multilineoutput'):
  192. self.add_multilineoutput(check.get('multilineoutput'))
  193. statusline = []
  194. for retcode in self.returncode_priority:
  195. if len(out[retcode]):
  196. statusline.append(str(len(out[retcode])) + ' ' + self.RETURNSTRINGS[retcode] + ': ' + ' '.join(out[retcode]))
  197. statusline = ', '.join(statusline)
  198. self.add_output(statusline)
  199. for pd in self.__brain_perfdata:
  200. self.format_add_performancedata(**pd)
  201. def value_to_human_binary(self, value, unit=''):
  202. for power in self.powers_binary:
  203. if value < 1024.0:
  204. return "%3.1f%s%s" % (value, power, unit)
  205. value /= 1024.0
  206. if float(value) not in [float('inf'), float('-inf')]:
  207. return "%3.1fYi%s" % (value, unit)
  208. else:
  209. return value
  210. def value_to_human_si(self, value, unit=''):
  211. for power in self.powers_si:
  212. if value < 1000.0:
  213. return "%3.1f%s%s" % (value, power, unit)
  214. value /= 1000.0
  215. if float(value) not in [float('inf'), float('-inf')]:
  216. return "%3.1fY%s" % (value, unit)
  217. else:
  218. return value
  219. def seconds_to_hms(self, seconds):
  220. seconds = int(seconds)
  221. hours = int(seconds / 3600)
  222. seconds -= (hours * 3600)
  223. minutes = seconds / 60
  224. seconds -= (minutes * 60)
  225. return '%i:%02i:%02i' % (hours, minutes, seconds)
  226. def seconds_to_timedelta(self, seconds):
  227. return datetime.timedelta(seconds=long(seconds))
  228. def human_to_number(self, value, total=None, unit=['',]):
  229. if total:
  230. if not self.is_float(total):
  231. total = self.human_to_number(total, unit=unit)
  232. if type(unit) == list:
  233. unit = [u.lower() for u in unit]
  234. elif type(unit) == str:
  235. unit = [unit.lower(),]
  236. else:
  237. unit = ['',]
  238. if value.lower()[-1] in unit:
  239. value = value[0:-1]
  240. if self.is_float(value):
  241. return float(value)
  242. elif value[-1] == '%':
  243. if total:
  244. return float(value[:-1])/100.0 * float(total)
  245. else:
  246. if total in [0, 0.0]:
  247. return 0.0
  248. else:
  249. return float(value[:-1]) # FIXME: Good idea?
  250. elif value[-1].lower() in self.powers_si_lower:
  251. return 1000.0 ** self.powers_si_lower.index(value[-1].lower()) * float(value[:-1])
  252. elif value[-2:].lower() in self.powers_binary_lower:
  253. return 1024.0 ** self.powers_binary_lower.index(value[-2:].lower()) * float(value[:-2])
  254. else:
  255. return value
  256. def range_dehumanize(self, range, total=None, unit=['',]):
  257. newrange = ''
  258. if len(range):
  259. if range[0] == '@':
  260. newrange += '@'
  261. range = range[1:]
  262. parts = range.split(':')
  263. newrange += ('%s' % self.human_to_number(parts[0], total, unit)).rstrip('0').rstrip('.')
  264. if len(parts) > 1:
  265. newrange += ':' + ('%s' % self.human_to_number(parts[1], total, unit)).rstrip('0').rstrip('.')
  266. if range != newrange:
  267. self.verbose(3, 'Changed range/thresold from "' + range + '" to "' + newrange + '"')
  268. return newrange
  269. else:
  270. return ''
  271. def verbose(self, level, output, prefix=None):
  272. if level <= self.options.verbose:
  273. bol = 'V' + str(level) + ':' + ' ' * level
  274. if prefix:
  275. bol += '%s' % prefix
  276. if type(output) in [str, unicode, ]:
  277. print(bol + output)
  278. elif type(output) in [list, ]:
  279. print('\n'.join( ['%s%s' % (bol, l) for l in output] ) )
  280. else:
  281. print('%s%s' % (bol, output) )
  282. def max_returncode(self, returncodes):
  283. for rc in self.returncode_priority:
  284. if rc in returncodes:
  285. break
  286. return rc
  287. def exit(self):
  288. returncode = self.max_returncode(self.__returncode)
  289. self.back2nagios(returncode, statusline=self.__output, multiline=self.__multilineoutput, performancedata=self.__performancedata)
  290. def back2nagios(self, returncode, statusline=None, multiline=None, performancedata=None, subtag=None, exit=True):
  291. # FIXME: Make 'returncode' also accept strings
  292. # Build status line
  293. out = self.__tagforstatusline
  294. if subtag:
  295. out += '(' + subtag.replace('|', ' ') + ')'
  296. out += ' ' + self.RETURNSTRINGS[returncode]
  297. # Check if there's a status line text and build it
  298. if statusline:
  299. out += ' - '
  300. if type(statusline) == str:
  301. out += statusline
  302. elif type(statusline) in [list, tuple]:
  303. out += ', '.join(statusline).replace('|', ' ')
  304. # Check if we have multi line output and build it
  305. if multiline:
  306. if type(multiline) == str:
  307. out += '\n' + multiline.replace('|', ' ')
  308. elif type(multiline) in [list, tuple]:
  309. if type(multiline[0]) in [list, tuple]:
  310. out += '\n' + '\n'.join([item for sublist in multiline for item in sublist]).replace('|', ' ')
  311. else:
  312. out += '\n' + '\n'.join(multiline).replace('|', ' ')
  313. # Check if there's perfdata
  314. if performancedata:
  315. out += '|'
  316. if type(performancedata) == str:
  317. out += performancedata
  318. elif type(performancedata) in [list, tuple]:
  319. out += ' '.join(performancedata).replace('|', ' ')
  320. # Exit program or return output line(s)
  321. if exit:
  322. print out
  323. sys.exit(returncode)
  324. else:
  325. return (returncode, out)
  326. ##############################################################################
  327. class SNMPMonitoringPlugin(MonitoringPlugin):
  328. def __init__(self, *args, **kwargs):
  329. # Same as "MonitoringPlugin.__init__(*args, **kwargs)" but a little bit more flexible
  330. #super(MonitoringPlugin, self).__init__(*args, **kwargs)
  331. MonitoringPlugin.__init__(self, *args, **kwargs)
  332. self.add_cmdlineoption('-H', '', 'host', 'Host to check', default='127.0.0.1')
  333. self.add_cmdlineoption('-P', '', 'snmpversion', 'SNMP protocol version', metavar='1', default='1')
  334. self.add_cmdlineoption('-C', '', 'snmpauth', 'SNMP v1/v2c community OR SNMP v3 quadruple', metavar='public', default='public')
  335. self.add_cmdlineoption('', '--snmpcmdlinepath', 'snmpcmdlinepath', 'Path to "snmpget" and "snmpwalk"', metavar='/usr/bin/', default='/usr/bin')
  336. # FIXME
  337. self.add_cmdlineoption('', '--nonetsnmp', 'nonetsnmp', 'Do not use NET-SNMP python bindings', action='store_true')
  338. # self.__optparser.add_option('', '--nonetsnmp', dest='nonetsnmp', help='Do not use NET-SNMP python bindings', action='store_true')
  339. self.__SNMP_Cache = {}
  340. self.__use_netsnmp = False
  341. self.__prepared_snmp = False
  342. def prepare_snmp(self):
  343. if not self._cmdlineoptions_parsed:
  344. self.parse_cmdlineoptions()
  345. if not self.options.nonetsnmp:
  346. try:
  347. import netsnmp
  348. self.__use_netsnmp = True
  349. except ImportError:
  350. pass
  351. if self.__use_netsnmp:
  352. self.verbose(1, 'Using NET-SNMP Python bindings')
  353. self.SNMPGET_wrapper = self.__SNMPGET_netsnmp
  354. self.SNMPWALK_wrapper = self.__SNMPWALK_netsnmp
  355. if self.options.snmpversion == '2c':
  356. self.options.snmpversion = '2'
  357. else:
  358. self.verbose(1, 'Using NET-SNMP command line tools')
  359. self.SNMPGET_wrapper = self.__SNMPGET_cmdline
  360. self.SNMPWALK_wrapper = self.__SNMPWALK_cmdline
  361. # Building command lines
  362. self.__CMDLINE_get = os.path.join(self.options.snmpcmdlinepath, 'snmpget') + ' -OqevtU '
  363. self.__CMDLINE_walk = os.path.join(self.options.snmpcmdlinepath, 'snmpwalk') + ' -OqevtU '
  364. if self.options.snmpversion in [1, 2, '1', '2', '2c']:
  365. if self.options.snmpversion in [2, '2']:
  366. self.options.snmpversion = '2c'
  367. self.__CMDLINE_get += ' -v' + str(self.options.snmpversion) + ' -c' + self.options.snmpauth + ' '
  368. self.__CMDLINE_walk += ' -v' + str(self.options.snmpversion) + ' -c' + self.options.snmpauth + ' '
  369. elif options.snmpversion == [3, '3']:
  370. # FIXME: Better error handling
  371. try:
  372. snmpauth = self.options.snmpauth.split(':')
  373. self.__CMDLINE_get += ' -v3 -l' + snmpauth[0] + ' -u' + snmpauth[1] + ' -a' + snmpauth[2] + ' -A' + snmpauth[3] + ' '
  374. self.__CMDLINE_walk += ' -v3 -l' + snmpauth[0] + ' -u' + snmpauth[1] + ' -a' + snmpauth[2] + ' -A' + snmpauth[3] + ' '
  375. except:
  376. self.back2nagios(3, 'Could not build SNMPv3 command line, need "SecLevel:SecName:AuthProtocol:AuthKey"')
  377. else:
  378. self.back2nagios(3, 'Unknown SNMP version "' + str(self.options.snmpversion) + '"')
  379. self.__CMDLINE_get += ' ' + self.options.host + ' %s 2>/dev/null'
  380. self.__CMDLINE_walk += ' ' + self.options.host + ' %s 2>/dev/null'
  381. self.verbose(3, 'Using commandline: ' + self.__CMDLINE_get)
  382. self.verbose(3, 'Using commandline: ' + self.__CMDLINE_walk)
  383. # Test if snmp(get|walk) are executable
  384. for fpath in [self.__CMDLINE_get, self.__CMDLINE_walk,]:
  385. fpath = fpath.split(' ',1)[0]
  386. if not( os.path.exists(fpath) and os.path.isfile(fpath) and os.access(fpath, os.X_OK) ):
  387. self.back2nagios(3, 'Could not execute "%s"' % fpath)
  388. self.__prepared_snmp = True
  389. def find_index_for_value(self, list_indexes, list_values, wanted):
  390. self.verbose(2, 'Look for "' + str(wanted) + '"')
  391. index = None
  392. if len(list_indexes) != len(list_values):
  393. self.verbose(1, 'Length of index and value lists do not match!')
  394. return None
  395. try:
  396. index = list_values.index(wanted)
  397. index = list_indexes[index]
  398. except ValueError:
  399. pass
  400. if index:
  401. self.verbose(2, 'Found "' + str(wanted) +'" with index "' + str(index) + '"')
  402. else:
  403. self.verbose(2, 'Nothing found!')
  404. return index
  405. def find_in_table(self, oid_index, oid_values, wanted):
  406. self.verbose(2, 'Look for "' + str(wanted) + '" in "' + str(oid_values) +'"')
  407. index = None
  408. indexes = list(self.SNMPWALK(oid_index))
  409. values = list(self.SNMPWALK(oid_values))
  410. if len(indexes) != len(values):
  411. self.back2nagios(3, 'Different data from 2 SNMP Walks!')
  412. return self.find_index_for_value(indexes, values, wanted)
  413. def SNMPGET(self, baseoid, idx=None, exitonerror=True):
  414. if type(baseoid) in (list, tuple):
  415. if idx not in ['', None]:
  416. idx = '.' + str(idx)
  417. else:
  418. idx = ''
  419. if self.options.snmpversion in [1, '1']:
  420. value_low = long(self.SNMPGET_wrapper(baseoid[1] + idx, exitonerror=exitonerror))
  421. if value_low < 0L:
  422. value_low += 2 ** 32
  423. value_hi = long(self.SNMPGET_wrapper(baseoid[2] + idx, exitonerror=exitonerror))
  424. if value_hi < 0L:
  425. value_hi += 2 ** 32
  426. return value_hi * 2L ** 32L + value_low
  427. elif self.options.snmpversion in [2, 3, '2', '2c', '3']:
  428. return long(self.SNMPGET_wrapper(baseoid[0] + idx, exitonerror=exitonerror))
  429. elif type(baseoid) in (str, ) and idx != None:
  430. return self.SNMPGET_wrapper(baseoid + '.' + str(idx), exitonerror=exitonerror)
  431. else:
  432. return self.SNMPGET_wrapper(baseoid, exitonerror=exitonerror)
  433. def SNMPWALK(self, baseoid, exitonerror=True):
  434. return self.SNMPWALK_wrapper(baseoid, exitonerror=exitonerror)
  435. def __SNMPGET_netsnmp(self, oid, exitonerror=True):
  436. if not self.__prepared_snmp:
  437. self.prepare_snmp()
  438. if oid in self.__SNMP_Cache:
  439. self.verbose(2, "%40s -> (CACHED) %s" % (oid, self.__SNMP_Cache[oid]))
  440. return self.__SNMP_Cache[oid]
  441. result = netsnmp.snmpget(oid, Version=int(self.options.snmpversion), DestHost=self.options.host, Community=self.options.snmpauth)[0]
  442. if not result:
  443. if exitonerror:
  444. self.back2nagios(3, 'Timeout or no answer from "%s" looking for "%s"' % (self.options.host, oid))
  445. else:
  446. return None
  447. self.__SNMP_Cache[oid] = result
  448. self.verbose(2, "%40s -> %s" % (oid, result))
  449. return result
  450. def __SNMPWALK_netsnmp(self, oid, exitonerror=True):
  451. if not self.__prepared_snmp:
  452. self.prepare_snmp()
  453. if oid in self.__SNMP_Cache:
  454. self.verbose(2, "%40s -> (CACHED) %s" % (oid, self.__SNMP_Cache[oid]))
  455. return self.__SNMP_Cache[oid]
  456. result = netsnmp.snmpwalk(oid, Version=int(self.options.snmpversion), DestHost=self.options.host, Community=self.options.snmpauth)
  457. if not result:
  458. if exitonerror:
  459. self.back2nagios(3, 'Timeout or no answer from "%s" looking for "%s"' % (self.options.host, oid))
  460. else:
  461. return None
  462. self.__SNMP_Cache[oid] = result
  463. self.verbose(2, "%40s -> %s" % (oid, result))
  464. return result
  465. def __SNMPGET_cmdline(self, oid, exitonerror=True):
  466. if not self.__prepared_snmp:
  467. self.prepare_snmp()
  468. cmdline = self.__CMDLINE_get % oid
  469. self.verbose(2, cmdline)
  470. if oid in self.__SNMP_Cache:
  471. self.verbose(2, "(CACHED) %s" % (self.__SNMP_Cache[oid]))
  472. return self.__SNMP_Cache[oid]
  473. cmd = os.popen(cmdline)
  474. out = cmd.readline().rstrip().replace('"','')
  475. retcode = cmd.close()
  476. if retcode:
  477. if not exitonerror:
  478. return None
  479. if retcode == 256:
  480. self.back2nagios(3, 'Timeout - no SNMP answer from "' + self.options.host + '"')
  481. elif retcode ==512:
  482. self.back2nagios(3, 'OID "' + oid + '" not found')
  483. else:
  484. self.back2nagios(3, 'Unknown error code "' + str(retcode) + '" from command line utils')
  485. self.__SNMP_Cache[oid] = out
  486. self.verbose(1, out)
  487. return out
  488. def __SNMPWALK_cmdline(self, oid, exitonerror=True):
  489. if not self.__prepared_snmp:
  490. self.prepare_snmp()
  491. cmdline = self.__CMDLINE_walk % oid
  492. self.verbose(2, cmdline)
  493. if oid in self.__SNMP_Cache:
  494. self.verbose(2, "(CACHED) %s" % (self.__SNMP_Cache[oid]))
  495. return self.__SNMP_Cache[oid]
  496. cmd = os.popen(cmdline)
  497. out = cmd.readlines()
  498. retcode = cmd.close()
  499. if retcode:
  500. if not exitonerror:
  501. return None
  502. if retcode == 256:
  503. self.back2nagios(3, 'Timeout - no SNMP answer from "' + self.options.host + '"')
  504. elif retcode ==512:
  505. self.back2nagios(3, 'OID "' + oid + '" not found')
  506. else:
  507. self.back2nagios(3, 'Unknown error code "' + str(retcode) + '" from command line utils')
  508. for line in range(0,len(out)):
  509. out[line] = out[line].rstrip().replace('"','')
  510. self.__SNMP_Cache[oid] = out
  511. self.verbose(1, str(out))
  512. return out
  513. ##############################################################################
  514. def main():
  515. myplugin = MonitoringPlugin(pluginname='check_testplugin', tagforstatusline='TEST')
  516. from pprint import pprint
  517. pprint(myplugin.back2nagios(0, 'Nr. 01: Simple plugin', exit=False) )
  518. pprint(myplugin.back2nagios(0, 'Nr. 02: Simple plugin with sub tag', subtag='MySubTag', exit=False) )
  519. pprint(myplugin.back2nagios(0, 'Nr. 10: Exit Code OK', exit=False) )
  520. pprint(myplugin.back2nagios(1, 'Nr. 11: Exit Code WARNING', exit=False) )
  521. pprint(myplugin.back2nagios(2, 'Nr. 12: Exit Code CRITICAL', exit=False) )
  522. pprint(myplugin.back2nagios(3, 'Nr. 13: Exit Code UNKNOWN', exit=False) )
  523. ret = myplugin.back2nagios(0, 'Nr. 20: Plugin with string-based multiline output', 'Line 2\nLine 3\nLine4', exit=False)
  524. print ret[1]
  525. print 'Returncode: ' + str(ret[0])
  526. ret = myplugin.back2nagios(0, 'Nr. 21: Plugin with list-based multiline output', ['Line 2', 'Line 3', 'Line4'], exit=False)
  527. print ret[1]
  528. print 'Returncode: ' + str(ret[0])
  529. ret = myplugin.back2nagios(0, 'Nr. 22: Plugin with tuple-based multiline output', ('Line 2', 'Line 3', 'Line4'), exit=False)
  530. print ret[1]
  531. print 'Returncode: ' + str(ret[0])
  532. myplugin.add_performancedata('Val1', 42, '')
  533. myplugin.add_performancedata('Val2', 23, 'c', warn=10, crit=20, min=0, max=100)
  534. myplugin.add_performancedata('Val 3', '2342', 'c', warn=10, crit=20, min=0, max=100)
  535. pprint(myplugin.back2nagios(0, 'Nr. 30: With perfdatas', exit=False) )
  536. myplugin.back2nagios(0, 'Nr. 99: Exit test suite with OK')
  537. if __name__ == '__main__':
  538. main()
  539. #vim: ts=4 sw=4