summaryrefslogtreecommitdiffstats
path: root/lib/utils_base.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/utils_base.c')
-rw-r--r--lib/utils_base.c155
1 files changed, 106 insertions, 49 deletions
diff --git a/lib/utils_base.c b/lib/utils_base.c
index 90a4aaa5..c49a473f 100644
--- a/lib/utils_base.c
+++ b/lib/utils_base.c
@@ -55,22 +55,24 @@ void np_init(char *plugin_name, int argc, char **argv) {
55 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 55 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
56 } 56 }
57 this_monitoring_plugin->plugin_name = strdup(plugin_name); 57 this_monitoring_plugin->plugin_name = strdup(plugin_name);
58 if (this_monitoring_plugin->plugin_name == NULL) 58 if (this_monitoring_plugin->plugin_name == NULL) {
59 die(STATE_UNKNOWN, _("Cannot execute strdup: %s"), strerror(errno)); 59 die(STATE_UNKNOWN, _("Cannot execute strdup: %s"), strerror(errno));
60 }
60 this_monitoring_plugin->argc = argc; 61 this_monitoring_plugin->argc = argc;
61 this_monitoring_plugin->argv = argv; 62 this_monitoring_plugin->argv = argv;
62 } 63 }
63} 64}
64 65
65void np_set_args(int argc, char **argv) { 66void np_set_args(int argc, char **argv) {
66 if (this_monitoring_plugin == NULL) 67 if (this_monitoring_plugin == NULL) {
67 die(STATE_UNKNOWN, _("This requires np_init to be called")); 68 die(STATE_UNKNOWN, _("This requires np_init to be called"));
69 }
68 70
69 this_monitoring_plugin->argc = argc; 71 this_monitoring_plugin->argc = argc;
70 this_monitoring_plugin->argv = argv; 72 this_monitoring_plugin->argv = argv;
71} 73}
72 74
73void np_cleanup() { 75void np_cleanup(void) {
74 if (this_monitoring_plugin != NULL) { 76 if (this_monitoring_plugin != NULL) {
75 if (this_monitoring_plugin->state != NULL) { 77 if (this_monitoring_plugin->state != NULL) {
76 if (this_monitoring_plugin->state->state_data) { 78 if (this_monitoring_plugin->state->state_data) {
@@ -162,8 +164,9 @@ range *parse_range_string(char *str) {
162int _set_thresholds(thresholds **my_thresholds, char *warn_string, char *critical_string) { 164int _set_thresholds(thresholds **my_thresholds, char *warn_string, char *critical_string) {
163 thresholds *temp_thresholds = NULL; 165 thresholds *temp_thresholds = NULL;
164 166
165 if ((temp_thresholds = calloc(1, sizeof(thresholds))) == NULL) 167 if ((temp_thresholds = calloc(1, sizeof(thresholds))) == NULL) {
166 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 168 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
169 }
167 170
168 temp_thresholds->warning = NULL; 171 temp_thresholds->warning = NULL;
169 temp_thresholds->critical = NULL; 172 temp_thresholds->critical = NULL;
@@ -215,6 +218,34 @@ void print_thresholds(const char *threshold_name, thresholds *my_threshold) {
215 printf("\n"); 218 printf("\n");
216} 219}
217 220
221/* Returns true if alert should be raised based on the range, false otherwise */
222bool mp_check_range(const mp_perfdata_value value, const mp_range my_range) {
223 bool is_inside = false;
224
225 if (my_range.end_infinity == false && my_range.start_infinity == false) {
226 // range: .........|---inside---|...........
227 // value
228 is_inside = ((cmp_perfdata_value(my_range.start, value) < 1) && (cmp_perfdata_value(value, my_range.end) <= 0));
229 } else if (my_range.start_infinity == false && my_range.end_infinity == true) {
230 // range: .........|---inside---------
231 // value
232 is_inside = (cmp_perfdata_value(my_range.start, value) < 0);
233 } else if (my_range.start_infinity == true && my_range.end_infinity == false) {
234 // range: -inside--------|....................
235 // value
236 is_inside = (cmp_perfdata_value(value, my_range.end) == -1);
237 } else {
238 // range from -inf to inf, so always inside
239 is_inside = true;
240 }
241
242 if ((is_inside && my_range.alert_on_inside_range == INSIDE) || (!is_inside && my_range.alert_on_inside_range == OUTSIDE)) {
243 return true;
244 }
245
246 return false;
247}
248
218/* Returns true if alert should be raised based on the range */ 249/* Returns true if alert should be raised based on the range */
219bool check_range(double value, range *my_range) { 250bool check_range(double value, range *my_range) {
220 bool no = false; 251 bool no = false;
@@ -228,24 +259,24 @@ bool check_range(double value, range *my_range) {
228 if (my_range->end_infinity == false && my_range->start_infinity == false) { 259 if (my_range->end_infinity == false && my_range->start_infinity == false) {
229 if ((my_range->start <= value) && (value <= my_range->end)) { 260 if ((my_range->start <= value) && (value <= my_range->end)) {
230 return no; 261 return no;
231 } else {
232 return yes;
233 } 262 }
234 } else if (my_range->start_infinity == false && my_range->end_infinity == true) { 263 return yes;
264 }
265
266 if (my_range->start_infinity == false && my_range->end_infinity == true) {
235 if (my_range->start <= value) { 267 if (my_range->start <= value) {
236 return no; 268 return no;
237 } else {
238 return yes;
239 } 269 }
240 } else if (my_range->start_infinity == true && my_range->end_infinity == false) { 270 return yes;
271 }
272
273 if (my_range->start_infinity == true && my_range->end_infinity == false) {
241 if (value <= my_range->end) { 274 if (value <= my_range->end) {
242 return no; 275 return no;
243 } else {
244 return yes;
245 } 276 }
246 } else { 277 return yes;
247 return no;
248 } 278 }
279 return no;
249} 280}
250 281
251/* Returns status */ 282/* Returns status */
@@ -265,7 +296,8 @@ int get_status(double value, thresholds *my_thresholds) {
265 296
266char *np_escaped_string(const char *string) { 297char *np_escaped_string(const char *string) {
267 char *data; 298 char *data;
268 int i, j = 0; 299 int i;
300 int j = 0;
269 data = strdup(string); 301 data = strdup(string);
270 for (i = 0; data[i]; i++) { 302 for (i = 0; data[i]; i++) {
271 if (data[i] == '\\') { 303 if (data[i] == '\\') {
@@ -302,7 +334,8 @@ int np_check_if_root(void) { return (geteuid() == 0); }
302 * data strings. 334 * data strings.
303 */ 335 */
304char *np_extract_value(const char *varlist, const char *name, char sep) { 336char *np_extract_value(const char *varlist, const char *name, char sep) {
305 char *tmp = NULL, *value = NULL; 337 char *tmp = NULL;
338 char *value = NULL;
306 int i; 339 int i;
307 340
308 while (1) { 341 while (1) {
@@ -325,15 +358,17 @@ char *np_extract_value(const char *varlist, const char *name, char sep) {
325 358
326 if ((tmp = index(varlist, sep))) { 359 if ((tmp = index(varlist, sep))) {
327 /* Value is delimited by a comma */ 360 /* Value is delimited by a comma */
328 if (tmp - varlist == 0) 361 if (tmp - varlist == 0) {
329 continue; 362 continue;
363 }
330 value = (char *)calloc(1, tmp - varlist + 1); 364 value = (char *)calloc(1, tmp - varlist + 1);
331 strncpy(value, varlist, tmp - varlist); 365 strncpy(value, varlist, tmp - varlist);
332 value[tmp - varlist] = '\0'; 366 value[tmp - varlist] = '\0';
333 } else { 367 } else {
334 /* Value is delimited by a \0 */ 368 /* Value is delimited by a \0 */
335 if (strlen(varlist) == 0) 369 if (strlen(varlist) == 0) {
336 continue; 370 continue;
371 }
337 value = (char *)calloc(1, strlen(varlist) + 1); 372 value = (char *)calloc(1, strlen(varlist) + 1);
338 strncpy(value, varlist, strlen(varlist)); 373 strncpy(value, varlist, strlen(varlist));
339 value[strlen(varlist)] = '\0'; 374 value[strlen(varlist)] = '\0';
@@ -351,9 +386,11 @@ char *np_extract_value(const char *varlist, const char *name, char sep) {
351 } 386 }
352 387
353 /* Clean-up trailing spaces/newlines */ 388 /* Clean-up trailing spaces/newlines */
354 if (value) 389 if (value) {
355 for (i = strlen(value) - 1; isspace(value[i]); i--) 390 for (i = strlen(value) - 1; isspace(value[i]); i--) {
356 value[i] = '\0'; 391 value[i] = '\0';
392 }
393 }
357 394
358 return value; 395 return value;
359} 396}
@@ -378,14 +415,18 @@ const char *state_text(int result) {
378 * return the corresponding STATE_ value or ERROR) 415 * return the corresponding STATE_ value or ERROR)
379 */ 416 */
380int mp_translate_state(char *state_text) { 417int mp_translate_state(char *state_text) {
381 if (!strcasecmp(state_text, "OK") || !strcmp(state_text, "0")) 418 if (!strcasecmp(state_text, "OK") || !strcmp(state_text, "0")) {
382 return STATE_OK; 419 return STATE_OK;
383 if (!strcasecmp(state_text, "WARNING") || !strcmp(state_text, "1")) 420 }
421 if (!strcasecmp(state_text, "WARNING") || !strcmp(state_text, "1")) {
384 return STATE_WARNING; 422 return STATE_WARNING;
385 if (!strcasecmp(state_text, "CRITICAL") || !strcmp(state_text, "2")) 423 }
424 if (!strcasecmp(state_text, "CRITICAL") || !strcmp(state_text, "2")) {
386 return STATE_CRITICAL; 425 return STATE_CRITICAL;
387 if (!strcasecmp(state_text, "UNKNOWN") || !strcmp(state_text, "3")) 426 }
427 if (!strcasecmp(state_text, "UNKNOWN") || !strcmp(state_text, "3")) {
388 return STATE_UNKNOWN; 428 return STATE_UNKNOWN;
429 }
389 return ERROR; 430 return ERROR;
390} 431}
391 432
@@ -394,7 +435,7 @@ int mp_translate_state(char *state_text) {
394 * hopefully a unique key per service/plugin invocation. Use the extra-opts 435 * hopefully a unique key per service/plugin invocation. Use the extra-opts
395 * parse of argv, so that uniqueness in parameters are reflected there. 436 * parse of argv, so that uniqueness in parameters are reflected there.
396 */ 437 */
397char *_np_state_generate_key() { 438char *_np_state_generate_key(void) {
398 int i; 439 int i;
399 char **argv = this_monitoring_plugin->argv; 440 char **argv = this_monitoring_plugin->argv;
400 char keyname[41]; 441 char keyname[41];
@@ -441,7 +482,7 @@ char *_np_state_generate_key() {
441 return p; 482 return p;
442} 483}
443 484
444void _cleanup_state_data() { 485void _cleanup_state_data(void) {
445 if (this_monitoring_plugin->state->state_data != NULL) { 486 if (this_monitoring_plugin->state->state_data != NULL) {
446 np_free(this_monitoring_plugin->state->state_data->data); 487 np_free(this_monitoring_plugin->state->state_data->data);
447 np_free(this_monitoring_plugin->state->state_data); 488 np_free(this_monitoring_plugin->state->state_data);
@@ -453,19 +494,21 @@ void _cleanup_state_data() {
453 * envvar NAGIOS_PLUGIN_STATE_DIRECTORY 494 * envvar NAGIOS_PLUGIN_STATE_DIRECTORY
454 * statically compiled shared state directory 495 * statically compiled shared state directory
455 */ 496 */
456char *_np_state_calculate_location_prefix() { 497char *_np_state_calculate_location_prefix(void) {
457 char *env_dir; 498 char *env_dir;
458 499
459 /* Do not allow passing MP_STATE_PATH in setuid plugins 500 /* Do not allow passing MP_STATE_PATH in setuid plugins
460 * for security reasons */ 501 * for security reasons */
461 if (!mp_suid()) { 502 if (!mp_suid()) {
462 env_dir = getenv("MP_STATE_PATH"); 503 env_dir = getenv("MP_STATE_PATH");
463 if (env_dir && env_dir[0] != '\0') 504 if (env_dir && env_dir[0] != '\0') {
464 return env_dir; 505 return env_dir;
506 }
465 /* This is the former ENV, for backward-compatibility */ 507 /* This is the former ENV, for backward-compatibility */
466 env_dir = getenv("NAGIOS_PLUGIN_STATE_DIRECTORY"); 508 env_dir = getenv("NAGIOS_PLUGIN_STATE_DIRECTORY");
467 if (env_dir && env_dir[0] != '\0') 509 if (env_dir && env_dir[0] != '\0') {
468 return env_dir; 510 return env_dir;
511 }
469 } 512 }
470 513
471 return NP_STATE_DIR_PREFIX; 514 return NP_STATE_DIR_PREFIX;
@@ -483,19 +526,22 @@ void np_enable_state(char *keyname, int expected_data_version) {
483 char *p = NULL; 526 char *p = NULL;
484 int ret; 527 int ret;
485 528
486 if (this_monitoring_plugin == NULL) 529 if (this_monitoring_plugin == NULL) {
487 die(STATE_UNKNOWN, _("This requires np_init to be called")); 530 die(STATE_UNKNOWN, _("This requires np_init to be called"));
531 }
488 532
489 this_state = (state_key *)calloc(1, sizeof(state_key)); 533 this_state = (state_key *)calloc(1, sizeof(state_key));
490 if (this_state == NULL) 534 if (this_state == NULL) {
491 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 535 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
536 }
492 537
493 if (keyname == NULL) { 538 if (keyname == NULL) {
494 temp_keyname = _np_state_generate_key(); 539 temp_keyname = _np_state_generate_key();
495 } else { 540 } else {
496 temp_keyname = strdup(keyname); 541 temp_keyname = strdup(keyname);
497 if (temp_keyname == NULL) 542 if (temp_keyname == NULL) {
498 die(STATE_UNKNOWN, _("Cannot execute strdup: %s"), strerror(errno)); 543 die(STATE_UNKNOWN, _("Cannot execute strdup: %s"), strerror(errno));
544 }
499 } 545 }
500 /* Die if invalid characters used for keyname */ 546 /* Die if invalid characters used for keyname */
501 p = temp_keyname; 547 p = temp_keyname;
@@ -513,8 +559,9 @@ void np_enable_state(char *keyname, int expected_data_version) {
513 /* Calculate filename */ 559 /* Calculate filename */
514 ret = asprintf(&temp_filename, "%s/%lu/%s/%s", _np_state_calculate_location_prefix(), (unsigned long)geteuid(), 560 ret = asprintf(&temp_filename, "%s/%lu/%s/%s", _np_state_calculate_location_prefix(), (unsigned long)geteuid(),
515 this_monitoring_plugin->plugin_name, this_state->name); 561 this_monitoring_plugin->plugin_name, this_state->name);
516 if (ret < 0) 562 if (ret < 0) {
517 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 563 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
564 }
518 565
519 this_state->_filename = temp_filename; 566 this_state->_filename = temp_filename;
520 567
@@ -528,21 +575,23 @@ void np_enable_state(char *keyname, int expected_data_version) {
528 * If numerically lower, then return as no previous state. die with UNKNOWN 575 * If numerically lower, then return as no previous state. die with UNKNOWN
529 * if exceptional error. 576 * if exceptional error.
530 */ 577 */
531state_data *np_state_read() { 578state_data *np_state_read(void) {
532 state_data *this_state_data = NULL; 579 state_data *this_state_data = NULL;
533 FILE *statefile; 580 FILE *statefile;
534 bool rc = false; 581 bool rc = false;
535 582
536 if (this_monitoring_plugin == NULL) 583 if (this_monitoring_plugin == NULL) {
537 die(STATE_UNKNOWN, _("This requires np_init to be called")); 584 die(STATE_UNKNOWN, _("This requires np_init to be called"));
585 }
538 586
539 /* Open file. If this fails, no previous state found */ 587 /* Open file. If this fails, no previous state found */
540 statefile = fopen(this_monitoring_plugin->state->_filename, "r"); 588 statefile = fopen(this_monitoring_plugin->state->_filename, "r");
541 if (statefile != NULL) { 589 if (statefile != NULL) {
542 590
543 this_state_data = (state_data *)calloc(1, sizeof(state_data)); 591 this_state_data = (state_data *)calloc(1, sizeof(state_data));
544 if (this_state_data == NULL) 592 if (this_state_data == NULL) {
545 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 593 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
594 }
546 595
547 this_state_data->data = NULL; 596 this_state_data->data = NULL;
548 this_monitoring_plugin->state->state_data = this_state_data; 597 this_monitoring_plugin->state->state_data = this_state_data;
@@ -581,8 +630,9 @@ bool _np_state_read_file(FILE *f) {
581 630
582 /* Note: This introduces a limit of 1024 bytes in the string data */ 631 /* Note: This introduces a limit of 1024 bytes in the string data */
583 line = (char *)calloc(1, 1024); 632 line = (char *)calloc(1, 1024);
584 if (line == NULL) 633 if (line == NULL) {
585 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 634 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
635 }
586 636
587 while (!failure && (fgets(line, 1024, f)) != NULL) { 637 while (!failure && (fgets(line, 1024, f)) != NULL) {
588 pos = strlen(line); 638 pos = strlen(line);
@@ -590,38 +640,42 @@ bool _np_state_read_file(FILE *f) {
590 line[pos - 1] = '\0'; 640 line[pos - 1] = '\0';
591 } 641 }
592 642
593 if (line[0] == '#') 643 if (line[0] == '#') {
594 continue; 644 continue;
645 }
595 646
596 switch (expected) { 647 switch (expected) {
597 case STATE_FILE_VERSION: 648 case STATE_FILE_VERSION:
598 i = atoi(line); 649 i = atoi(line);
599 if (i != NP_STATE_FORMAT_VERSION) 650 if (i != NP_STATE_FORMAT_VERSION) {
600 failure++; 651 failure++;
601 else 652 } else {
602 expected = STATE_DATA_VERSION; 653 expected = STATE_DATA_VERSION;
654 }
603 break; 655 break;
604 case STATE_DATA_VERSION: 656 case STATE_DATA_VERSION:
605 i = atoi(line); 657 i = atoi(line);
606 if (i != this_monitoring_plugin->state->data_version) 658 if (i != this_monitoring_plugin->state->data_version) {
607 failure++; 659 failure++;
608 else 660 } else {
609 expected = STATE_DATA_TIME; 661 expected = STATE_DATA_TIME;
662 }
610 break; 663 break;
611 case STATE_DATA_TIME: 664 case STATE_DATA_TIME:
612 /* If time > now, error */ 665 /* If time > now, error */
613 data_time = strtoul(line, NULL, 10); 666 data_time = strtoul(line, NULL, 10);
614 if (data_time > current_time) 667 if (data_time > current_time) {
615 failure++; 668 failure++;
616 else { 669 } else {
617 this_monitoring_plugin->state->state_data->time = data_time; 670 this_monitoring_plugin->state->state_data->time = data_time;
618 expected = STATE_DATA_TEXT; 671 expected = STATE_DATA_TEXT;
619 } 672 }
620 break; 673 break;
621 case STATE_DATA_TEXT: 674 case STATE_DATA_TEXT:
622 this_monitoring_plugin->state->state_data->data = strdup(line); 675 this_monitoring_plugin->state->state_data->data = strdup(line);
623 if (this_monitoring_plugin->state->state_data->data == NULL) 676 if (this_monitoring_plugin->state->state_data->data == NULL) {
624 die(STATE_UNKNOWN, _("Cannot execute strdup: %s"), strerror(errno)); 677 die(STATE_UNKNOWN, _("Cannot execute strdup: %s"), strerror(errno));
678 }
625 expected = STATE_DATA_END; 679 expected = STATE_DATA_END;
626 status = true; 680 status = true;
627 break; 681 break;
@@ -648,16 +702,18 @@ void np_state_write_string(time_t data_time, char *data_string) {
648 char *directories = NULL; 702 char *directories = NULL;
649 char *p = NULL; 703 char *p = NULL;
650 704
651 if (data_time == 0) 705 if (data_time == 0) {
652 time(&current_time); 706 time(&current_time);
653 else 707 } else {
654 current_time = data_time; 708 current_time = data_time;
709 }
655 710
656 /* If file doesn't currently exist, create directories */ 711 /* If file doesn't currently exist, create directories */
657 if (access(this_monitoring_plugin->state->_filename, F_OK) != 0) { 712 if (access(this_monitoring_plugin->state->_filename, F_OK) != 0) {
658 result = asprintf(&directories, "%s", this_monitoring_plugin->state->_filename); 713 result = asprintf(&directories, "%s", this_monitoring_plugin->state->_filename);
659 if (result < 0) 714 if (result < 0) {
660 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 715 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
716 }
661 717
662 for (p = directories + 1; *p; p++) { 718 for (p = directories + 1; *p; p++) {
663 if (*p == '/') { 719 if (*p == '/') {
@@ -674,8 +730,9 @@ void np_state_write_string(time_t data_time, char *data_string) {
674 } 730 }
675 731
676 result = asprintf(&temp_file, "%s.XXXXXX", this_monitoring_plugin->state->_filename); 732 result = asprintf(&temp_file, "%s.XXXXXX", this_monitoring_plugin->state->_filename);
677 if (result < 0) 733 if (result < 0) {
678 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); 734 die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno));
735 }
679 736
680 if ((fd = mkstemp(temp_file)) == -1) { 737 if ((fd = mkstemp(temp_file)) == -1) {
681 np_free(temp_file); 738 np_free(temp_file);