summaryrefslogtreecommitdiffstats
path: root/lib/output.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/output.c')
-rw-r--r--lib/output.c553
1 files changed, 553 insertions, 0 deletions
diff --git a/lib/output.c b/lib/output.c
new file mode 100644
index 00000000..c408a2f5
--- /dev/null
+++ b/lib/output.c
@@ -0,0 +1,553 @@
1#include "./output.h"
2#include "./utils_base.h"
3#include "../plugins/utils.h"
4
5#include <assert.h>
6#include <stdlib.h>
7#include <string.h>
8#include <strings.h>
9// #include <cjson/cJSON.h>
10#include "./vendor/cJSON/cJSON.h"
11#include "perfdata.h"
12#include "states.h"
13
14// == Global variables
15static mp_output_format output_format = MP_FORMAT_DEFAULT;
16static mp_output_detail_level level_of_detail = MP_DETAIL_ALL;
17
18// == Prototypes ==
19static char *fmt_subcheck_output(mp_output_format output_format, mp_subcheck check, unsigned int indentation);
20static inline cJSON *json_serialize_subcheck(mp_subcheck subcheck);
21
22// == Implementation ==
23
24/*
25 * Generate output string for a mp_subcheck object
26 */
27static inline char *fmt_subcheck_perfdata(mp_subcheck check) {
28 char *result = strdup("");
29 int added = 0;
30
31 if (check.perfdata != NULL) {
32 added = asprintf(&result, "%s", pd_list_to_string(*check.perfdata));
33 }
34
35 if (check.subchecks == NULL) {
36 // No subchecks, return here
37 return result;
38 }
39
40 mp_subcheck_list *subchecks = check.subchecks;
41
42 while (subchecks != NULL) {
43 if (added > 0) {
44 added = asprintf(&result, "%s%s", result, fmt_subcheck_perfdata(subchecks->subcheck));
45 } else {
46 // TODO free previous result here?
47 added = asprintf(&result, "%s", fmt_subcheck_perfdata(subchecks->subcheck));
48 }
49
50 subchecks = subchecks->next;
51 }
52
53 return result;
54}
55
56/*
57 * Initialiser for a mp_check object. Always use this to get a new one!
58 * It sets useful defaults
59 */
60mp_check mp_check_init(void) {
61 mp_check check = {0};
62 return check;
63}
64
65/*
66 * Initialiser for a mp_subcheck object. Always use this to get a new one!
67 * It sets useful defaults
68 */
69mp_subcheck mp_subcheck_init(void) {
70 mp_subcheck tmp = {0};
71 tmp.default_state = STATE_UNKNOWN; // Default state is unknown
72 tmp.state_set_explicitly = false;
73 return tmp;
74}
75
76/*
77 * Add a subcheck to a (the one and only) check object
78 */
79int mp_add_subcheck_to_check(mp_check check[static 1], mp_subcheck subcheck) {
80 assert(subcheck.output != NULL); // There must be output in a subcheck
81
82 mp_subcheck_list *tmp = NULL;
83
84 if (check->subchecks == NULL) {
85 check->subchecks = (mp_subcheck_list *)calloc(1, sizeof(mp_subcheck_list));
86 if (check->subchecks == NULL) {
87 die(STATE_UNKNOWN, "%s - %s #%d: %s", __FILE__, __func__, __LINE__, "malloc failed");
88 }
89
90 check->subchecks->subcheck = subcheck;
91 check->subchecks->next = NULL;
92 } else {
93 // Search for the end
94 tmp = (mp_subcheck_list *)calloc(1, sizeof(mp_subcheck_list));
95 if (tmp == NULL) {
96 die(STATE_UNKNOWN, "%s - %s #%d: %s", __FILE__, __func__, __LINE__, "malloc failed");
97 }
98
99 tmp->subcheck = subcheck;
100 tmp->next = check->subchecks;
101
102 check->subchecks = tmp;
103 }
104
105 return 0;
106}
107
108/*
109 * Add a mp_perfdata data point to a mp_subcheck object
110 */
111void mp_add_perfdata_to_subcheck(mp_subcheck check[static 1], const mp_perfdata perfData) {
112 if (check->perfdata == NULL) {
113 check->perfdata = pd_list_init();
114 }
115 pd_list_append(check->perfdata, perfData);
116}
117
118/*
119 * Add a mp_subcheck object to another one. The seconde mp_subcheck (argument) is the lower in the
120 * hierarchy
121 */
122int mp_add_subcheck_to_subcheck(mp_subcheck check[static 1], mp_subcheck subcheck) {
123 if (subcheck.output == NULL) {
124 die(STATE_UNKNOWN, "%s - %s #%d: %s", __FILE__, __func__, __LINE__, "Sub check output is NULL");
125 }
126
127 mp_subcheck_list *tmp = NULL;
128
129 if (check->subchecks == NULL) {
130 check->subchecks = (mp_subcheck_list *)calloc(1, sizeof(mp_subcheck_list));
131 if (check->subchecks == NULL) {
132 die(STATE_UNKNOWN, "%s - %s #%d: %s", __FILE__, __func__, __LINE__, "malloc failed");
133 }
134
135 tmp = check->subchecks;
136 } else {
137 // Search for the end
138 tmp = check->subchecks;
139
140 while (tmp->next != NULL) {
141 tmp = tmp->next;
142 }
143
144 tmp->next = (mp_subcheck_list *)calloc(1, sizeof(mp_subcheck_list));
145 if (tmp->next == NULL) {
146 die(STATE_UNKNOWN, "%s - %s #%d: %s", __FILE__, __func__, __LINE__, "malloc failed");
147 }
148
149 tmp = tmp->next;
150 }
151
152 tmp->subcheck = subcheck;
153
154 return 0;
155}
156
157/*
158 * Add a manual summary to a mp_check object, effectively replacing
159 * the autogenerated one
160 */
161void mp_add_summary(mp_check check[static 1], char *summary) { check->summary = summary; }
162
163/*
164 * Generate the summary string of a mp_check object based on it's subchecks
165 */
166char *get_subcheck_summary(mp_check check) {
167 mp_subcheck_list *subchecks = check.subchecks;
168
169 unsigned int ok = 0;
170 unsigned int warning = 0;
171 unsigned int critical = 0;
172 unsigned int unknown = 0;
173 while (subchecks != NULL) {
174 switch (subchecks->subcheck.state) {
175 case STATE_OK:
176 ok++;
177 break;
178 case STATE_WARNING:
179 warning++;
180 break;
181 case STATE_CRITICAL:
182 critical++;
183 break;
184 case STATE_UNKNOWN:
185 unknown++;
186 break;
187 default:
188 die(STATE_UNKNOWN, "Unknown state in get_subcheck_summary");
189 }
190 subchecks = subchecks->next;
191 }
192 char *result = NULL;
193 asprintf(&result, "ok=%d, warning=%d, critical=%d, unknown=%d", ok, warning, critical, unknown);
194 return result;
195}
196
197/*
198 * Generate the result state of a mp_subcheck object based on it's own state and it's subchecks states
199 */
200mp_state_enum mp_compute_subcheck_state(const mp_subcheck check) {
201 if (check.state_set_explicitly) {
202 return check.state;
203 }
204
205 mp_subcheck_list *scl = check.subchecks;
206
207 if (scl == NULL) {
208 return check.default_state;
209 }
210
211 mp_state_enum result = STATE_OK;
212
213 while (scl != NULL) {
214 result = max_state_alt(result, mp_compute_subcheck_state(scl->subcheck));
215 scl = scl->next;
216 }
217
218 return result;
219}
220
221/*
222 * Generate the result state of a mp_check object based on it's own state and it's subchecks states
223 */
224mp_state_enum mp_compute_check_state(const mp_check check) {
225 assert(check.subchecks != NULL); // a mp_check without subchecks is invalid, die here
226
227 mp_subcheck_list *scl = check.subchecks;
228 mp_state_enum result = STATE_OK;
229
230 while (scl != NULL) {
231 result = max_state_alt(result, mp_compute_subcheck_state(scl->subcheck));
232 scl = scl->next;
233 }
234
235 return result;
236}
237
238/*
239 * Generate output string for a mp_check object
240 * Non static to be available for testing functions
241 */
242char *mp_fmt_output(mp_check check) {
243 char *result = NULL;
244
245 switch (output_format) {
246 case MP_FORMAT_MULTI_LINE: {
247 if (check.summary == NULL) {
248 check.summary = get_subcheck_summary(check);
249 }
250
251 asprintf(&result, "[%s] - %s", state_text(mp_compute_check_state(check)), check.summary);
252
253 mp_subcheck_list *subchecks = check.subchecks;
254
255 while (subchecks != NULL) {
256 if (level_of_detail == MP_DETAIL_ALL || mp_compute_subcheck_state(subchecks->subcheck) != STATE_OK) {
257 asprintf(&result, "%s\n%s", result, fmt_subcheck_output(MP_FORMAT_MULTI_LINE, subchecks->subcheck, 1));
258 }
259 subchecks = subchecks->next;
260 }
261
262 char *pd_string = NULL;
263 subchecks = check.subchecks;
264
265 while (subchecks != NULL) {
266 if (pd_string == NULL) {
267 asprintf(&pd_string, "%s", fmt_subcheck_perfdata(subchecks->subcheck));
268 } else {
269 asprintf(&pd_string, "%s %s", pd_string, fmt_subcheck_perfdata(subchecks->subcheck));
270 }
271
272 subchecks = subchecks->next;
273 }
274
275 if (pd_string != NULL && strlen(pd_string) > 0) {
276 asprintf(&result, "%s|%s", result, pd_string);
277 }
278
279 break;
280 }
281 case MP_FORMAT_TEST_JSON: {
282 cJSON *resultObject = cJSON_CreateObject();
283 if (resultObject == NULL) {
284 die(STATE_UNKNOWN, "cJSON_CreateObject failed");
285 }
286
287 cJSON *resultState = cJSON_CreateString(state_text(mp_compute_check_state(check)));
288 cJSON_AddItemToObject(resultObject, "state", resultState);
289
290 if (check.summary == NULL) {
291 check.summary = get_subcheck_summary(check);
292 }
293
294 cJSON *summary = cJSON_CreateString(check.summary);
295 cJSON_AddItemToObject(resultObject, "summary", summary);
296
297 if (check.subchecks != NULL) {
298 cJSON *subchecks = cJSON_CreateArray();
299
300 mp_subcheck_list *sc = check.subchecks;
301
302 while (sc != NULL) {
303 cJSON *sc_json = json_serialize_subcheck(sc->subcheck);
304 cJSON_AddItemToArray(subchecks, sc_json);
305 sc = sc->next;
306 }
307
308 cJSON_AddItemToObject(resultObject, "checks", subchecks);
309 }
310
311 result = cJSON_PrintUnformatted(resultObject);
312 break;
313 }
314 default:
315 die(STATE_UNKNOWN, "Invalid format");
316 }
317
318 return result;
319}
320
321/*
322 * Helper function to properly indent the output lines when using multiline
323 * formats
324 */
325static char *generate_indentation_string(unsigned int indentation) {
326 char *result = calloc(indentation + 1, sizeof(char));
327
328 for (unsigned int i = 0; i < indentation; i++) {
329 result[i] = '\t';
330 }
331
332 return result;
333}
334
335/*
336 * Helper function to generate the output string of mp_subcheck
337 */
338static inline char *fmt_subcheck_output(mp_output_format output_format, mp_subcheck check, unsigned int indentation) {
339 char *result = NULL;
340 mp_subcheck_list *subchecks = NULL;
341
342 switch (output_format) {
343 case MP_FORMAT_MULTI_LINE:
344 asprintf(&result, "%s\\_[%s] - %s", generate_indentation_string(indentation), state_text(mp_compute_subcheck_state(check)),
345 check.output);
346
347 subchecks = check.subchecks;
348
349 while (subchecks != NULL) {
350 asprintf(&result, "%s\n%s", result, fmt_subcheck_output(output_format, subchecks->subcheck, indentation + 1));
351 subchecks = subchecks->next;
352 }
353 return result;
354 default:
355 die(STATE_UNKNOWN, "Invalid format");
356 }
357}
358
359static inline cJSON *json_serialise_pd_value(mp_perfdata_value value) {
360 cJSON *result = cJSON_CreateObject();
361
362 switch (value.type) {
363 case PD_TYPE_DOUBLE:
364 cJSON_AddStringToObject(result, "type", "double");
365 break;
366 case PD_TYPE_INT:
367 cJSON_AddStringToObject(result, "type", "int");
368 break;
369 case PD_TYPE_UINT:
370 cJSON_AddStringToObject(result, "type", "uint");
371 break;
372 case PD_TYPE_NONE:
373 die(STATE_UNKNOWN, "Perfdata type was None in json_serialise_pd_value");
374 }
375 cJSON_AddStringToObject(result, "value", pd_value_to_string(value));
376
377 return result;
378}
379
380static inline cJSON *json_serialise_range(mp_range range) {
381 cJSON *result = cJSON_CreateObject();
382
383 if (range.alert_on_inside_range) {
384 cJSON_AddBoolToObject(result, "alert_on_inside", true);
385 } else {
386 cJSON_AddBoolToObject(result, "alert_on_inside", false);
387 }
388
389 if (range.end_infinity) {
390 cJSON_AddStringToObject(result, "end", "inf");
391 } else {
392 cJSON_AddItemToObject(result, "end", json_serialise_pd_value(range.end));
393 }
394
395 if (range.start_infinity) {
396 cJSON_AddStringToObject(result, "start", "inf");
397 } else {
398 cJSON_AddItemToObject(result, "start", json_serialise_pd_value(range.end));
399 }
400
401 return result;
402}
403
404static inline cJSON *json_serialise_pd(mp_perfdata pd_val) {
405 cJSON *result = cJSON_CreateObject();
406
407 // Label
408 cJSON_AddStringToObject(result, "label", pd_val.label);
409
410 // Value
411 cJSON_AddItemToObject(result, "value", json_serialise_pd_value(pd_val.value));
412
413 // Uom
414 cJSON_AddStringToObject(result, "uom", pd_val.uom);
415
416 // Warn/Crit
417 if (pd_val.warn_present) {
418 cJSON *warn = json_serialise_range(pd_val.warn);
419 cJSON_AddItemToObject(result, "warn", warn);
420 }
421 if (pd_val.crit_present) {
422 cJSON *crit = json_serialise_range(pd_val.crit);
423 cJSON_AddItemToObject(result, "crit", crit);
424 }
425
426 if (pd_val.min_present) {
427 cJSON_AddItemToObject(result, "min", json_serialise_pd_value(pd_val.min));
428 }
429 if (pd_val.max_present) {
430 cJSON_AddItemToObject(result, "max", json_serialise_pd_value(pd_val.max));
431 }
432
433 return result;
434}
435
436static inline cJSON *json_serialise_pd_list(pd_list *list) {
437 cJSON *result = cJSON_CreateArray();
438
439 do {
440 cJSON *pd_value = json_serialise_pd(list->data);
441 cJSON_AddItemToArray(result, pd_value);
442 list = list->next;
443 } while (list != NULL);
444
445 return result;
446}
447
448static inline cJSON *json_serialize_subcheck(mp_subcheck subcheck) {
449 cJSON *result = cJSON_CreateObject();
450
451 // Human readable output
452 cJSON *output = cJSON_CreateString(subcheck.output);
453 cJSON_AddItemToObject(result, "output", output);
454
455 // Test state (aka Exit Code)
456 cJSON *state = cJSON_CreateString(state_text(mp_compute_subcheck_state(subcheck)));
457 cJSON_AddItemToObject(result, "state", state);
458
459 // Perfdata
460 if (subcheck.perfdata != NULL) {
461 cJSON *perfdata = json_serialise_pd_list(subcheck.perfdata);
462 cJSON_AddItemToObject(result, "perfdata", perfdata);
463 }
464
465 if (subcheck.subchecks != NULL) {
466 cJSON *subchecks = cJSON_CreateArray();
467
468 mp_subcheck_list *sc = subcheck.subchecks;
469
470 while (sc != NULL) {
471 cJSON *sc_json = json_serialize_subcheck(sc->subcheck);
472 cJSON_AddItemToArray(subchecks, sc_json);
473 sc = sc->next;
474 }
475
476 cJSON_AddItemToObject(result, "checks", subchecks);
477 }
478
479 return result;
480}
481
482/*
483 * Wrapper function to print the output string of a mp_check object
484 * Use this in concrete plugins.
485 */
486void mp_print_output(mp_check check) { puts(mp_fmt_output(check)); }
487
488/*
489 * Convenience function to print the output string of a mp_check object and exit
490 * the program with the resulting state.
491 * Intended to be used to exit a monitoring plugin.
492 */
493void mp_exit(mp_check check) {
494 mp_print_output(check);
495 if (output_format == MP_FORMAT_TEST_JSON) {
496 exit(0);
497 }
498
499 exit(mp_compute_check_state(check));
500}
501
502/*
503 * Function to set the result state of a mp_subcheck object explicitly.
504 * This will overwrite the default state AND states derived from it's subchecks
505 */
506mp_subcheck mp_set_subcheck_state(mp_subcheck check, mp_state_enum state) {
507 check.state = state;
508 check.state_set_explicitly = true;
509 return check;
510}
511
512/*
513 * Function to set the default result state of a mp_subcheck object. This state
514 * will be used if neither an explicit state is set (see *mp_set_subcheck_state*)
515 * nor does it include other subchecks
516 */
517mp_subcheck mp_set_subcheck_default_state(mp_subcheck check, mp_state_enum state) {
518 check.default_state = state;
519 return check;
520}
521
522char *mp_output_format_map[] = {
523 [MP_FORMAT_MULTI_LINE] = "multi-line",
524 [MP_FORMAT_TEST_JSON] = "mp-test-json",
525};
526
527/*
528 * Function to parse the output from a string
529 */
530parsed_output_format mp_parse_output_format(char *format_string) {
531 parsed_output_format result = {
532 .parsing_success = false,
533 .output_format = MP_FORMAT_DEFAULT,
534 };
535
536 for (mp_output_format i = 0; i < (sizeof(mp_output_format_map) / sizeof(char *)); i++) {
537 if (strcasecmp(mp_output_format_map[i], format_string) == 0) {
538 result.parsing_success = true;
539 result.output_format = i;
540 break;
541 }
542 }
543
544 return result;
545}
546
547void mp_set_format(mp_output_format format) { output_format = format; }
548
549mp_output_format mp_get_format(void) { return output_format; }
550
551void mp_set_level_of_detail(mp_output_detail_level level) { level_of_detail = level; }
552
553mp_output_detail_level mp_get_level_of_detail(void) { return level_of_detail; }