check_nwstat.c 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892
  1. /******************************************************************************
  2. *
  3. * Program: NetWare statistics plugin for Nagios
  4. * License: GPL
  5. *
  6. * License Information:
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation; either version 2 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program; if not, write to the Free Software
  20. * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  21. *
  22. * $Id$
  23. *
  24. *****************************************************************************/
  25. const char *progname = "check_nwstat";
  26. #define REVISION "$Revision$"
  27. #define COPYRIGHT "Copyright (c) 1999-2001 Ethan Galstad"
  28. #define SUMMARY "\
  29. This plugin attempts to contact the MRTGEXT NLM running on a Novell server\n\
  30. to gather the requested system information.\n"
  31. #define OPTIONS "\
  32. -H host [-v variable] [-w warning] [-c critical]\n\
  33. [-p port] [-t timeout]"
  34. #define LONGOPTIONS "\
  35. -H, --hostname=HOST\n\
  36. Name of the host to check\n\
  37. -v, --variable=STRING\n\
  38. Variable to check. Valid variables include:\n\
  39. LOAD1 = 1 minute average CPU load\n\
  40. LOAD5 = 5 minute average CPU load\n\
  41. LOAD15 = 15 minute average CPU load\n\
  42. CONNS = number of currently licensed connections\n\
  43. VPF<vol> = percent free space on volume <vol>\n\
  44. VKF<vol> = KB of free space on volume <vol>\n\
  45. LTCH = percent long term cache hits\n\
  46. CBUFF = current number of cache buffers\n\
  47. CDBUFF = current number of dirty cache buffers\n\
  48. LRUM = LRU sitting time in minutes\n\
  49. DSDB = check to see if DS Database is open\n\
  50. LOGINS = check to see if logins are enabled\n\
  51. UPRB = used packet receive buffers\n\
  52. PUPRB = percent (of max) used packet receive buffers\n\
  53. SAPENTRIES = number of entries in the SAP table\n\
  54. SAPENTRIES<n> = number of entries in the SAP table for SAP type <n>\n\
  55. OFILES = number of open files\n\
  56. VPP<vol> = percent purgeable space on volume <vol>\n\
  57. VKP<vol> = KB of purgeable space on volume <vol>\n\
  58. VPNP<vol> = percent not yet purgeable space on volume <vol>\n\
  59. VKNP<vol> = KB of not yet purgeable space on volume <vol>\n\
  60. ABENDS = number of abended threads (NW 5.x only)\n\
  61. CSPROCS = number of current service processes (NW 5.x only)\n\
  62. TSYNC = timesync status \n\
  63. LRUS = LRU sitting time in seconds\n\
  64. DCB = dirty cache buffers as a percentage of the total\n\
  65. TCB = dirty cache buffers as a percentage of the original\n\
  66. -w, --warning=INTEGER\n\
  67. Threshold which will result in a warning status\n\
  68. -c, --critical=INTEGER\n\
  69. Threshold which will result in a critical status\n\
  70. -p, --port=INTEGER\n\
  71. Optional port number (default: %d)\n\
  72. -t, --timeout=INTEGER\n\
  73. Seconds before connection attempt times out (default: %d)\n\
  74. -o, --osversion\n\
  75. Include server version string in results\n\
  76. -h, --help\n\
  77. Print this help screen\n\
  78. -V, --version\n\
  79. Print version information\n"
  80. #define DESCRIPTION "\
  81. Notes:\n\
  82. - This plugin requres that the MRTGEXT.NLM file from James Drews' MRTG\n\
  83. extension for NetWare be loaded on the Novell servers you wish to check.\n\
  84. (available from http://www.engr.wisc.edu/~drews/mrtg/)\n\
  85. - Values for critical thresholds should be lower than warning thresholds\n\
  86. when the following variables are checked: VPF, VKF, LTCH, CBUFF, DCB, \n\
  87. TCB, LRUS and LRUM.\n"
  88. #include "config.h"
  89. #include "common.h"
  90. #include "netutils.h"
  91. #include "utils.h"
  92. #define CHECK_NONE 0
  93. #define CHECK_LOAD1 1 /* check 1 minute CPU load */
  94. #define CHECK_LOAD5 2 /* check 5 minute CPU load */
  95. #define CHECK_LOAD15 3 /* check 15 minute CPU load */
  96. #define CHECK_CONNS 4 /* check number of connections */
  97. #define CHECK_VPF 5 /* check % free space on volume */
  98. #define CHECK_VKF 6 /* check KB free space on volume */
  99. #define CHECK_LTCH 7 /* check long-term cache hit percentage */
  100. #define CHECK_CBUFF 8 /* check total cache buffers */
  101. #define CHECK_CDBUFF 9 /* check dirty cache buffers */
  102. #define CHECK_LRUM 10 /* check LRU sitting time in minutes */
  103. #define CHECK_DSDB 11 /* check to see if DS Database is open */
  104. #define CHECK_LOGINS 12 /* check to see if logins are enabled */
  105. #define CHECK_PUPRB 13 /* check % of used packet receive buffers */
  106. #define CHECK_UPRB 14 /* check used packet receive buffers */
  107. #define CHECK_SAPENTRIES 15 /* check SAP entries */
  108. #define CHECK_OFILES 16 /* check number of open files */
  109. #define CHECK_VKP 17 /* check KB purgeable space on volume */
  110. #define CHECK_VPP 18 /* check % purgeable space on volume */
  111. #define CHECK_VKNP 19 /* check KB not yet purgeable space on volume */
  112. #define CHECK_VPNP 20 /* check % not yet purgeable space on volume */
  113. #define CHECK_ABENDS 21 /* check abended thread count */
  114. #define CHECK_CSPROCS 22 /* check number of current service processes */
  115. #define CHECK_TSYNC 23 /* check timesync status 0=no 1=yes in sync to the network */
  116. #define CHECK_LRUS 24 /* check LRU sitting time in seconds */
  117. #define CHECK_DCB 25 /* check dirty cache buffers as a percentage of the total */
  118. #define CHECK_TCB 26 /* check total cache buffers as a percentage of the original */
  119. #define PORT 9999
  120. char *server_address=NULL;
  121. char *volume_name=NULL;
  122. int server_port=PORT;
  123. unsigned long warning_value=0L;
  124. unsigned long critical_value=0L;
  125. int check_warning_value=FALSE;
  126. int check_critical_value=FALSE;
  127. int check_netware_version=FALSE;
  128. unsigned long vars_to_check=CHECK_NONE;
  129. int sap_number=-1;
  130. int process_arguments(int, char **);
  131. void print_usage(void);
  132. void print_help(void);
  133. int main(int argc, char **argv){
  134. int result;
  135. char *send_buffer=NULL;
  136. char recv_buffer[MAX_INPUT_BUFFER];
  137. char *output_message=NULL;
  138. char *temp_buffer=NULL;
  139. char *netware_version=NULL;
  140. int total_cache_buffers=0;
  141. int dirty_cache_buffers=0;
  142. int time_sync_status=0;
  143. int open_files=0;
  144. int abended_threads=0;
  145. int max_service_processes=0;
  146. int current_service_processes=0;
  147. unsigned long free_disk_space=0L;
  148. unsigned long total_disk_space=0L;
  149. unsigned long purgeable_disk_space=0L;
  150. unsigned long non_purgeable_disk_space=0L;
  151. int percent_free_space=0;
  152. int percent_purgeable_space=0;
  153. int percent_non_purgeable_space=0;
  154. unsigned long current_connections=0L;
  155. unsigned long utilization=0L;
  156. int cache_hits=0;
  157. unsigned long cache_buffers=0L;
  158. unsigned long lru_time=0L;
  159. char uptime[MAX_INPUT_BUFFER];
  160. int max_packet_receive_buffers=0;
  161. int used_packet_receive_buffers=0;
  162. unsigned long percent_used_packet_receive_buffers=0L;
  163. int sap_entries=0;
  164. if(process_arguments(argc,argv)==ERROR)
  165. usage("Could not parse arguments\n");
  166. /* initialize alarm signal handling */
  167. signal(SIGALRM,socket_timeout_alarm_handler);
  168. /* set socket timeout */
  169. alarm(socket_timeout);
  170. /* get OS version string */
  171. if (check_netware_version==TRUE) {
  172. send_buffer = strscpy(send_buffer,"S19\r\n");
  173. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  174. if(result!=STATE_OK)
  175. return result;
  176. if(!strcmp(recv_buffer,"-1\n"))
  177. asprintf(&netware_version,"");
  178. else {
  179. recv_buffer[strlen(recv_buffer)-1]=0;
  180. asprintf(&netware_version,"NetWare %s: ",recv_buffer);
  181. }
  182. } else
  183. asprintf(&netware_version,"");
  184. /* check CPU load */
  185. if (vars_to_check==CHECK_LOAD1 || vars_to_check==CHECK_LOAD5 || vars_to_check==CHECK_LOAD15) {
  186. switch(vars_to_check){
  187. case CHECK_LOAD1:
  188. temp_buffer = strscpy(temp_buffer,"1");
  189. break;
  190. case CHECK_LOAD5:
  191. temp_buffer = strscpy(temp_buffer,"5");
  192. break;
  193. default:
  194. temp_buffer = strscpy(temp_buffer,"15");
  195. break;
  196. }
  197. asprintf(&send_buffer,"UTIL%s\r\n",temp_buffer);
  198. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  199. if(result!=STATE_OK)
  200. return result;
  201. utilization=strtoul(recv_buffer,NULL,10);
  202. send_buffer = strscpy(send_buffer,"UPTIME\r\n");
  203. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  204. if(result!=STATE_OK)
  205. return result;
  206. recv_buffer[strlen(recv_buffer)-1]=0;
  207. sprintf(uptime,"Up %s,",recv_buffer);
  208. if(check_critical_value==TRUE && utilization >= critical_value)
  209. result=STATE_CRITICAL;
  210. else if(check_warning_value==TRUE && utilization >= warning_value)
  211. result=STATE_WARNING;
  212. asprintf(&output_message,"Load %s - %s %s-min load average = %lu%%",(result==STATE_OK)?"ok":"problem",uptime,temp_buffer,utilization);
  213. /* check number of user connections */
  214. } else if (vars_to_check==CHECK_CONNS) {
  215. send_buffer = strscpy(send_buffer,"CONNECT\r\n");
  216. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  217. if(result!=STATE_OK)
  218. return result;
  219. current_connections=strtoul(recv_buffer,NULL,10);
  220. if(check_critical_value==TRUE && current_connections >= critical_value)
  221. result=STATE_CRITICAL;
  222. else if(check_warning_value==TRUE && current_connections >= warning_value)
  223. result=STATE_WARNING;
  224. asprintf(&output_message,"Conns %s - %lu current connections",(result==STATE_OK)?"ok":"problem",current_connections);
  225. /* check % long term cache hits */
  226. } else if (vars_to_check==CHECK_LTCH) {
  227. send_buffer = strscpy(send_buffer,"S1\r\n");
  228. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  229. if(result!=STATE_OK)
  230. return result;
  231. cache_hits=atoi(recv_buffer);
  232. if(check_critical_value==TRUE && cache_hits <= critical_value)
  233. result=STATE_CRITICAL;
  234. else if(check_warning_value==TRUE && cache_hits <= warning_value)
  235. result=STATE_WARNING;
  236. asprintf(&output_message,"Long term cache hits = %d%%",cache_hits);
  237. /* check cache buffers */
  238. } else if (vars_to_check==CHECK_CBUFF) {
  239. send_buffer = strscpy(send_buffer,"S2\r\n");
  240. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  241. if(result!=STATE_OK)
  242. return result;
  243. cache_buffers=strtoul(recv_buffer,NULL,10);
  244. if(check_critical_value==TRUE && cache_buffers <= critical_value)
  245. result=STATE_CRITICAL;
  246. else if(check_warning_value==TRUE && cache_buffers <= warning_value)
  247. result=STATE_WARNING;
  248. asprintf(&output_message,"Total cache buffers = %lu",cache_buffers);
  249. /* check dirty cache buffers */
  250. } else if (vars_to_check==CHECK_CDBUFF) {
  251. send_buffer = strscpy(send_buffer,"S3\r\n");
  252. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  253. if(result!=STATE_OK)
  254. return result;
  255. cache_buffers=strtoul(recv_buffer,NULL,10);
  256. if(check_critical_value==TRUE && cache_buffers >= critical_value)
  257. result=STATE_CRITICAL;
  258. else if(check_warning_value==TRUE && cache_buffers >= warning_value)
  259. result=STATE_WARNING;
  260. asprintf(&output_message,"Dirty cache buffers = %lu",cache_buffers);
  261. /* check LRU sitting time in minutes */
  262. } else if (vars_to_check==CHECK_LRUM) {
  263. send_buffer = strscpy(send_buffer,"S5\r\n");
  264. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  265. if(result!=STATE_OK)
  266. return result;
  267. lru_time=strtoul(recv_buffer,NULL,10);
  268. if(check_critical_value==TRUE && lru_time <= critical_value)
  269. result=STATE_CRITICAL;
  270. else if(check_warning_value==TRUE && lru_time <= warning_value)
  271. result=STATE_WARNING;
  272. asprintf(&output_message,"LRU sitting time = %lu minutes",lru_time);
  273. /* check KB free space on volume */
  274. } else if (vars_to_check==CHECK_VKF) {
  275. asprintf(&send_buffer,"VKF%s\r\n",volume_name);
  276. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  277. if(result!=STATE_OK)
  278. return result;
  279. if (!strcmp(recv_buffer,"-1\n")) {
  280. asprintf(&output_message,"Error: Volume '%s' does not exist!",volume_name);
  281. result=STATE_CRITICAL;
  282. } else {
  283. free_disk_space=strtoul(recv_buffer,NULL,10);
  284. if(check_critical_value==TRUE && free_disk_space <= critical_value)
  285. result=STATE_CRITICAL;
  286. else if(check_warning_value==TRUE && free_disk_space <= warning_value)
  287. result=STATE_WARNING;
  288. asprintf(&output_message,"%s%lu KB free on volume %s",(result==STATE_OK)?"":"Only ",free_disk_space,volume_name);
  289. }
  290. /* check % free space on volume */
  291. } else if (vars_to_check==CHECK_VPF) {
  292. asprintf(&send_buffer,"VKF%s\r\n",volume_name);
  293. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  294. if(result!=STATE_OK)
  295. return result;
  296. if(!strcmp(recv_buffer,"-1\n")){
  297. asprintf(&output_message,"Error: Volume '%s' does not exist!",volume_name);
  298. result=STATE_CRITICAL;
  299. } else {
  300. free_disk_space=strtoul(recv_buffer,NULL,10);
  301. asprintf(&send_buffer,"VKS%s\r\n",volume_name);
  302. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  303. if(result!=STATE_OK)
  304. return result;
  305. total_disk_space=strtoul(recv_buffer,NULL,10);
  306. percent_free_space=(int)(((double)free_disk_space/(double)total_disk_space)*100.0);
  307. if(check_critical_value==TRUE && percent_free_space <= critical_value)
  308. result=STATE_CRITICAL;
  309. else if(check_warning_value==TRUE && percent_free_space <= warning_value)
  310. result=STATE_WARNING;
  311. free_disk_space/=1024;
  312. asprintf(&output_message,"%lu MB (%d%%) free on volume %s",free_disk_space,percent_free_space,volume_name);
  313. }
  314. /* check to see if DS Database is open or closed */
  315. } else if(vars_to_check==CHECK_DSDB) {
  316. send_buffer = strscpy(send_buffer,"S11\r\n");
  317. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  318. if(result!=STATE_OK)
  319. return result;
  320. if(atoi(recv_buffer)==1)
  321. result=STATE_OK;
  322. else
  323. result=STATE_WARNING;
  324. send_buffer = strscpy(send_buffer,"S13\r\n");
  325. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  326. temp_buffer=strtok(recv_buffer,"\r\n");
  327. asprintf(&output_message,"Directory Services Database is %s (DS version %s)",(result==STATE_OK)?"open":"closed",temp_buffer);
  328. /* check to see if logins are enabled */
  329. } else if (vars_to_check==CHECK_LOGINS) {
  330. send_buffer = strscpy(send_buffer,"S12\r\n");
  331. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  332. if(result!=STATE_OK)
  333. return result;
  334. if(atoi(recv_buffer)==1)
  335. result=STATE_OK;
  336. else
  337. result=STATE_WARNING;
  338. asprintf(&output_message,"Logins are %s",(result==STATE_OK)?"enabled":"disabled");
  339. /* check packet receive buffers */
  340. } else if (vars_to_check==CHECK_UPRB || vars_to_check==CHECK_PUPRB) {
  341. asprintf(&send_buffer,"S15\r\n",volume_name);
  342. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  343. if(result!=STATE_OK)
  344. return result;
  345. used_packet_receive_buffers=atoi(recv_buffer);
  346. asprintf(&send_buffer,"S16\r\n",volume_name);
  347. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  348. if(result!=STATE_OK)
  349. return result;
  350. max_packet_receive_buffers=atoi(recv_buffer);
  351. percent_used_packet_receive_buffers=(unsigned long)(((double)used_packet_receive_buffers/(double)max_packet_receive_buffers)*100.0);
  352. if(vars_to_check==CHECK_UPRB){
  353. if(check_critical_value==TRUE && used_packet_receive_buffers >= critical_value)
  354. result=STATE_CRITICAL;
  355. else if(check_warning_value==TRUE && used_packet_receive_buffers >= warning_value)
  356. result=STATE_WARNING;
  357. } else {
  358. if(check_critical_value==TRUE && percent_used_packet_receive_buffers >= critical_value)
  359. result=STATE_CRITICAL;
  360. else if(check_warning_value==TRUE && percent_used_packet_receive_buffers >= warning_value)
  361. result=STATE_WARNING;
  362. }
  363. asprintf(&output_message,"%d of %d (%lu%%) packet receive buffers used",used_packet_receive_buffers,max_packet_receive_buffers,percent_used_packet_receive_buffers);
  364. /* check SAP table entries */
  365. } else if (vars_to_check==CHECK_SAPENTRIES) {
  366. if(sap_number==-1)
  367. asprintf(&send_buffer,"S9\r\n");
  368. else
  369. asprintf(&send_buffer,"S9.%d\r\n",sap_number);
  370. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  371. if(result!=STATE_OK)
  372. return result;
  373. sap_entries=atoi(recv_buffer);
  374. if(check_critical_value==TRUE && sap_entries >= critical_value)
  375. result=STATE_CRITICAL;
  376. else if(check_warning_value==TRUE && sap_entries >= warning_value)
  377. result=STATE_WARNING;
  378. if(sap_number==-1)
  379. asprintf(&output_message,"%d entries in SAP table",sap_entries);
  380. else
  381. asprintf(&output_message,"%d entries in SAP table for SAP type %d",sap_entries,sap_number);
  382. /* check KB purgeable space on volume */
  383. } else if (vars_to_check==CHECK_VKP) {
  384. asprintf(&send_buffer,"VKP%s\r\n",volume_name);
  385. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  386. if(result!=STATE_OK)
  387. return result;
  388. if (!strcmp(recv_buffer,"-1\n")) {
  389. asprintf(&output_message,"Error: Volume '%s' does not exist!",volume_name);
  390. result=STATE_CRITICAL;
  391. } else {
  392. purgeable_disk_space=strtoul(recv_buffer,NULL,10);
  393. if(check_critical_value==TRUE && purgeable_disk_space >= critical_value)
  394. result=STATE_CRITICAL;
  395. else if(check_warning_value==TRUE && purgeable_disk_space >= warning_value)
  396. result=STATE_WARNING;
  397. asprintf(&output_message,"%s%lu KB purgeable on volume %s",(result==STATE_OK)?"":"Only ",purgeable_disk_space,volume_name);
  398. }
  399. /* check % purgeable space on volume */
  400. } else if (vars_to_check==CHECK_VPP) {
  401. asprintf(&send_buffer,"VKP%s\r\n",volume_name);
  402. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  403. if(result!=STATE_OK)
  404. return result;
  405. if(!strcmp(recv_buffer,"-1\n")){
  406. asprintf(&output_message,"Error: Volume '%s' does not exist!",volume_name);
  407. result=STATE_CRITICAL;
  408. } else {
  409. purgeable_disk_space=strtoul(recv_buffer,NULL,10);
  410. asprintf(&send_buffer,"VKS%s\r\n",volume_name);
  411. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  412. if(result!=STATE_OK)
  413. return result;
  414. total_disk_space=strtoul(recv_buffer,NULL,10);
  415. percent_purgeable_space=(int)(((double)purgeable_disk_space/(double)total_disk_space)*100.0);
  416. if(check_critical_value==TRUE && percent_purgeable_space >= critical_value)
  417. result=STATE_CRITICAL;
  418. else if(check_warning_value==TRUE && percent_purgeable_space >= warning_value)
  419. result=STATE_WARNING;
  420. purgeable_disk_space/=1024;
  421. asprintf(&output_message,"%lu MB (%d%%) purgeable on volume %s",purgeable_disk_space,percent_purgeable_space,volume_name);
  422. }
  423. /* check KB not yet purgeable space on volume */
  424. } else if (vars_to_check==CHECK_VKNP) {
  425. asprintf(&send_buffer,"VKNP%s\r\n",volume_name);
  426. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  427. if(result!=STATE_OK)
  428. return result;
  429. if (!strcmp(recv_buffer,"-1\n")) {
  430. asprintf(&output_message,"Error: Volume '%s' does not exist!",volume_name);
  431. result=STATE_CRITICAL;
  432. } else {
  433. non_purgeable_disk_space=strtoul(recv_buffer,NULL,10);
  434. if(check_critical_value==TRUE && non_purgeable_disk_space >= critical_value)
  435. result=STATE_CRITICAL;
  436. else if(check_warning_value==TRUE && non_purgeable_disk_space >= warning_value)
  437. result=STATE_WARNING;
  438. asprintf(&output_message,"%s%lu KB not yet purgeable on volume %s",(result==STATE_OK)?"":"Only ",non_purgeable_disk_space,volume_name);
  439. }
  440. /* check % not yet purgeable space on volume */
  441. } else if (vars_to_check==CHECK_VPNP) {
  442. asprintf(&send_buffer,"VKNP%s\r\n",volume_name);
  443. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  444. if(result!=STATE_OK)
  445. return result;
  446. if(!strcmp(recv_buffer,"-1\n")){
  447. asprintf(&output_message,"Error: Volume '%s' does not exist!",volume_name);
  448. result=STATE_CRITICAL;
  449. } else {
  450. non_purgeable_disk_space=strtoul(recv_buffer,NULL,10);
  451. asprintf(&send_buffer,"VKS%s\r\n",volume_name);
  452. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  453. if(result!=STATE_OK)
  454. return result;
  455. total_disk_space=strtoul(recv_buffer,NULL,10);
  456. percent_non_purgeable_space=(int)(((double)non_purgeable_disk_space/(double)total_disk_space)*100.0);
  457. if(check_critical_value==TRUE && percent_non_purgeable_space >= critical_value)
  458. result=STATE_CRITICAL;
  459. else if(check_warning_value==TRUE && percent_non_purgeable_space >= warning_value)
  460. result=STATE_WARNING;
  461. purgeable_disk_space/=1024;
  462. asprintf(&output_message,"%lu MB (%d%%) not yet purgeable on volume %s",non_purgeable_disk_space,percent_non_purgeable_space,volume_name);
  463. }
  464. /* check # of open files */
  465. } else if (vars_to_check==CHECK_OFILES) {
  466. asprintf(&send_buffer,"S18\r\n");
  467. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  468. if(result!=STATE_OK)
  469. return result;
  470. open_files=atoi(recv_buffer);
  471. if(check_critical_value==TRUE && open_files >= critical_value)
  472. result=STATE_CRITICAL;
  473. else if(check_warning_value==TRUE && open_files >= warning_value)
  474. result=STATE_WARNING;
  475. asprintf(&output_message,"%d open files",open_files);
  476. /* check # of abended threads (Netware 5.x only) */
  477. } else if (vars_to_check==CHECK_ABENDS) {
  478. asprintf(&send_buffer,"S17\r\n");
  479. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  480. if(result!=STATE_OK)
  481. return result;
  482. abended_threads=atoi(recv_buffer);
  483. if(check_critical_value==TRUE && abended_threads >= critical_value)
  484. result=STATE_CRITICAL;
  485. else if(check_warning_value==TRUE && abended_threads >= warning_value)
  486. result=STATE_WARNING;
  487. asprintf(&output_message,"%d abended threads",abended_threads);
  488. /* check # of current service processes (Netware 5.x only) */
  489. } else if (vars_to_check==CHECK_CSPROCS) {
  490. asprintf(&send_buffer,"S20\r\n");
  491. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  492. if(result!=STATE_OK)
  493. return result;
  494. max_service_processes=atoi(recv_buffer);
  495. asprintf(&send_buffer,"S21\r\n");
  496. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  497. if(result!=STATE_OK)
  498. return result;
  499. current_service_processes=atoi(recv_buffer);
  500. if(check_critical_value==TRUE && current_service_processes >= critical_value)
  501. result=STATE_CRITICAL;
  502. else if(check_warning_value==TRUE && current_service_processes >= warning_value)
  503. result=STATE_WARNING;
  504. asprintf(&output_message,"%d current service processes (%d max)",current_service_processes,max_service_processes);
  505. /* check # Timesync Status */
  506. } else if (vars_to_check==CHECK_TSYNC) {
  507. asprintf(&send_buffer,"S22\r\n");
  508. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  509. if(result!=STATE_OK)
  510. return result;
  511. time_sync_status=atoi(recv_buffer);
  512. if(time_sync_status==0) {
  513. result=STATE_CRITICAL;
  514. asprintf(&output_message,"Critical: Time not in sync with network!");
  515. }
  516. else {
  517. asprintf(&output_message,"OK! Time in sync with network!");
  518. }
  519. /* check LRU sitting time in secondss */
  520. } else if (vars_to_check==CHECK_LRUS) {
  521. send_buffer = strscpy(send_buffer,"S4\r\n");
  522. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  523. if(result!=STATE_OK)
  524. return result;
  525. lru_time=strtoul(recv_buffer,NULL,10);
  526. if(check_critical_value==TRUE && lru_time <= critical_value)
  527. result=STATE_CRITICAL;
  528. else if(check_warning_value==TRUE && lru_time <= warning_value)
  529. result=STATE_WARNING;
  530. asprintf(&output_message,"LRU sitting time = %lu seconds",lru_time);
  531. /* check % dirty cache buffers as a percentage of the total*/
  532. } else if (vars_to_check==CHECK_DCB) {
  533. send_buffer = strscpy(send_buffer,"S6\r\n");
  534. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  535. if(result!=STATE_OK)
  536. return result;
  537. dirty_cache_buffers=atoi(recv_buffer);
  538. if(check_critical_value==TRUE && dirty_cache_buffers <= critical_value)
  539. result=STATE_CRITICAL;
  540. else if(check_warning_value==TRUE && dirty_cache_buffers <= warning_value)
  541. result=STATE_WARNING;
  542. asprintf(&output_message,"dirty cache buffers = %d%% of the total",dirty_cache_buffers);
  543. /* check % total cache buffers as a percentage of the original*/
  544. } else if (vars_to_check==CHECK_TCB) {
  545. send_buffer = strscpy(send_buffer,"S7\r\n");
  546. result=process_tcp_request(server_address,server_port,send_buffer,recv_buffer,sizeof(recv_buffer));
  547. if(result!=STATE_OK)
  548. return result;
  549. total_cache_buffers=atoi(recv_buffer);
  550. if(check_critical_value==TRUE && total_cache_buffers <= critical_value)
  551. result=STATE_CRITICAL;
  552. else if(check_warning_value==TRUE && total_cache_buffers <= warning_value)
  553. result=STATE_WARNING;
  554. asprintf(&output_message,"total cache buffers = %d%% of the original",total_cache_buffers);
  555. } else {
  556. output_message = strscpy(output_message,"Nothing to check!\n");
  557. result=STATE_UNKNOWN;
  558. }
  559. /* reset timeout */
  560. alarm(0);
  561. printf("%s%s\n",netware_version,output_message);
  562. return result;
  563. }
  564. /* process command-line arguments */
  565. int process_arguments(int argc, char **argv){
  566. int c;
  567. int option_index = 0;
  568. static struct option long_options[] =
  569. {
  570. {"port", required_argument,0,'p'},
  571. {"timeout", required_argument,0,'t'},
  572. {"critical", required_argument,0,'c'},
  573. {"warning", required_argument,0,'w'},
  574. {"variable", required_argument,0,'v'},
  575. {"hostname", required_argument,0,'H'},
  576. {"osversion",no_argument, 0,'o'},
  577. {"version", no_argument, 0,'V'},
  578. {"help", no_argument, 0,'h'},
  579. {0,0,0,0}
  580. };
  581. /* no options were supplied */
  582. if(argc<2) return ERROR;
  583. /* backwards compatibility */
  584. if (! is_option(argv[1])) {
  585. server_address=argv[1];
  586. argv[1]=argv[0];
  587. argv=&argv[1];
  588. argc--;
  589. }
  590. for (c=1;c<argc;c++) {
  591. if(strcmp("-to",argv[c])==0)
  592. strcpy(argv[c],"-t");
  593. else if (strcmp("-wv",argv[c])==0)
  594. strcpy(argv[c],"-w");
  595. else if (strcmp("-cv",argv[c])==0)
  596. strcpy(argv[c],"-c");
  597. }
  598. while (1){
  599. c = getopt_long(argc,argv,"+hoVH:t:c:w:p:v:",long_options,&option_index);
  600. if (c==-1||c==EOF||c==1)
  601. break;
  602. switch (c)
  603. {
  604. case '?': /* print short usage statement if args not parsable */
  605. printf ("%s: Unknown argument: %s\n\n", progname, optarg);
  606. print_usage();
  607. exit(STATE_UNKNOWN);
  608. case 'h': /* help */
  609. print_help();
  610. exit(STATE_OK);
  611. case 'V': /* version */
  612. print_revision(progname,"$Revision$");
  613. exit(STATE_OK);
  614. case 'H': /* hostname */
  615. server_address=optarg;
  616. break;
  617. case 'o': /* display nos version */
  618. check_netware_version=TRUE;
  619. break;
  620. case 'p': /* port */
  621. if (is_intnonneg(optarg))
  622. server_port=atoi(optarg);
  623. else
  624. terminate(STATE_UNKNOWN,"Server port an integer (seconds)\nType '%s -h' for additional help\n",progname);
  625. break;
  626. case 'v':
  627. if(strlen(optarg)<3)
  628. return ERROR;
  629. if(!strcmp(optarg,"LOAD1"))
  630. vars_to_check=CHECK_LOAD1;
  631. else if(!strcmp(optarg,"LOAD5"))
  632. vars_to_check=CHECK_LOAD5;
  633. else if(!strcmp(optarg,"LOAD15"))
  634. vars_to_check=CHECK_LOAD15;
  635. else if(!strcmp(optarg,"CONNS"))
  636. vars_to_check=CHECK_CONNS;
  637. else if(!strcmp(optarg,"LTCH"))
  638. vars_to_check=CHECK_LTCH;
  639. else if(!strcmp(optarg,"DCB"))
  640. vars_to_check=CHECK_DCB;
  641. else if(!strcmp(optarg,"TCB"))
  642. vars_to_check=CHECK_TCB;
  643. else if(!strcmp(optarg,"CBUFF"))
  644. vars_to_check=CHECK_CBUFF;
  645. else if(!strcmp(optarg,"CDBUFF"))
  646. vars_to_check=CHECK_CDBUFF;
  647. else if(!strcmp(optarg,"LRUM"))
  648. vars_to_check=CHECK_LRUM;
  649. else if(!strcmp(optarg,"LRUS"))
  650. vars_to_check=CHECK_LRUS;
  651. else if(strncmp(optarg,"VPF",3)==0){
  652. vars_to_check=CHECK_VPF;
  653. volume_name = strscpy(volume_name,optarg+3);
  654. if(!strcmp(volume_name,""))
  655. volume_name = strscpy(volume_name,"SYS");
  656. }
  657. else if(strncmp(optarg,"VKF",3)==0){
  658. vars_to_check=CHECK_VKF;
  659. volume_name = strscpy(volume_name,optarg+3);
  660. if(!strcmp(volume_name,""))
  661. volume_name = strscpy(volume_name,"SYS");
  662. }
  663. else if(!strcmp(optarg,"DSDB"))
  664. vars_to_check=CHECK_DSDB;
  665. else if(!strcmp(optarg,"LOGINS"))
  666. vars_to_check=CHECK_LOGINS;
  667. else if(!strcmp(optarg,"UPRB"))
  668. vars_to_check=CHECK_UPRB;
  669. else if(!strcmp(optarg,"PUPRB"))
  670. vars_to_check=CHECK_PUPRB;
  671. else if(!strncmp(optarg,"SAPENTRIES",10)){
  672. vars_to_check=CHECK_SAPENTRIES;
  673. if(strlen(optarg)>10)
  674. sap_number=atoi(optarg+10);
  675. else
  676. sap_number=-1;
  677. }
  678. else if(!strcmp(optarg,"OFILES"))
  679. vars_to_check=CHECK_OFILES;
  680. else if(strncmp(optarg,"VKP",3)==0){
  681. vars_to_check=CHECK_VKP;
  682. volume_name = strscpy(volume_name,optarg+3);
  683. if(!strcmp(volume_name,""))
  684. volume_name = strscpy(volume_name,"SYS");
  685. }
  686. else if(strncmp(optarg,"VPP",3)==0){
  687. vars_to_check=CHECK_VPP;
  688. volume_name = strscpy(volume_name,optarg+3);
  689. if(!strcmp(volume_name,""))
  690. volume_name = strscpy(volume_name,"SYS");
  691. }
  692. else if(strncmp(optarg,"VKNP",4)==0){
  693. vars_to_check=CHECK_VKNP;
  694. volume_name = strscpy(volume_name,optarg+4);
  695. if(!strcmp(volume_name,""))
  696. volume_name = strscpy(volume_name,"SYS");
  697. }
  698. else if(strncmp(optarg,"VPNP",4)==0){
  699. vars_to_check=CHECK_VPNP;
  700. volume_name = strscpy(volume_name,optarg+4);
  701. if(!strcmp(volume_name,""))
  702. volume_name = strscpy(volume_name,"SYS");
  703. }
  704. else if(!strcmp(optarg,"ABENDS"))
  705. vars_to_check=CHECK_ABENDS;
  706. else if(!strcmp(optarg,"CSPROCS"))
  707. vars_to_check=CHECK_CSPROCS;
  708. else if(!strcmp(optarg,"TSYNC"))
  709. vars_to_check=CHECK_TSYNC;
  710. else
  711. return ERROR;
  712. break;
  713. case 'w': /* warning threshold */
  714. warning_value=strtoul(optarg,NULL,10);
  715. check_warning_value=TRUE;
  716. break;
  717. case 'c': /* critical threshold */
  718. critical_value=strtoul(optarg,NULL,10);
  719. check_critical_value=TRUE;
  720. break;
  721. case 't': /* timeout */
  722. socket_timeout=atoi(optarg);
  723. if(socket_timeout<=0)
  724. return ERROR;
  725. }
  726. }
  727. return OK;
  728. }
  729. void print_usage(void)
  730. {
  731. printf
  732. ("Usage:\n"
  733. " %s %s\n"
  734. " %s (-h | --help) for detailed help\n"
  735. " %s (-V | --version) for version information\n",
  736. progname, OPTIONS, progname, progname);
  737. }
  738. void print_help(void)
  739. {
  740. print_revision (progname, REVISION);
  741. printf ("%s\n\n%s\n", COPYRIGHT, SUMMARY);
  742. print_usage();
  743. printf
  744. ("\nOptions:\n" LONGOPTIONS "\n" DESCRIPTION "\n",
  745. PORT, DEFAULT_SOCKET_TIMEOUT);
  746. support ();
  747. }