check_naf.py 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962
  1. #!/usr/bin/env python
  2. # -*- encoding: utf-8 -*-
  3. #####################################################################
  4. # (c) 2006-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. import os
  27. import sys
  28. try:
  29. from monitoringplugin import SNMPMonitoringPlugin
  30. except ImportError:
  31. print '=========================='
  32. print 'AIKS! Python import error!'
  33. print '==========================\n'
  34. print 'Could not find "monitoringplugin.py"!\n'
  35. print 'Did you download "%s"' % os.path.basename(sys.argv[0])
  36. print 'without "monitoringplugin.py"?\n'
  37. print 'Please go back to'
  38. print 'https://gogs.velt.biz/velt.biz/MyMonPlugins and download it,'
  39. print 'or even better:'
  40. print 'get a full archive at http://gogs.velt.biz/velt.biz/MyMonPlugins/releases'
  41. print 'or a master snapshot at http://gogs.velt.biz/velt.biz/MyMonPlugins/archive/master.tar.gz\n'
  42. sys.exit(127)
  43. class CheckNAF(SNMPMonitoringPlugin):
  44. OID = {
  45. 'Cluster_Settings': '.1.3.6.1.4.1.789.1.2.3.1.0',
  46. 'Cluster_State': '.1.3.6.1.4.1.789.1.2.3.2.0',
  47. 'Cluster_InterconnectStatus': '.1.3.6.1.4.1.789.1.2.3.8.0',
  48. 'Cluster_CannotTakeOverCause': '.1.3.6.1.4.1.789.1.2.3.3.0',
  49. 'CIFS_Connected_Users': '.1.3.6.1.4.1.789.1.7.2.9.0',
  50. 'CIFS_Total_Ops': '.1.3.6.1.4.1.789.1.7.3.1.1.1.0',
  51. 'CIFS_Total_Calls': '.1.3.6.1.4.1.789.1.7.3.1.1.2.0',
  52. 'CIFS_Bad_Calls': '.1.3.6.1.4.1.789.1.7.3.1.1.3.0',
  53. 'CIFS_Get_Attrs': '.1.3.6.1.4.1.789.1.7.3.1.1.4.0',
  54. 'CIFS_Reads': '.1.3.6.1.4.1.789.1.7.3.1.1.5.0',
  55. 'CIFS_Writes': '.1.3.6.1.4.1.789.1.7.3.1.1.6.0',
  56. 'CIFS_Locks': '.1.3.6.1.4.1.789.1.7.3.1.1.7.0',
  57. 'CIFS_Opens': '.1.3.6.1.4.1.789.1.7.3.1.1.8.0',
  58. 'CIFS_DirOps': '.1.3.6.1.4.1.789.1.7.3.1.1.9.0',
  59. 'CIFS_Others': '.1.3.6.1.4.1.789.1.7.3.1.1.10.0',
  60. 'CP': '.1.3.6.1.4.1.789.1.2.6',
  61. 'CPU_Arch': '.1.3.6.1.4.1.789.1.1.11.0',
  62. 'CPU_Time_Busy': '.1.3.6.1.4.1.789.1.2.1.3.0',
  63. 'CPU_Time_Idle': '.1.3.6.1.4.1.789.1.2.1.5.0',
  64. 'CPU_Context_Switches': '.1.3.6.1.4.1.789.1.2.1.8.0',
  65. 'Disks_Total': '.1.3.6.1.4.1.789.1.6.4.1.0',
  66. 'Disks_Active': '.1.3.6.1.4.1.789.1.6.4.2.0',
  67. 'Disks_Reconstructing': '.1.3.6.1.4.1.789.1.6.4.3.0',
  68. 'Disks_ReconstParity': '.1.3.6.1.4.1.789.1.6.4.4.0',
  69. 'Disks_Scrubbing': '.1.3.6.1.4.1.789.1.6.4.6.0',
  70. 'Disks_Failed': '.1.3.6.1.4.1.789.1.6.4.7.0',
  71. 'Disks_Spare': '.1.3.6.1.4.1.789.1.6.4.8.0',
  72. 'Disks_ZeroDisks': '.1.3.6.1.4.1.789.1.6.4.9.0',
  73. 'Disks_Failed_Descr': '.1.3.6.1.4.1.789.1.6.4.10.0',
  74. 'ExtCache_Type': '.1.3.6.1.4.1.789.1.26.1.0',
  75. 'ExtCache_SubType': '.1.3.6.1.4.1.789.1.26.2.0',
  76. 'ExtCache_Size': '.1.3.6.1.4.1.789.1.26.4.0',
  77. 'ExtCache_Usedsize': '.1.3.6.1.4.1.789.1.26.5.0',
  78. 'ExtCache_Options': '.1.3.6.1.4.1.789.1.26.7.0',
  79. 'ExtCache_Hits': '.1.3.6.1.4.1.789.1.26.8.0',
  80. 'ExtCache_Misses': '.1.3.6.1.4.1.789.1.26.9.0',
  81. 'ExtCache_Inserts': '.1.3.6.1.4.1.789.1.26.10.0',
  82. 'ExtCache_Evicts': '.1.3.6.1.4.1.789.1.26.11.0',
  83. 'ExtCache_Invalidates': '.1.3.6.1.4.1.789.1.26.12.0',
  84. 'ExtCache_MetaData': '.1.3.6.1.4.1.789.1.26.15.0',
  85. 'Global_Status': '.1.3.6.1.4.1.789.1.2.2.4.0',
  86. 'Global_Status_Message': '.1.3.6.1.4.1.789.1.2.2.25.0',
  87. 'Net_ifIndex': '.1.3.6.1.4.1.789.1.22.1.2.1.1',
  88. 'Net_ifDescr': '.1.3.6.1.4.1.789.1.22.1.2.1.2',
  89. 'Net_InBytes': ['.1.3.6.1.4.1.789.1.22.1.2.1.25', '.1.3.6.1.4.1.789.1.22.1.2.1.4', '.1.3.6.1.4.1.789.1.22.1.2.1.3',],
  90. 'Net_OutBytes': ['.1.3.6.1.4.1.789.1.22.1.2.1.31', '.1.3.6.1.4.1.789.1.22.1.2.1.16', '.1.3.6.1.4.1.789.1.22.1.2.1.15',],
  91. 'Net_InDiscards': ['.1.3.6.1.4.1.789.1.22.1.2.1.28', '.1.3.6.1.4.1.789.1.22.1.2.1.10', '.1.3.6.1.4.1.789.1.22.1.2.1.9',],
  92. 'Net_OutDiscards': ['.1.3.6.1.4.1.789.1.22.1.2.1.34', '.1.3.6.1.4.1.789.1.22.1.2.1.22', '.1.3.6.1.4.1.789.1.22.1.2.1.21',],
  93. 'Net_InErrors': ['.1.3.6.1.4.1.789.1.22.1.2.1.29', '.1.3.6.1.4.1.789.1.22.1.2.1.12', '.1.3.6.1.4.1.789.1.22.1.2.1.11',],
  94. 'Net_OutErrors': ['.1.3.6.1.4.1.789.1.22.1.2.1.35', '.1.3.6.1.4.1.789.1.22.1.2.1.24', '.1.3.6.1.4.1.789.1.22.1.2.1.23',],
  95. 'IO_DiskReadBy': ['.1.3.6.1.4.1.789.1.2.2.32.0', '.1.3.6.1.4.1.789.1.2.2.16.0', '.1.3.6.1.4.1.789.1.2.2.15.0',],
  96. 'IO_DiskWriteBy': ['.1.3.6.1.4.1.789.1.2.2.33.0', '.1.3.6.1.4.1.789.1.2.2.18.0', '.1.3.6.1.4.1.789.1.2.2.17.0',],
  97. 'IO_NetInBy': ['.1.3.6.1.4.1.789.1.2.2.30.0', '.1.3.6.1.4.1.789.1.2.2.12.0', '.1.3.6.1.4.1.789.1.2.2.11.0',],
  98. 'IO_NetOutBy': ['.1.3.6.1.4.1.789.1.2.2.31.0', '.1.3.6.1.4.1.789.1.2.2.14.0', '.1.3.6.1.4.1.789.1.2.2.13.0',],
  99. 'IO_TapeReadBy': ['.1.3.6.1.4.1.789.1.2.2.34.0', '.1.3.6.1.4.1.789.1.2.2.20.0', '.1.3.6.1.4.1.789.1.2.2.19.0',],
  100. 'IO_TapeWriteBy': ['.1.3.6.1.4.1.789.1.2.2.35.0', '.1.3.6.1.4.1.789.1.2.2.22.0', '.1.3.6.1.4.1.789.1.2.2.21.0',],
  101. 'IO_FCPReadBy': ['.1.3.6.1.4.1.789.1.17.20.0', '.1.3.6.1.4.1.789.1.17.3.0', '.1.3.6.1.4.1.789.1.17.4.0',],
  102. 'IO_FCPWriteBy': ['.1.3.6.1.4.1.789.1.17.21.0', '.1.3.6.1.4.1.789.1.17.5.0', '.1.3.6.1.4.1.789.1.17.6.0',],
  103. 'IO_iSCSIReadBy': ['.1.3.6.1.4.1.789.1.17.22.0', '.1.3.6.1.4.1.789.1.17.7.0', '.1.3.6.1.4.1.789.1.17.8.0',],
  104. 'IO_iSCSIWriteBy': ['.1.3.6.1.4.1.789.1.17.23.0', '.1.3.6.1.4.1.789.1.17.9.0', '.1.3.6.1.4.1.789.1.17.10.0',],
  105. 'NVRAM_Status': '.1.3.6.1.4.1.789.1.2.5.1.0',
  106. 'OPs_NFS': ['.1.3.6.1.4.1.789.1.2.2.27.0', '.1.3.6.1.4.1.789.1.2.2.6.0', '.1.3.6.1.4.1.789.1.2.2.5.0',],
  107. 'OPs_CIFS': ['.1.3.6.1.4.1.789.1.2.2.28.0', '.1.3.6.1.4.1.789.1.2.2.8.0', '.1.3.6.1.4.1.789.1.2.2.7.0',],
  108. 'OPs_HTTP': ['.1.3.6.1.4.1.789.1.2.2.29.0', '.1.3.6.1.4.1.789.1.2.2.10.0', '.1.3.6.1.4.1.789.1.2.2.9.0',],
  109. 'OPs_FCP': ['.1.3.6.1.4.1.789.1.17.25.0', '.1.3.6.1.4.1.789.1.17.14.0', '.1.3.6.1.4.1.789.1.17.13.0',],
  110. 'OPs_iSCSI': ['.1.3.6.1.4.1.789.1.17.24.0', '.1.3.6.1.4.1.789.1.17.12.0', '.1.3.6.1.4.1.789.1.17.11.0',],
  111. 'Snapmirror_On': '.1.3.6.1.4.1.789.1.9.1.0',
  112. 'Snapmirror_License': '.1.3.6.1.4.1.789.1.9.19.0',
  113. 'Snapmirror_Index': '.1.3.6.1.4.1.789.1.9.20.1.1',
  114. 'Snapmirror_Src': '.1.3.6.1.4.1.789.1.9.20.1.2',
  115. 'Snapmirror_Dst': '.1.3.6.1.4.1.789.1.9.20.1.3',
  116. 'Snapmirror_Status': '.1.3.6.1.4.1.789.1.9.20.1.4',
  117. 'Snapmirror_State': '.1.3.6.1.4.1.789.1.9.20.1.5',
  118. 'Snapmirror_Lag': '.1.3.6.1.4.1.789.1.9.20.1.6',
  119. 'Snapvault_On': '.1.3.6.1.4.1.789.1.19.1.0',
  120. 'Snapvault_LicensePrimary': '.1.3.6.1.4.1.789.1.19.9.0',
  121. 'Snapvault_LicenseSecondary': '.1.3.6.1.4.1.789.1.19.10.0',
  122. 'Snapvault_Index': '.1.3.6.1.4.1.789.1.19.11.1.1',
  123. 'Snapvault_Src': '.1.3.6.1.4.1.789.1.19.11.1.2',
  124. 'Snapvault_Dst': '.1.3.6.1.4.1.789.1.19.11.1.3',
  125. 'Snapvault_Status': '.1.3.6.1.4.1.789.1.19.11.1.4',
  126. 'Snapvault_State': '.1.3.6.1.4.1.789.1.19.11.1.5',
  127. 'Snapvault_Lag': '.1.3.6.1.4.1.789.1.19.11.1.6',
  128. 'Model': '.1.3.6.1.4.1.789.1.1.5.0',
  129. 'ONTAP_Version': '.1.3.6.1.4.1.789.1.1.2.0',
  130. 'df_FS_Index': '.1.3.6.1.4.1.789.1.5.4.1.1',
  131. 'df_FS_Name': '.1.3.6.1.4.1.789.1.5.4.1.2',
  132. 'df_FS_Mounted_On': '.1.3.6.1.4.1.789.1.5.4.1.10',
  133. 'df_FS_Status': '.1.3.6.1.4.1.789.1.5.4.1.20',
  134. 'df_FS_Type': '.1.3.6.1.4.1.789.1.5.4.1.23',
  135. 'df_FS_kBTotal': ['.1.3.6.1.4.1.789.1.5.4.1.29', '.1.3.6.1.4.1.789.1.5.4.1.15', '.1.3.6.1.4.1.789.1.5.4.1.14',],
  136. 'df_FS_kBUsed': ['.1.3.6.1.4.1.789.1.5.4.1.30', '.1.3.6.1.4.1.789.1.5.4.1.17', '.1.3.6.1.4.1.789.1.5.4.1.16',],
  137. 'df_FS_kBAvail': ['.1.3.6.1.4.1.789.1.5.4.1.31', '.1.3.6.1.4.1.789.1.5.4.1.19', '.1.3.6.1.4.1.789.1.5.4.1.18',],
  138. 'df_FS_INodeUsed': '.1.3.6.1.4.1.789.1.5.4.1.7',
  139. 'df_FS_INodeFree': '.1.3.6.1.4.1.789.1.5.4.1.8',
  140. 'df_FS_MaxFilesAvail': '.1.3.6.1.4.1.789.1.5.4.1.11',
  141. 'df_FS_MaxFilesUsed': '.1.3.6.1.4.1.789.1.5.4.1.12',
  142. 'df_FS_MaxFilesPossible': '.1.3.6.1.4.1.789.1.5.4.1.13',
  143. }
  144. OWC = {
  145. 'Cluster_InterconnectStatus': ( (4,), (3,), (1,2,), ),
  146. 'Cluster_Settings': ( (2,), (1,3,4,), (5,), ),
  147. 'Cluster_State': ( (2,4,), (), (1,3,), ),
  148. 'Global_Status': ( (3,), (4,), (5,6), ),
  149. 'NVRAM_Status': ( (1,9), (2,5,8), (3,4,6), ),
  150. 'Snapmirror_State': ( (2,5,), (1,4,6,), (3,), ),
  151. 'Snapvault_State': ( (2,5,7,), (1,3,4,6,), (None,), ),
  152. }
  153. Status2String = {
  154. 'Cluster_CannotTakeOverCause': { '1' : 'ok', '2' : 'unknownReason', '3' : 'disabledByOperator', '4' : 'interconnectOffline', '5' : 'disabledByPartner', '6' : 'takeoverFailed', },
  155. 'Cluster_InterconnectStatus': { '1' : 'notPresent', '2' : 'down', '3' : 'partialFailure', '4' : 'up', },
  156. 'Cluster_Settings': { '1' : 'notConfigured', '2' : 'enabled', '3' : 'disabled', '4' : 'takeoverByPartnerDisabled', '5' : 'thisNodeDead', },
  157. 'Cluster_State': { '1' : 'dead', '2' : 'canTakeover', '3' : 'cannotTakeover', '4' : 'takeover', },
  158. 'CPU_Arch' : { '1' : 'x86', '2' : 'alpha', '3' : 'mips', '4' : 'sparc', '5' : 'amd64', },
  159. 'NVRAM_Status' : { '1' : 'ok', '2' : 'partiallyDischarged', '3' : 'fullyDischarged', '4' : 'notPresent', '5' : 'nearEndOfLife', '6' : 'atEndOfLife', '7' : 'unknown', '8' : 'overCharged', '9' : 'fullyCharged', },
  160. 'df_FS_Status' : { '1' : 'unmounted', '2' : 'mounted', '3' : 'frozen', '4' : 'destroying', '5' : 'creating', '6' : 'mounting', '7' : 'unmounting', '8' : 'nofsinfo', '9' : 'replaying', '10': 'replayed', },
  161. 'df_FS_Type' : { '1' : 'traditionalVolume', '2' : 'flexibleVolume', '3' : 'aggregate', },
  162. 'Snapmirror_Status': { '1' : 'idle', '2' : 'transferring', '3' : 'pending', '4' : 'aborting', '5' : 'migrating', '6' : 'quiescing', '7' : 'resyncing', '8' : 'waiting', '9' : 'syncing', '10': 'in-sync', },
  163. 'Snapmirror_State': { '1' : 'uninitialized', '2' : 'snapmirrored', '3' : 'broken-off', '4' : 'quiesced', '5' : 'source', '6' : 'unknown', },
  164. 'Snapvault_Status': { '1' : 'idle', '2' : 'transferring', '3' : 'pending', '4' : 'aborting', '5' : 'unknown_5', '6' : 'quiescing', '7' : 'resyncing', '8' : 'unknown_8', '9' : 'unknown_9', '10': 'unknown_10', '11': 'unknown_11', '12': 'paused', },
  165. 'Snapvault_State': { '1' : 'uninitialized', '2' : 'snapvaulted', '3' : 'brokenOff', '4' : 'quiesced', '5' : 'source', '6' : 'unknown', '7' : 'restoring', },
  166. }
  167. def map_status_to_returncode(self, value, mapping):
  168. for returncode in xrange(0,3):
  169. if value in mapping[returncode]:
  170. return returncode
  171. return 3
  172. def check_cifs(self, target='', warn='', crit=''):
  173. if target == 'users':
  174. cifsConnectedUsers = long(self.SNMPGET(self.OID['CIFS_Connected_Users']))
  175. returncode = self.value_wc_to_returncode(cifsConnectedUsers, warn, crit)
  176. output = "%s connected users" % cifsConnectedUsers
  177. perfdata = [{'label':'nacifs_users', 'value':cifsConnectedUsers, 'unit':'', 'warn':warn, 'crit':crit, 'min':0},]
  178. elif target == 'stats':
  179. cifsTotalOps = self.SNMPGET(self.OID['CIFS_Total_Ops'])
  180. cifsTotalCalls = self.SNMPGET(self.OID['CIFS_Total_Calls'])
  181. cifsBadCalls = self.SNMPGET(self.OID['CIFS_Bad_Calls'])
  182. cifsGetAttrs = self.SNMPGET(self.OID['CIFS_Get_Attrs'])
  183. cifsReads = self.SNMPGET(self.OID['CIFS_Reads'])
  184. cifsWrites = self.SNMPGET(self.OID['CIFS_Writes'])
  185. cifsLocks = self.SNMPGET(self.OID['CIFS_Locks'])
  186. cifsOpens = self.SNMPGET(self.OID['CIFS_Opens'])
  187. cifsDirOps = self.SNMPGET(self.OID['CIFS_DirOps'])
  188. cifsOthers = self.SNMPGET(self.OID['CIFS_Others'])
  189. returncode = self.RETURNCODE['OK']
  190. output = 'CIFS statistics'
  191. perfdata = []
  192. perfdata.append({'label':'nacifs_totalcalls', 'value':cifsTotalCalls, 'unit':'c', 'min':'0'})
  193. perfdata.append({'label':'nacifs_badcalls', 'value':cifsBadCalls, 'unit':'c', 'min':'0'})
  194. perfdata.append({'label':'nacifs_getattrs', 'value':cifsGetAttrs, 'unit':'c', 'min':'0'})
  195. perfdata.append({'label':'nacifs_reads', 'value':cifsReads, 'unit':'c', 'min':'0'})
  196. perfdata.append({'label':'nacifs_writes', 'value':cifsWrites, 'unit':'c', 'min':'0'})
  197. perfdata.append({'label':'nacifs_locks', 'value':cifsLocks, 'unit':'c', 'min':'0'})
  198. perfdata.append({'label':'nacifs_opens', 'value':cifsOpens, 'unit':'c', 'min':'0'})
  199. perfdata.append({'label':'nacifs_dirops', 'value':cifsDirOps, 'unit':'c', 'min':'0'})
  200. perfdata.append({'label':'nacifs_others', 'value':cifsOthers, 'unit':'c', 'min':'0'})
  201. else:
  202. returncode = self.RETURNCODE['UNKNOWN']
  203. output = 'Unknown CIFS check/target: "%s"' % target
  204. perfdata = []
  205. return self.remember_check('cifs', returncode, output, perfdata=perfdata, target=target)
  206. def check_cp(self):
  207. cp = self.SNMPWALK(self.OID['CP'])
  208. output = 'Consistency Point (in progress %s seconds), Ops: Total(%s) ' % (float(cp[0])/100, cp[7])
  209. output += 'Snapshot(%s) LowWaterMark(%s) HighWaterMark(%s) LogFull(%s) CP-b2b(%s) ' % tuple(cp[2:7])
  210. output += 'Flush(%s) Sync(%s) LowVBuf(%s) CpDeferred(%s) LowDatavecs(%s)' % tuple(cp[8:13])
  211. returncode = self.RETURNCODE['OK']
  212. perfdata = []
  213. perfdata.append({'label':'nacp_progress', 'value':float(cp[0])/100, 'unit':'s'})
  214. perfdata.append({'label':'nacp_total', 'value':cp[7], 'unit':'c'})
  215. perfdata.append({'label':'nacp_snapshot', 'value':cp[2], 'unit':'c'})
  216. perfdata.append({'label':'nacp_lowwatermark', 'value':cp[3], 'unit':'c'})
  217. perfdata.append({'label':'nacp_highwatermark', 'value':cp[4], 'unit':'c'})
  218. perfdata.append({'label':'nacp_logfull', 'value':cp[5], 'unit':'c'})
  219. perfdata.append({'label':'nacp_b2b', 'value':cp[6], 'unit':'c'})
  220. perfdata.append({'label':'nacp_flush', 'value':cp[8], 'unit':'c'})
  221. perfdata.append({'label':'nacp_sync', 'value':cp[9], 'unit':'c'})
  222. perfdata.append({'label':'nacp_lowvbuf', 'value':cp[10], 'unit':'c'})
  223. perfdata.append({'label':'nacp_cpdeferred', 'value':cp[11], 'unit':'c'})
  224. perfdata.append({'label':'nacp_lowdatavecs', 'value':cp[12], 'unit':'c'})
  225. return self.remember_check('cp', returncode, output, perfdata=perfdata)
  226. def check_cpu(self, warn='', crit=''):
  227. cpu_arch = self.SNMPGET(self.OID['CPU_Arch'])
  228. cpu_timebusy = int(self.SNMPGET(self.OID['CPU_Time_Busy']))
  229. # cputimeidle = int(self.SNMPGET(self.OID['CPU_Time_Idle']))
  230. cpu_cs = self.SNMPGET(self.OID['CPU_Context_Switches'])
  231. if '%' in warn:
  232. warn = warn[:-1]
  233. if '%' in crit:
  234. crit = crit[:-1]
  235. returncode = self.value_wc_to_returncode(cpu_timebusy, warn, crit)
  236. output = 'CPU ' + str(cpu_timebusy) + '% busy, CPU architecture: ' + self.Status2String['CPU_Arch'].get(cpu_arch)
  237. perfdata = []
  238. pd = {'label':'nacpu', 'value':cpu_timebusy, 'unit':'%', 'min':0, 'max':100}
  239. if warn:
  240. pd['warn'] = warn
  241. if crit:
  242. pd['crit'] = crit
  243. perfdata.append(pd)
  244. perfdata.append({'label':'nacs', 'value':cpu_cs, 'unit':'c'})
  245. return self.remember_check('cpu', returncode, output, perfdata=perfdata)
  246. def check_cluster(self):
  247. cl_settings = int(self.SNMPGET(self.OID['Cluster_Settings']))
  248. if cl_settings == 1: # notConfigured
  249. return self.remember_check('cluster', self.RETURNCODE['WARNING'], 'No cluster configured!')
  250. cl_state = int(self.SNMPGET(self.OID['Cluster_State']))
  251. cl_interconnectstatus = int(self.SNMPGET(self.OID['Cluster_InterconnectStatus']))
  252. returncode = []
  253. returncode.append(self.map_status_to_returncode(cl_settings, self.OWC['Cluster_Settings']))
  254. returncode.append(self.map_status_to_returncode(cl_state, self.OWC['Cluster_State']))
  255. returncode.append(self.map_status_to_returncode(cl_interconnectstatus, self.OWC['Cluster_InterconnectStatus']))
  256. returncode = max(returncode)
  257. output = 'Settings: ' + self.Status2String['Cluster_Settings'][str(cl_settings)] + ', '
  258. output += 'state: ' + self.Status2String['Cluster_State'][str(cl_state)] + ', '
  259. output += 'interconnect state: ' + self.Status2String['Cluster_InterconnectStatus'][str(cl_interconnectstatus)]
  260. if cl_state == 4: # cannotTakeover
  261. cl_cannottakeovercause = self.SNMPGET(self.OID['Cluster_CannotTakeOverCause'])
  262. output = 'Cannot takeover, reason: ' + self.Status2String['Cluster_CannotTakeOverCause'][cl_cannottakeovercause] + '! ' + output
  263. return self.remember_check('cluster', returncode, output)
  264. def check_disk(self, target='failed', warn='', crit=''):
  265. di_total = int(self.SNMPGET(self.OID['Disks_Total']))
  266. di_active = int(self.SNMPGET(self.OID['Disks_Active']))
  267. di_reconstructing = int(self.SNMPGET(self.OID['Disks_Reconstructing']))
  268. di_reconstparity = int(self.SNMPGET(self.OID['Disks_ReconstParity']))
  269. # di_scrubbing = int(self.SNMPGET(self.OID['Disks_Scrubbing']))
  270. di_failed = int(self.SNMPGET(self.OID['Disks_Failed']))
  271. di_spare = int(self.SNMPGET(self.OID['Disks_Spare']))
  272. # di_zerodisks = int(self.SNMPGET(self.OID['Disks_ZeroDisks']))
  273. di_reconstr = di_reconstructing + di_reconstparity
  274. if target == 'spare':
  275. returncode = self.value_wc_to_returncode(di_spare, warn, crit)
  276. output = str(di_spare) + ' spare disk'
  277. if di_spare > 1:
  278. output += 's'
  279. else:
  280. target = 'failed' # Set to defined value
  281. returncode = self.value_wc_to_returncode(di_failed, warn, crit)
  282. if returncode == 0:
  283. output = 'No failed disks'
  284. else:
  285. output = self.SNMPGET(self.OID['Disks_Failed_Descr'])
  286. perfdata = []
  287. perfdata.append({'label':'nadisk_total', 'value':di_total, 'unit':'', 'min':0})
  288. perfdata.append({'label':'nadisk_active', 'value':di_active, 'unit':'', 'min':0})
  289. pd = {'label':'nadisk_spare', 'value':di_spare, 'unit':'', 'min':0}
  290. if warn and target=='spare':
  291. pd['warn'] = warn
  292. if crit and target=='spare':
  293. pd['crit'] = crit
  294. perfdata.append(pd)
  295. pd = {'label':'nadisk_failed', 'value':di_failed, 'unit':'', 'min':0}
  296. if warn and target=='failed':
  297. pd['warn'] = warn
  298. if crit and target=='failed':
  299. pd['crit'] = crit
  300. perfdata.append(pd)
  301. return self.remember_check('disk', returncode, output, perfdata=perfdata, target=target)
  302. def check_extcache(self):
  303. if self.options.snmpversion in [1, '1']:
  304. return self.remember_check('extcache', self.RETURNCODE['UNKNOWN'], 'Need SNMP v2c/v3 for "extcache" check!',)
  305. ec_size = long(self.SNMPGET(self.OID['ExtCache_Size']))
  306. ec_usedsize = long(self.SNMPGET(self.OID['ExtCache_Usedsize']))
  307. ec_hits = long(self.SNMPGET(self.OID['ExtCache_Hits']))
  308. # ec_meta = long(self.SNMPGET(self.OID['ExtCache_MetaData']))
  309. ec_miss = long(self.SNMPGET(self.OID['ExtCache_Misses']))
  310. ec_evict = long(self.SNMPGET(self.OID['ExtCache_Evicts']))
  311. ec_inval = long(self.SNMPGET(self.OID['ExtCache_Invalidates']))
  312. ec_insert = long(self.SNMPGET(self.OID['ExtCache_Inserts']))
  313. ec_usage = float(ec_usedsize) / float(ec_size) * 100.0
  314. ec_hitpct = float(ec_hits) / float(ec_hits + ec_miss) * 100.0
  315. ec_size_human = self.value_to_human_binary(ec_size, unit='B')
  316. output = 'Cache size %s, cache usage %5.2f%%, total hits %5.2f%% ' % (ec_size_human, ec_usage, ec_hitpct)
  317. returncode = self.RETURNCODE['OK']
  318. perfdata = []
  319. perfdata.append({'label':'nacache_usage', 'value':float('%5.2f' % ec_usage), 'unit':'%'})
  320. perfdata.append({'label':'nacache_hits', 'value':ec_hits, 'unit':'c'})
  321. perfdata.append({'label':'nacache_miss', 'value':ec_miss, 'unit':'c'})
  322. perfdata.append({'label':'nacache_evict', 'value':ec_evict, 'unit':'c'})
  323. perfdata.append({'label':'nacache_inval', 'value':ec_inval, 'unit':'c'})
  324. perfdata.append({'label':'nacache_insert', 'value':ec_insert, 'unit':'c'})
  325. return self.remember_check('extcache', returncode, output, perfdata=perfdata)
  326. def check_extcache_info(self):
  327. ec_type = self.SNMPGET(self.OID['ExtCache_Type'])
  328. ec_subtype = self.SNMPGET(self.OID['ExtCache_SubType'])
  329. ec_size = long(self.SNMPGET(self.OID['ExtCache_Size']))
  330. ec_options = self.SNMPGET(self.OID['ExtCache_Options'])
  331. ec_size_human = self.value_to_human_binary(ec_size, unit='B')
  332. output = 'Cache type: "' + ec_type + '/' + ec_subtype + '", size: ' + ec_size_human + ', options: "' + ec_options + '"'
  333. returncode = 0
  334. return self.remember_check('extcache_info', returncode, output)
  335. def check_global(self):
  336. model = self.SNMPGET(self.OID['Model'])
  337. globalstatus = int(self.SNMPGET(self.OID['Global_Status']))
  338. globalstatusmsg = self.SNMPGET(self.OID['Global_Status_Message'])[:255]
  339. returncode = self.map_status_to_returncode(globalstatus, self.OWC['Global_Status'])
  340. output = model + ': ' + globalstatusmsg
  341. return self.remember_check('global', returncode, output)
  342. def check_ifstat(self, nic):
  343. idx = self.find_in_table(self.OID['Net_ifIndex'], self.OID['Net_ifDescr'] , nic)
  344. if idx == None:
  345. return self.remember_check('ifstat:'+nic, self.RETURNCODE['UNKNOWN'], 'NIC "' + nic + '" not found!')
  346. n_bytes_in = self.SNMPGET(self.OID['Net_InBytes'], idx)
  347. n_bytes_out = self.SNMPGET(self.OID['Net_OutBytes'], idx)
  348. n_discards_in = self.SNMPGET(self.OID['Net_InDiscards'], idx)
  349. n_discards_out = self.SNMPGET(self.OID['Net_OutDiscards'], idx)
  350. n_errors_in = self.SNMPGET(self.OID['Net_InErrors'], idx)
  351. n_errors_out = self.SNMPGET(self.OID['Net_OutErrors'], idx)
  352. returncode = self.RETURNCODE['OK']
  353. output = 'Network statistics for %s' % nic
  354. perfdata = []
  355. perfdata.append({'label':'naifbyin_'+nic, 'value':n_bytes_in, 'unit':'c'})
  356. perfdata.append({'label':'naifbyout_'+nic, 'value':n_bytes_out, 'unit':'c'})
  357. perfdata.append({'label':'naifdiscin_'+nic, 'value':n_discards_in, 'unit':'c'})
  358. perfdata.append({'label':'naifdiscout_'+nic, 'value':n_discards_out, 'unit':'c'})
  359. perfdata.append({'label':'naiferrin_'+nic, 'value':n_errors_in, 'unit':'c'})
  360. perfdata.append({'label':'naiferrout_'+nic, 'value':n_errors_out, 'unit':'c'})
  361. return self.remember_check('ifstat:'+nic, returncode, output, perfdata=perfdata)
  362. def check_io(self):
  363. disk_read = self.SNMPGET(self.OID['IO_DiskReadBy'])
  364. disk_write = self.SNMPGET(self.OID['IO_DiskWriteBy'])
  365. net_in = self.SNMPGET(self.OID['IO_NetInBy'])
  366. net_out = self.SNMPGET(self.OID['IO_NetOutBy'])
  367. tape_read = self.SNMPGET(self.OID['IO_TapeReadBy'])
  368. tape_write = self.SNMPGET(self.OID['IO_TapeWriteBy'])
  369. fcp_read = self.SNMPGET(self.OID['IO_FCPReadBy'])
  370. fcp_write = self.SNMPGET(self.OID['IO_FCPWriteBy'])
  371. iscsi_read = self.SNMPGET(self.OID['IO_iSCSIReadBy'])
  372. iscsi_write = self.SNMPGET(self.OID['IO_iSCSIWriteBy'])
  373. output = 'I/O statistics'
  374. returncode = self.RETURNCODE['OK']
  375. perfdata = []
  376. perfdata.append({'label':'naio_netin', 'value':net_in, 'unit':'c'})
  377. perfdata.append({'label':'naio_netout', 'value':net_out, 'unit':'c'})
  378. perfdata.append({'label':'naio_diskread', 'value':disk_read, 'unit':'c'})
  379. perfdata.append({'label':'naio_diskwrite', 'value':disk_write, 'unit':'c'})
  380. perfdata.append({'label':'naio_taperead', 'value':tape_read, 'unit':'c'})
  381. perfdata.append({'label':'naio_tapewrite', 'value':tape_write, 'unit':'c'})
  382. perfdata.append({'label':'naio_fcpread', 'value':fcp_read, 'unit':'c'})
  383. perfdata.append({'label':'naio_fcpwrite', 'value':fcp_write, 'unit':'c'})
  384. perfdata.append({'label':'naio_iscsiread', 'value':iscsi_read, 'unit':'c'})
  385. perfdata.append({'label':'naio_iscsiwrite', 'value':iscsi_write, 'unit':'c'})
  386. return self.remember_check('io', returncode, output, perfdata=perfdata)
  387. def check_nvram(self):
  388. nvramstatus = int(self.SNMPGET(self.OID['NVRAM_Status']))
  389. returncode = self.map_status_to_returncode(nvramstatus, self.OWC['NVRAM_Status'])
  390. output = 'NVRAM battery status is "' + self.Status2String['NVRAM_Status'].get(str(nvramstatus)) + '"'
  391. return self.remember_check('nvram', returncode, output)
  392. def check_ops(self):
  393. ops_nfs = self.SNMPGET(self.OID['OPs_NFS'])
  394. ops_cifs = self.SNMPGET(self.OID['OPs_CIFS'])
  395. ops_http = self.SNMPGET(self.OID['OPs_HTTP'])
  396. ops_fcp = self.SNMPGET(self.OID['OPs_FCP'])
  397. ops_iscsi = self.SNMPGET(self.OID['OPs_iSCSI'])
  398. output = 'Total ops statistics'
  399. returncode = self.RETURNCODE['OK']
  400. perfdata = []
  401. perfdata.append({'label':'naops_nfs', 'value':ops_nfs, 'unit':'c'})
  402. perfdata.append({'label':'naops_cifs', 'value':ops_cifs, 'unit':'c'})
  403. perfdata.append({'label':'naops_http', 'value':ops_http, 'unit':'c'})
  404. perfdata.append({'label':'naops_fcp', 'value':ops_fcp, 'unit':'c'})
  405. perfdata.append({'label':'naops_iscsi', 'value':ops_iscsi, 'unit':'c'})
  406. return self.remember_check('ops', returncode, output, perfdata=perfdata)
  407. def common_sm_sv(self, what, idx, **kwa):
  408. if not 'lag' in kwa:
  409. kwa['lag'] = long(self.SNMPGET(self.OID[what+'_Lag'], idx)) / 100
  410. if not 'rc_lag' in kwa:
  411. kwa['rc_lag'] = self.value_wc_to_returncode(kwa['lag'], kwa['warn'], kwa['crit'])
  412. if not 'state' in kwa:
  413. kwa['state'] = int(self.SNMPGET(self.OID[what+'_State'], idx))
  414. if not 'rc_state' in kwa:
  415. kwa['rc_state'] = self.map_status_to_returncode(int(kwa['state']), self.OWC[what+'_State'])
  416. if not 'src' in kwa:
  417. kwa['src'] = self.SNMPGET(self.OID[what+'_Src'], idx)
  418. if not 'dst' in kwa:
  419. kwa['dst'] = self.SNMPGET(self.OID[what+'_Dst'], idx)
  420. if not 'status' in kwa:
  421. kwa['status'] = int(self.SNMPGET(self.OID[what+'_Status'], idx))
  422. returncode = self.max_returncode([kwa['rc_state'], kwa['rc_lag']])
  423. if kwa['rc_lag'] in [1,2]:
  424. max_lag = [None, kwa['warn'], kwa['crit']][kwa['rc_lag']]
  425. output = 'Lag too high (%s (%s) > %s (%s))! ' % (kwa['lag'], self.seconds_to_hms(kwa['lag']), max_lag, self.seconds_to_hms(max_lag))
  426. else:
  427. output = ''
  428. output += 'Source: "' + kwa['src'] + '", Destination: "' + kwa['dst'] + '", '
  429. output += 'State: ' + self.Status2String[what+'_State'].get(str(kwa['state'])) + ', '
  430. output += 'Status: ' + self.Status2String[what+'_Status'].get(str(kwa['status']))
  431. return (returncode, output)
  432. def check_sm_sv_all(self, what, target, warn, crit):
  433. verbose = 'VERB' in target
  434. debug = 'DEBUG' in target
  435. tagtarget = what.lower() + ':ALL'
  436. idxs = self.SNMPWALK(self.OID[what+'_Index'])
  437. lags = self.SNMPWALK(self.OID[what+'_Lag'])
  438. for i in xrange(0, len(lags)):
  439. lags[i] = long(lags[i]) / 100
  440. states = self.SNMPWALK(self.OID[what+'_State'])
  441. rcs_lag = []
  442. for lag in lags:
  443. rcs_lag.append(self.value_wc_to_returncode(lag, warn, crit))
  444. rcs_state = []
  445. for state in states:
  446. rcs_state.append(self.map_status_to_returncode(int(state), self.OWC[what+'_State']))
  447. rc_lag = self.max_returncode(rcs_lag)
  448. rc_state = self.max_returncode(rcs_state)
  449. returncode = self.max_returncode([rc_lag, rc_state])
  450. if returncode == self.RETURNCODE['OK'] and not debug:
  451. return self.remember_check(tagtarget, returncode, 'All ' + what + ' OK')
  452. srcs = self.SNMPWALK(self.OID[what+'_Src'])
  453. dsts = self.SNMPWALK(self.OID[what+'_Dst'])
  454. statuss = self.SNMPWALK(self.OID[what+'_Status'])
  455. if not( len(idxs) == len(lags) == len(states) == len(srcs) == len(dsts) == len(statuss) ):
  456. return self.remember_check(tagtarget, returncode, 'Wrong number of status informations, but sure an error!')
  457. output = []
  458. for i in xrange(0, len(idxs)):
  459. if rcs_lag[i] != self.RETURNCODE['OK'] or rcs_state != self.RETURNCODE['OK'] or (verbose or debug):
  460. (rc_thissv, output_thissv) = self.common_sm_sv(what, idxs[i], src=srcs[i], dst=dsts[i], lag=lags[i], state=states[i], status=statuss[i], rc_lag=rcs_lag[i], rc_state=rcs_state[i], warn=warn, crit=crit)
  461. if rc_thissv != self.RETURNCODE['OK'] or debug:
  462. output.append(output_thissv)
  463. output = ' / '.join(output)
  464. return self.remember_check(tagtarget, returncode, output)
  465. def check_sm_sv_one(self, what, target, warn, crit):
  466. idx = self.find_in_table(self.OID[what+'_Index'], self.OID[what+'_Src'], target)
  467. if idx == None:
  468. idx = self.find_in_table(self.OID[what+'_Index'], self.OID[what+'_Dst'], target)
  469. if idx == None:
  470. return self.remember_check(what.lower(), self.RETURNCODE['UNKNOWN'], 'No ' + what + ' with source or destination "' + target + '" found!')
  471. (returncode, output) = self.common_sm_sv(what, idx, warn=warn, crit=crit)
  472. return self.remember_check(what.lower() + ':' + target, returncode, output)
  473. def check_snapmirror(self, target, warn, crit):
  474. if self.SNMPGET(self.OID['Snapmirror_License']) != '2':
  475. return self.remember_check('snapmirror', self.RETURNCODE['CRITICAL'], 'No license for Snapmirror')
  476. if self.SNMPGET(self.OID['Snapmirror_On']) != '2':
  477. return self.remember_check('snapmirror', self.RETURNCODE['CRITICAL'], 'Snapmirror is turned off!')
  478. if not target:
  479. return self.remember_check('snapmirror', self.RETURNCODE['OK'], 'Snapmirror is turned on!')
  480. if target.startswith('ALL'):
  481. return self.check_sm_sv_all('Snapmirror', target, warn, crit)
  482. else:
  483. return self.check_sm_sv_one('Snapmirror', target, warn, crit)
  484. def check_snapvault(self, target, warn, crit):
  485. if self.SNMPGET(self.OID['Snapvault_LicensePrimary']) != '2' and self.SNMPGET(self.OID['Snapvault_LicenseSecondary']) != '2':
  486. return self.remember_check('snapvault', self.RETURNCODE['CRITICAL'], 'No license for Snapvault')
  487. if self.SNMPGET(self.OID['Snapvault_On']) != '2':
  488. return self.remember_check('snapvault', self.RETURNCODE['CRITICAL'], 'Snapvault is turned off!')
  489. if not target:
  490. return self.remember_check('snapvault', self.RETURNCODE['OK'], 'Snapvault is turned on!')
  491. if target.startswith('ALL'):
  492. return self.check_sm_sv_all('Snapvault', target, warn, crit)
  493. else:
  494. return self.check_sm_sv_one('Snapvault', target, warn, crit)
  495. def check_version(self):
  496. model = self.SNMPGET(self.OID['Model'])
  497. ontapversion = self.SNMPGET(self.OID['ONTAP_Version'])
  498. return self.remember_check('version', 0, model + ': ' + ontapversion)
  499. def common_vol_idx(self, volume):
  500. if volume.endswith('.snapshot') or volume.endswith('.snapshot/'):
  501. return (None, None)
  502. idx = self.find_in_table(self.OID['df_FS_Index'], self.OID['df_FS_Name'] , volume)
  503. if idx == None:
  504. # Retry without/with Slash
  505. if volume[-1] == '/':
  506. idx = self.find_in_table(self.OID['df_FS_Index'], self.OID['df_FS_Name'] , volume[:-1])
  507. else:
  508. idx = self.find_in_table(self.OID['df_FS_Index'], self.OID['df_FS_Name'] , volume + '/')
  509. if idx != None:
  510. sn_idx = int(idx) + 1
  511. sn_name = self.SNMPGET(self.OID['df_FS_Name'], sn_idx)
  512. if not (sn_name.endswith('.snapshot') or sn_name.endswith('.snapshot/')):
  513. sn_idx = None
  514. else:
  515. sn_idx = None
  516. return (idx, sn_idx)
  517. def common_vol_shorten_target(self, target):
  518. target = target.replace('/vol/', '')
  519. if target[-1] == "/":
  520. target = target[:-1]
  521. return target
  522. def check_vol_data_one(self, volume, warn, crit):
  523. (idx, sn_idx) = self.common_vol_idx(volume)
  524. if idx == None:
  525. return self.remember_check('vol_data', self.RETURNCODE['UNKNOWN'], '"' + volume + '" not found!')
  526. fs_total = long(self.SNMPGET(self.OID['df_FS_kBTotal'], idx)) * 1024L
  527. fs_used = long(self.SNMPGET(self.OID['df_FS_kBUsed'], idx)) * 1024L
  528. # fs_avail = long(self.SNMPGET(self.OID['df_FS_kBAvail'], idx)) * 1024L
  529. if sn_idx != None:
  530. sn_total = long(self.SNMPGET(self.OID['df_FS_kBTotal'], sn_idx)) * 1024L
  531. sn_used = long(self.SNMPGET(self.OID['df_FS_kBUsed'], sn_idx)) * 1024L
  532. # sn_avail = long(self.SNMPGET(self.OID['df_FS_kBAvail'], sn_idx)) * 1024L
  533. else:
  534. sn_total = 0L
  535. sn_used = 0L
  536. # sn_avail = 0L
  537. mountedon = self.SNMPGET(self.OID['df_FS_Mounted_On'] + "." + idx)
  538. status = self.Status2String['df_FS_Status'].get(self.SNMPGET(self.OID['df_FS_Status'] + "." + idx))
  539. fstype = self.Status2String['df_FS_Type'].get(self.SNMPGET(self.OID['df_FS_Type'] + "." + idx))
  540. fs_pctused = float(fs_used) / float(fs_total) * 100.0
  541. warn = self.range_dehumanize(warn, fs_total, unit=['b',])
  542. crit = self.range_dehumanize(crit, fs_total, unit=['b',])
  543. returncode = self.value_wc_to_returncode(fs_used, warn, crit)
  544. output = volume + ': Used ' + self.value_to_human_binary(fs_used, 'B')
  545. output += ' (' + '%3.1f' % fs_pctused + '%)'+ ' out of ' + self.value_to_human_binary(fs_total, 'B')
  546. target = self.common_vol_shorten_target(volume)
  547. perfdata = []
  548. perfdata.append({'label':'navdu_' + target, 'value':fs_used, 'unit':'B', 'warn':warn, 'crit':crit, 'min':0})
  549. perfdata.append({'label':'navdt_' + target, 'value':fs_total, 'unit':'B'})
  550. perfdata.append({'label':'navsu_' + target, 'value':sn_used, 'unit':'B', 'min':0})
  551. perfdata.append({'label':'navst_' + target, 'value':sn_total, 'unit':'B'})
  552. return self.remember_check('vol_data', returncode, output, perfdata=perfdata, target=target)
  553. def check_vol_data(self, volume, warn, crit):
  554. if volume.startswith('ALL'):
  555. volumes = self.SNMPWALK(self.OID['df_FS_Name'])
  556. for vol in volumes:
  557. if vol.endswith('.snapshot') or vol.endswith('.snapshot/'):
  558. continue
  559. self.check_vol_data_one(vol, warn, crit)
  560. else:
  561. return self.check_vol_data_one(volume, warn, crit)
  562. def check_vol_snap_one(self, volume, warn, crit):
  563. (idx, sn_idx) = self.common_vol_idx(volume)
  564. if idx == None:
  565. return self.remember_check('vol_snap', self.RETURNCODE['UNKNOWN'], '"' + volume + '" not found!')
  566. # fs_total = long(self.SNMPGET(self.OID['df_FS_kBTotal'], idx)) * 1024L
  567. # fs_used = long(self.SNMPGET(self.OID['df_FS_kBUsed'], idx)) * 1024L
  568. # fs_avail = long(self.SNMPGET(self.OID['df_FS_kBAvail'], idx)) * 1024L
  569. if sn_idx != None:
  570. sn_total = long(self.SNMPGET(self.OID['df_FS_kBTotal'], sn_idx)) * 1024L
  571. sn_used = long(self.SNMPGET(self.OID['df_FS_kBUsed'], sn_idx)) * 1024L
  572. # sn_avail = long(self.SNMPGET(self.OID['df_FS_kBAvail'], sn_idx)) * 1024L
  573. else:
  574. sn_total = 0L
  575. sn_used = 0L
  576. # sn_avail = 0L
  577. warn = self.range_dehumanize(warn, sn_total, unit=['b',])
  578. crit = self.range_dehumanize(crit, sn_total, unit=['b',])
  579. if sn_total != 0:
  580. # Snap reserve
  581. sn_pctused = float(sn_used) / float(sn_total) * 100.0
  582. else:
  583. # No snap reserve
  584. sn_pctused = 0.0
  585. returncode = self.value_wc_to_returncode(sn_used, warn, crit)
  586. output = volume + '.snapshot: Used ' + self.value_to_human_binary(sn_used, 'B')
  587. output += ' (' + '%3.1f' % sn_pctused + '%)'+ ' out of ' + self.value_to_human_binary(sn_total, 'B')
  588. target = self.common_vol_shorten_target(volume)
  589. perfdata = []
  590. perfdata.append({'label':'navsu_' + target, 'value':sn_used, 'unit':'B', 'warn':warn, 'crit':crit, 'min':0})
  591. perfdata.append({'label':'navst_' + target, 'value':sn_total, 'unit':'B'})
  592. return self.remember_check('vol_snap', returncode, output, perfdata=perfdata, target=target)
  593. def check_vol_snap(self, volume, warn, crit):
  594. if volume.startswith('ALL'):
  595. volumes = self.SNMPWALK(self.OID['df_FS_Name'])
  596. for vol in volumes:
  597. if vol.endswith('.snapshot') or vol.endswith('.snapshot/'):
  598. continue
  599. self.check_vol_snap_one(vol, warn, crit)
  600. else:
  601. return self.check_vol_snap_one(volume, warn, crit)
  602. def check_vol_inode_one(self, volume, warn, crit):
  603. (idx, sn_idx) = self.common_vol_idx(volume)
  604. if idx == None:
  605. return self.remember_check('vol_inode', self.RETURNCODE['UNKNOWN'], '"' + volume + '" not found!')
  606. in_used = long(self.SNMPGET(self.OID['df_FS_INodeUsed'] + '.' + idx))
  607. in_free = long(self.SNMPGET(self.OID['df_FS_INodeFree'] + '.' + idx))
  608. in_total = in_used + in_free
  609. in_pctused = float(in_used) / float(in_total) * 100.0
  610. warn = self.range_dehumanize(warn, in_total)
  611. crit = self.range_dehumanize(crit, in_total)
  612. returncode = self.value_wc_to_returncode(in_used, warn, crit)
  613. output = volume + ': Used inodes ' + self.value_to_human_si(in_used)
  614. output += ' (' + '%3.1f' % in_pctused + '%)'+ ' out of ' + self.value_to_human_si(in_total)
  615. target = self.common_vol_shorten_target(volume)
  616. perfdata = []
  617. perfdata.append({'label':'naviu_' + target, 'value':in_used, 'unit':None, 'warn':warn, 'crit':crit, 'min':0})
  618. perfdata.append({'label':'navit_' + target, 'value':in_total, 'unit':None})
  619. return self.remember_check('vol_inode', returncode, output, perfdata=perfdata, target=target)
  620. def check_vol_inode(self, volume, warn, crit):
  621. if volume.startswith('ALL'):
  622. volumes = self.SNMPWALK(self.OID['df_FS_Name'])
  623. for vol in volumes:
  624. if vol.endswith('.snapshot') or vol.endswith('.snapshot/'):
  625. continue
  626. self.check_vol_inode_one(vol, warn, crit)
  627. else:
  628. return self.check_vol_inode_one(volume, warn, crit)
  629. def check_vol_files_one(self, volume, warn, crit):
  630. (idx, sn_idx) = self.common_vol_idx(volume)
  631. if idx == None:
  632. return self.remember_check('vol_files', self.RETURNCODE['UNKNOWN'], '"' + volume + '" not found!')
  633. fi_avail = long(self.SNMPGET(self.OID['df_FS_MaxFilesAvail'] + '.' + idx))
  634. fi_used = long(self.SNMPGET(self.OID['df_FS_MaxFilesUsed'] + '.' + idx))
  635. fi_possible = long(self.SNMPGET(self.OID['df_FS_MaxFilesPossible'] + '.' + idx))
  636. fi_total = fi_used + fi_avail
  637. fi_pctused = float(fi_used) / float(fi_total) * 100.0
  638. warn = self.range_dehumanize(warn, fi_total)
  639. crit = self.range_dehumanize(crit, fi_total)
  640. returncode = self.value_wc_to_returncode(fi_used, warn, crit)
  641. output = volume + ': Used files ' + self.value_to_human_si(fi_used)
  642. output += ' (' + '%3.1f' % fi_pctused + '%)'+ ' out of ' + self.value_to_human_si(fi_total)
  643. output += ', may raised to ' + self.value_to_human_si(fi_possible)
  644. target = self.common_vol_shorten_target(volume)
  645. perfdata = []
  646. perfdata.append({'label':'navfu_' + target, 'value':fi_used, 'unit':None, 'warn':warn, 'crit':crit, 'min':0})
  647. perfdata.append({'label':'navft_' + target, 'value':fi_total, 'unit':None})
  648. return self.remember_check('vol_files', returncode, output, perfdata=perfdata, target=target)
  649. def check_vol_files(self, volume, warn, crit):
  650. if volume.startswith('ALL'):
  651. volumes = self.SNMPWALK(self.OID['df_FS_Name'])
  652. for vol in volumes:
  653. if vol.endswith('.snapshot') or vol.endswith('.snapshot/'):
  654. continue
  655. self.check_vol_files_one(vol, warn, crit)
  656. else:
  657. return self.check_vol_files_one(volume, warn, crit)
  658. def main():
  659. plugin = CheckNAF(pluginname='check_naf', tagforstatusline='NAF', description=u'Monitoring NetApp(tm) FAS systems', version='0.9')
  660. plugin.add_cmdlineoption('', '--separator', 'separator', 'Separator for check/target/warn/crit', metavar=',', default=',')
  661. plugin.add_cmdlineoption('', '--subseparator', 'subseparator', 'Separator for multiple checks or targets', metavar='+', default='+')
  662. plugin.add_cmdlineoption('', '--check', 'check', 'OBSOLETE - use new syntax!', default='')
  663. plugin.add_cmdlineoption('', '--target', 'target', 'OBSOLETE - use new syntax!', default='')
  664. plugin.add_cmdlineoption('-w', '', 'warn', 'OBSOLETE - use new syntax!', default='')
  665. plugin.add_cmdlineoption('-c', '', 'crit', 'OBSOLETE - use new syntax!', default='')
  666. plugin.add_cmdlineoption('', '--snmpwalk', 'snmpwalkoid', 'DEBUG: "list" OIDs or SNMPWALK it', default=None)
  667. plugin.parse_cmdlineoptions()
  668. plugin.prepare_snmp()
  669. if plugin.options.snmpwalkoid != None:
  670. if not plugin.options.snmpwalkoid in plugin.OID:
  671. print 'List of OIDs:'
  672. oids = plugin.OID.keys()
  673. oids.sort()
  674. for key in oids:
  675. print '- %s' % key
  676. sys.exit(0)
  677. print 'Walking "%s"...' % plugin.options.snmpwalkoid
  678. result = plugin.SNMPWALK(plugin.OID[plugin.options.snmpwalkoid], exitonerror=False)
  679. if result == None:
  680. result = plugin.SNMPGET(plugin.OID[plugin.options.snmpwalkoid], exitonerror=False)
  681. for value in result:
  682. print '=> %s' % value
  683. sys.exit(0)
  684. if plugin.options.check or plugin.options.target:
  685. arguments = plugin.options.check
  686. for s in [plugin.options.target, plugin.options.warn, plugin.options.crit]:
  687. arguments += plugin.options.separator + s
  688. plugin.back2nagios(3, 'Obsolete syntax - please use new syntax: "%s %s"' % (sys.argv[0], arguments))
  689. checks = []
  690. for quad in plugin.args:
  691. quad = quad.split(plugin.options.separator)
  692. quad = (quad + ['', '', ''])[:4] # Fix length to 4, fill with ''
  693. # Convert list of checks to list
  694. if plugin.options.subseparator in quad[0]:
  695. quad[0] = quad[0].split(plugin.options.subseparator)
  696. else:
  697. quad[0] = [quad[0],]
  698. # Convert list of targets to list
  699. if plugin.options.subseparator in quad[1]:
  700. quad[1] = quad[1].split(plugin.options.subseparator)
  701. else:
  702. quad[1] = [quad[1],]
  703. for target in quad[1]:
  704. for check in quad[0]:
  705. checks.append(tuple([check, target, quad[2], quad[3]]))
  706. if len(checks) == 0:
  707. checks = [('global','','','')]
  708. for quad in checks:
  709. (check, target, warn, crit) = tuple(quad)
  710. if check == 'global' or check == 'environment':
  711. result = plugin.check_global()
  712. elif check == 'cluster':
  713. result = plugin.check_cluster()
  714. elif check == 'cifs':
  715. result = plugin.check_cifs(target=target, warn=warn, crit=crit)
  716. elif check == 'cp':
  717. result = plugin.check_cp()
  718. elif check == 'cpu':
  719. result = plugin.check_cpu(warn=warn, crit=crit)
  720. elif check == 'disk':
  721. result = plugin.check_disk(target=target, warn=warn, crit=crit)
  722. elif check == 'extcache':
  723. result = plugin.check_extcache()
  724. elif check == 'extcache_info':
  725. result = plugin.check_extcache_info()
  726. elif check == 'ifstat':
  727. result = plugin.check_ifstat(target)
  728. elif check == 'io':
  729. result = plugin.check_io()
  730. elif check == 'nvram':
  731. result = plugin.check_nvram()
  732. elif check == 'ops':
  733. result = plugin.check_ops()
  734. elif check == 'snapmirror':
  735. result = plugin.check_snapmirror(target, warn=warn, crit=crit)
  736. elif check == 'snapvault':
  737. result = plugin.check_snapvault(target, warn=warn, crit=crit)
  738. elif check == 'version':
  739. result = plugin.check_version()
  740. elif check == 'vol_data':
  741. result = plugin.check_vol_data(volume=target, warn=warn, crit=crit)
  742. elif check == 'vol_snap':
  743. result = plugin.check_vol_snap(volume=target, warn=warn, crit=crit)
  744. elif check =='vol_inode':
  745. result = plugin.check_vol_inode(volume=target, warn=warn, crit=crit)
  746. elif check =='vol_files':
  747. result = plugin.check_vol_files(volume=target, warn=warn, crit=crit)
  748. else:
  749. result = plugin.remember_check(check, plugin.RETURNCODE['UNKNOWN'], 'Unknown check "' + check + '"!')
  750. # from pprint import pprint
  751. # pprint(plugin.dump_brain())
  752. plugin.brain2output()
  753. plugin.exit()
  754. if __name__ == '__main__':
  755. main()
  756. #vim: ts=4 sw=4 foldmethod=indent