diff options
Diffstat (limited to 'plugins/check_disk.d')
| -rw-r--r-- | plugins/check_disk.d/utils_disk.c | 528 | ||||
| -rw-r--r-- | plugins/check_disk.d/utils_disk.h | 160 |
2 files changed, 688 insertions, 0 deletions
diff --git a/plugins/check_disk.d/utils_disk.c b/plugins/check_disk.d/utils_disk.c new file mode 100644 index 00000000..0b89018d --- /dev/null +++ b/plugins/check_disk.d/utils_disk.c | |||
| @@ -0,0 +1,528 @@ | |||
| 1 | /***************************************************************************** | ||
| 2 | * | ||
| 3 | * Library for check_disk | ||
| 4 | * | ||
| 5 | * License: GPL | ||
| 6 | * Copyright (c) 1999-2024 Monitoring Plugins Development Team | ||
| 7 | * | ||
| 8 | * Description: | ||
| 9 | * | ||
| 10 | * This file contains utilities for check_disk. These are tested by libtap | ||
| 11 | * | ||
| 12 | * | ||
| 13 | * This program is free software: you can redistribute it and/or modify | ||
| 14 | * it under the terms of the GNU General Public License as published by | ||
| 15 | * the Free Software Foundation, either version 3 of the License, or | ||
| 16 | * (at your option) any later version. | ||
| 17 | * | ||
| 18 | * This program is distributed in the hope that it will be useful, | ||
| 19 | * but 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 program. If not, see <http://www.gnu.org/licenses/>. | ||
| 25 | * | ||
| 26 | * | ||
| 27 | *****************************************************************************/ | ||
| 28 | |||
| 29 | #include "../common.h" | ||
| 30 | #include "utils_disk.h" | ||
| 31 | #include "../../gl/fsusage.h" | ||
| 32 | #include "../../lib/thresholds.h" | ||
| 33 | #include "../../lib/states.h" | ||
| 34 | #include <stdint.h> | ||
| 35 | #include <stdio.h> | ||
| 36 | #include <string.h> | ||
| 37 | #include <assert.h> | ||
| 38 | |||
| 39 | void np_add_name(struct name_list **list, const char *name) { | ||
| 40 | struct name_list *new_entry; | ||
| 41 | new_entry = (struct name_list *)malloc(sizeof *new_entry); | ||
| 42 | new_entry->name = (char *)name; | ||
| 43 | new_entry->next = *list; | ||
| 44 | *list = new_entry; | ||
| 45 | } | ||
| 46 | |||
| 47 | /* @brief Initialises a new regex at the begin of list via regcomp(3) | ||
| 48 | * | ||
| 49 | * @details if the regex fails to compile the error code of regcomp(3) is returned | ||
| 50 | * and list is not modified, otherwise list is modified to point to the new | ||
| 51 | * element | ||
| 52 | * @param list Pointer to a linked list of regex_list elements | ||
| 53 | * @param regex the string containing the regex which should be inserted into the list | ||
| 54 | * @param clags the cflags parameter for regcomp(3) | ||
| 55 | */ | ||
| 56 | int np_add_regex(struct regex_list **list, const char *regex, int cflags) { | ||
| 57 | struct regex_list *new_entry = (struct regex_list *)malloc(sizeof *new_entry); | ||
| 58 | |||
| 59 | if (new_entry == NULL) { | ||
| 60 | die(STATE_UNKNOWN, _("Cannot allocate memory: %s"), strerror(errno)); | ||
| 61 | } | ||
| 62 | |||
| 63 | int regcomp_result = regcomp(&new_entry->regex, regex, cflags); | ||
| 64 | |||
| 65 | if (!regcomp_result) { | ||
| 66 | // regcomp succeeded | ||
| 67 | new_entry->next = *list; | ||
| 68 | *list = new_entry; | ||
| 69 | |||
| 70 | return 0; | ||
| 71 | } | ||
| 72 | // regcomp failed | ||
| 73 | free(new_entry); | ||
| 74 | |||
| 75 | return regcomp_result; | ||
| 76 | } | ||
| 77 | |||
| 78 | parameter_list_elem parameter_list_init(const char *name) { | ||
| 79 | parameter_list_elem result = { | ||
| 80 | .name = strdup(name), | ||
| 81 | .best_match = NULL, | ||
| 82 | |||
| 83 | .freespace_units = mp_thresholds_init(), | ||
| 84 | .freespace_percent = mp_thresholds_init(), | ||
| 85 | .freeinodes_percent = mp_thresholds_init(), | ||
| 86 | |||
| 87 | .group = NULL, | ||
| 88 | |||
| 89 | .inodes_total = 0, | ||
| 90 | .inodes_free = 0, | ||
| 91 | .inodes_free_to_root = 0, | ||
| 92 | .inodes_used = 0, | ||
| 93 | |||
| 94 | .used_bytes = 0, | ||
| 95 | .free_bytes = 0, | ||
| 96 | .total_bytes = 0, | ||
| 97 | |||
| 98 | .next = NULL, | ||
| 99 | .prev = NULL, | ||
| 100 | }; | ||
| 101 | return result; | ||
| 102 | } | ||
| 103 | |||
| 104 | /* Returns true if name is in list */ | ||
| 105 | bool np_find_name(struct name_list *list, const char *name) { | ||
| 106 | if (list == NULL || name == NULL) { | ||
| 107 | return false; | ||
| 108 | } | ||
| 109 | for (struct name_list *iterator = list; iterator; iterator = iterator->next) { | ||
| 110 | if (!strcmp(name, iterator->name)) { | ||
| 111 | return true; | ||
| 112 | } | ||
| 113 | } | ||
| 114 | return false; | ||
| 115 | } | ||
| 116 | |||
| 117 | /* Returns true if name is in list */ | ||
| 118 | bool np_find_regmatch(struct regex_list *list, const char *name) { | ||
| 119 | if (name == NULL) { | ||
| 120 | return false; | ||
| 121 | } | ||
| 122 | |||
| 123 | size_t len = strlen(name); | ||
| 124 | |||
| 125 | for (; list; list = list->next) { | ||
| 126 | /* Emulate a full match as if surrounded with ^( )$ | ||
| 127 | by checking whether the match spans the whole name */ | ||
| 128 | regmatch_t dummy_match; | ||
| 129 | if (!regexec(&list->regex, name, 1, &dummy_match, 0) && dummy_match.rm_so == 0 && | ||
| 130 | dummy_match.rm_eo == len) { | ||
| 131 | return true; | ||
| 132 | } | ||
| 133 | } | ||
| 134 | |||
| 135 | return false; | ||
| 136 | } | ||
| 137 | |||
| 138 | bool np_seen_name(struct name_list *list, const char *name) { | ||
| 139 | for (struct name_list *iterator = list; iterator; iterator = iterator->next) { | ||
| 140 | if (!strcmp(iterator->name, name)) { | ||
| 141 | return true; | ||
| 142 | } | ||
| 143 | } | ||
| 144 | return false; | ||
| 145 | } | ||
| 146 | |||
| 147 | bool np_regex_match_mount_entry(struct mount_entry *me, regex_t *re) { | ||
| 148 | return ((regexec(re, me->me_devname, (size_t)0, NULL, 0) == 0) || | ||
| 149 | (regexec(re, me->me_mountdir, (size_t)0, NULL, 0) == 0)); | ||
| 150 | } | ||
| 151 | |||
| 152 | check_disk_config check_disk_config_init() { | ||
| 153 | check_disk_config tmp = { | ||
| 154 | .erronly = false, | ||
| 155 | .display_mntp = false, | ||
| 156 | .show_local_fs = false, | ||
| 157 | .stat_remote_fs = false, | ||
| 158 | .display_inodes_perfdata = false, | ||
| 159 | |||
| 160 | .exact_match = false, | ||
| 161 | .freespace_ignore_reserved = false, | ||
| 162 | |||
| 163 | .ignore_missing = false, | ||
| 164 | .path_ignored = false, | ||
| 165 | |||
| 166 | // FS Filters | ||
| 167 | .fs_exclude_list = NULL, | ||
| 168 | .fs_include_list = NULL, | ||
| 169 | .device_path_exclude_list = NULL, | ||
| 170 | |||
| 171 | // Actual filesystems paths to investigate | ||
| 172 | .path_select_list = filesystem_list_init(), | ||
| 173 | |||
| 174 | .mount_list = NULL, | ||
| 175 | .seen = NULL, | ||
| 176 | |||
| 177 | .display_unit = Humanized, | ||
| 178 | // .unit = MebiBytes, | ||
| 179 | |||
| 180 | .output_format_is_set = false, | ||
| 181 | }; | ||
| 182 | return tmp; | ||
| 183 | } | ||
| 184 | |||
| 185 | char *get_unit_string(byte_unit_enum unit) { | ||
| 186 | switch (unit) { | ||
| 187 | case Bytes: | ||
| 188 | return "Bytes"; | ||
| 189 | case KibiBytes: | ||
| 190 | return "KiB"; | ||
| 191 | case MebiBytes: | ||
| 192 | return "MiB"; | ||
| 193 | case GibiBytes: | ||
| 194 | return "GiB"; | ||
| 195 | case TebiBytes: | ||
| 196 | return "TiB"; | ||
| 197 | case PebiBytes: | ||
| 198 | return "PiB"; | ||
| 199 | case ExbiBytes: | ||
| 200 | return "EiB"; | ||
| 201 | case KiloBytes: | ||
| 202 | return "KB"; | ||
| 203 | case MegaBytes: | ||
| 204 | return "MB"; | ||
| 205 | case GigaBytes: | ||
| 206 | return "GB"; | ||
| 207 | case TeraBytes: | ||
| 208 | return "TB"; | ||
| 209 | case PetaBytes: | ||
| 210 | return "PB"; | ||
| 211 | case ExaBytes: | ||
| 212 | return "EB"; | ||
| 213 | default: | ||
| 214 | assert(false); | ||
| 215 | } | ||
| 216 | } | ||
| 217 | |||
| 218 | measurement_unit measurement_unit_init() { | ||
| 219 | measurement_unit tmp = { | ||
| 220 | .name = NULL, | ||
| 221 | .filesystem_type = NULL, | ||
| 222 | .is_group = false, | ||
| 223 | |||
| 224 | .freeinodes_percent_thresholds = mp_thresholds_init(), | ||
| 225 | .freespace_percent_thresholds = mp_thresholds_init(), | ||
| 226 | .freespace_bytes_thresholds = mp_thresholds_init(), | ||
| 227 | |||
| 228 | .free_bytes = 0, | ||
| 229 | .used_bytes = 0, | ||
| 230 | .total_bytes = 0, | ||
| 231 | |||
| 232 | .inodes_total = 0, | ||
| 233 | .inodes_free = 0, | ||
| 234 | .inodes_free_to_root = 0, | ||
| 235 | .inodes_used = 0, | ||
| 236 | }; | ||
| 237 | return tmp; | ||
| 238 | } | ||
| 239 | |||
| 240 | // Add a given element to the list, memory for the new element is freshly allocated | ||
| 241 | // Returns a pointer to new element | ||
| 242 | measurement_unit_list *add_measurement_list(measurement_unit_list *list, measurement_unit elem) { | ||
| 243 | // find last element | ||
| 244 | measurement_unit_list *new = NULL; | ||
| 245 | if (list == NULL) { | ||
| 246 | new = calloc(1, sizeof(measurement_unit_list)); | ||
| 247 | if (new == NULL) { | ||
| 248 | die(STATE_UNKNOWN, _("allocation failed")); | ||
| 249 | } | ||
| 250 | } else { | ||
| 251 | measurement_unit_list *list_elem = list; | ||
| 252 | while (list_elem->next != NULL) { | ||
| 253 | list_elem = list_elem->next; | ||
| 254 | } | ||
| 255 | |||
| 256 | new = calloc(1, sizeof(measurement_unit_list)); | ||
| 257 | if (new == NULL) { | ||
| 258 | die(STATE_UNKNOWN, _("allocation failed")); | ||
| 259 | } | ||
| 260 | |||
| 261 | list_elem->next = new; | ||
| 262 | } | ||
| 263 | |||
| 264 | new->unit = elem; | ||
| 265 | new->next = NULL; | ||
| 266 | return new; | ||
| 267 | } | ||
| 268 | |||
| 269 | measurement_unit add_filesystem_to_measurement_unit(measurement_unit unit, | ||
| 270 | parameter_list_elem filesystem) { | ||
| 271 | |||
| 272 | unit.free_bytes += filesystem.free_bytes; | ||
| 273 | unit.used_bytes += filesystem.used_bytes; | ||
| 274 | unit.total_bytes += filesystem.total_bytes; | ||
| 275 | |||
| 276 | unit.inodes_total += filesystem.inodes_total; | ||
| 277 | unit.inodes_free += filesystem.inodes_free; | ||
| 278 | unit.inodes_free_to_root += filesystem.inodes_free_to_root; | ||
| 279 | unit.inodes_used += filesystem.inodes_used; | ||
| 280 | return unit; | ||
| 281 | } | ||
| 282 | |||
| 283 | measurement_unit create_measurement_unit_from_filesystem(parameter_list_elem filesystem, | ||
| 284 | bool display_mntp) { | ||
| 285 | measurement_unit result = measurement_unit_init(); | ||
| 286 | if (!display_mntp) { | ||
| 287 | result.name = strdup(filesystem.best_match->me_mountdir); | ||
| 288 | } else { | ||
| 289 | result.name = strdup(filesystem.best_match->me_devname); | ||
| 290 | } | ||
| 291 | |||
| 292 | if (filesystem.group) { | ||
| 293 | result.is_group = true; | ||
| 294 | } else { | ||
| 295 | result.is_group = false; | ||
| 296 | if (filesystem.best_match) { | ||
| 297 | result.filesystem_type = filesystem.best_match->me_type; | ||
| 298 | } | ||
| 299 | } | ||
| 300 | |||
| 301 | result.freeinodes_percent_thresholds = filesystem.freeinodes_percent; | ||
| 302 | result.freespace_percent_thresholds = filesystem.freespace_percent; | ||
| 303 | result.freespace_bytes_thresholds = filesystem.freespace_units; | ||
| 304 | result.free_bytes = filesystem.free_bytes; | ||
| 305 | result.total_bytes = filesystem.total_bytes; | ||
| 306 | result.used_bytes = filesystem.used_bytes; | ||
| 307 | result.inodes_total = filesystem.inodes_total; | ||
| 308 | result.inodes_used = filesystem.inodes_used; | ||
| 309 | result.inodes_free = filesystem.inodes_free; | ||
| 310 | result.inodes_free_to_root = filesystem.inodes_free_to_root; | ||
| 311 | return result; | ||
| 312 | } | ||
| 313 | |||
| 314 | #define RANDOM_STRING_LENGTH 64 | ||
| 315 | |||
| 316 | char *humanize_byte_value(unsigned long long value, bool use_si_units) { | ||
| 317 | // Idea: A reasonable output should have at most 3 orders of magnitude | ||
| 318 | // before the decimal separator | ||
| 319 | // 353GiB is ok, 2444 GiB should be 2.386 TiB | ||
| 320 | char *result = calloc(RANDOM_STRING_LENGTH, sizeof(char)); | ||
| 321 | if (result == NULL) { | ||
| 322 | die(STATE_UNKNOWN, _("allocation failed")); | ||
| 323 | } | ||
| 324 | const byte_unit KibiBytes_factor = 1024; | ||
| 325 | const byte_unit MebiBytes_factor = 1048576; | ||
| 326 | const byte_unit GibiBytes_factor = 1073741824; | ||
| 327 | const byte_unit TebiBytes_factor = 1099511627776; | ||
| 328 | const byte_unit PebiBytes_factor = 1125899906842624; | ||
| 329 | const byte_unit ExbiBytes_factor = 1152921504606846976; | ||
| 330 | const byte_unit KiloBytes_factor = 1000; | ||
| 331 | const byte_unit MegaBytes_factor = 1000000; | ||
| 332 | const byte_unit GigaBytes_factor = 1000000000; | ||
| 333 | const byte_unit TeraBytes_factor = 1000000000000; | ||
| 334 | const byte_unit PetaBytes_factor = 1000000000000000; | ||
| 335 | const byte_unit ExaBytes_factor = 1000000000000000000; | ||
| 336 | |||
| 337 | if (use_si_units) { | ||
| 338 | // SI units, powers of 10 | ||
| 339 | if (value < KiloBytes_factor) { | ||
| 340 | snprintf(result, RANDOM_STRING_LENGTH, "%llu B", value); | ||
| 341 | } else if (value < MegaBytes_factor) { | ||
| 342 | snprintf(result, RANDOM_STRING_LENGTH, "%llu KB", value / KiloBytes_factor); | ||
| 343 | } else if (value < GigaBytes_factor) { | ||
| 344 | snprintf(result, RANDOM_STRING_LENGTH, "%llu MB", value / MegaBytes_factor); | ||
| 345 | } else if (value < TeraBytes_factor) { | ||
| 346 | snprintf(result, RANDOM_STRING_LENGTH, "%llu GB", value / GigaBytes_factor); | ||
| 347 | } else if (value < PetaBytes_factor) { | ||
| 348 | snprintf(result, RANDOM_STRING_LENGTH, "%llu TB", value / TeraBytes_factor); | ||
| 349 | } else if (value < ExaBytes_factor) { | ||
| 350 | snprintf(result, RANDOM_STRING_LENGTH, "%llu PB", value / PetaBytes_factor); | ||
| 351 | } else { | ||
| 352 | snprintf(result, RANDOM_STRING_LENGTH, "%llu EB", value / ExaBytes_factor); | ||
| 353 | } | ||
| 354 | } else { | ||
| 355 | // IEC units, powers of 2 ^ 10 | ||
| 356 | if (value < KibiBytes_factor) { | ||
| 357 | snprintf(result, RANDOM_STRING_LENGTH, "%llu B", value); | ||
| 358 | } else if (value < MebiBytes_factor) { | ||
| 359 | snprintf(result, RANDOM_STRING_LENGTH, "%llu KiB", value / KibiBytes_factor); | ||
| 360 | } else if (value < GibiBytes_factor) { | ||
| 361 | snprintf(result, RANDOM_STRING_LENGTH, "%llu MiB", value / MebiBytes_factor); | ||
| 362 | } else if (value < TebiBytes_factor) { | ||
| 363 | snprintf(result, RANDOM_STRING_LENGTH, "%llu GiB", value / GibiBytes_factor); | ||
| 364 | } else if (value < PebiBytes_factor) { | ||
| 365 | snprintf(result, RANDOM_STRING_LENGTH, "%llu TiB", value / TebiBytes_factor); | ||
| 366 | } else if (value < ExbiBytes_factor) { | ||
| 367 | snprintf(result, RANDOM_STRING_LENGTH, "%llu PiB", value / PebiBytes_factor); | ||
| 368 | } else { | ||
| 369 | snprintf(result, RANDOM_STRING_LENGTH, "%llu EiB", value / ExbiBytes_factor); | ||
| 370 | } | ||
| 371 | } | ||
| 372 | |||
| 373 | return result; | ||
| 374 | } | ||
| 375 | |||
| 376 | filesystem_list filesystem_list_init() { | ||
| 377 | filesystem_list tmp = { | ||
| 378 | .length = 0, | ||
| 379 | .first = NULL, | ||
| 380 | }; | ||
| 381 | return tmp; | ||
| 382 | } | ||
| 383 | |||
| 384 | parameter_list_elem *mp_int_fs_list_append(filesystem_list *list, const char *name) { | ||
| 385 | parameter_list_elem *current = list->first; | ||
| 386 | parameter_list_elem *new_path = (struct parameter_list *)malloc(sizeof *new_path); | ||
| 387 | *new_path = parameter_list_init(name); | ||
| 388 | |||
| 389 | if (current == NULL) { | ||
| 390 | list->first = new_path; | ||
| 391 | new_path->prev = NULL; | ||
| 392 | list->length = 1; | ||
| 393 | } else { | ||
| 394 | while (current->next) { | ||
| 395 | current = current->next; | ||
| 396 | } | ||
| 397 | current->next = new_path; | ||
| 398 | new_path->prev = current; | ||
| 399 | list->length++; | ||
| 400 | } | ||
| 401 | return new_path; | ||
| 402 | } | ||
| 403 | |||
| 404 | parameter_list_elem *mp_int_fs_list_find(filesystem_list list, const char *name) { | ||
| 405 | if (list.length == 0) { | ||
| 406 | return NULL; | ||
| 407 | } | ||
| 408 | |||
| 409 | for (parameter_list_elem *temp_list = list.first; temp_list; temp_list = temp_list->next) { | ||
| 410 | if (!strcmp(temp_list->name, name)) { | ||
| 411 | return temp_list; | ||
| 412 | } | ||
| 413 | } | ||
| 414 | |||
| 415 | return NULL; | ||
| 416 | } | ||
| 417 | |||
| 418 | parameter_list_elem *mp_int_fs_list_del(filesystem_list *list, parameter_list_elem *item) { | ||
| 419 | if (list->length == 0) { | ||
| 420 | return NULL; | ||
| 421 | } | ||
| 422 | |||
| 423 | if (item == NULL) { | ||
| 424 | // Got NULL for item, interpret this as "delete first element" | ||
| 425 | // as a kind of compatibility to the old function | ||
| 426 | item = list->first; | ||
| 427 | } | ||
| 428 | |||
| 429 | if (list->first == item) { | ||
| 430 | list->length--; | ||
| 431 | |||
| 432 | list->first = item->next; | ||
| 433 | if (list->first) { | ||
| 434 | list->first->prev = NULL; | ||
| 435 | } | ||
| 436 | return list->first; | ||
| 437 | } | ||
| 438 | |||
| 439 | // Was not the first element, continue | ||
| 440 | parameter_list_elem *prev = list->first; | ||
| 441 | parameter_list_elem *current = list->first->next; | ||
| 442 | |||
| 443 | while (current != item && current != NULL) { | ||
| 444 | prev = current; | ||
| 445 | current = current->next; | ||
| 446 | } | ||
| 447 | |||
| 448 | if (current == NULL) { | ||
| 449 | // didn't find that element .... | ||
| 450 | return NULL; | ||
| 451 | } | ||
| 452 | |||
| 453 | // remove the element | ||
| 454 | parameter_list_elem *next = current->next; | ||
| 455 | prev->next = next; | ||
| 456 | list->length--; | ||
| 457 | if (next) { | ||
| 458 | next->prev = prev; | ||
| 459 | } | ||
| 460 | |||
| 461 | if (item->name) { | ||
| 462 | free(item->name); | ||
| 463 | } | ||
| 464 | free(item); | ||
| 465 | |||
| 466 | return next; | ||
| 467 | } | ||
| 468 | |||
| 469 | parameter_list_elem *mp_int_fs_list_get_next(parameter_list_elem *current) { | ||
| 470 | if (!current) { | ||
| 471 | return NULL; | ||
| 472 | } | ||
| 473 | return current->next; | ||
| 474 | } | ||
| 475 | |||
| 476 | void mp_int_fs_list_set_best_match(filesystem_list list, struct mount_entry *mount_list, | ||
| 477 | bool exact) { | ||
| 478 | for (parameter_list_elem *elem = list.first; elem; elem = mp_int_fs_list_get_next(elem)) { | ||
| 479 | if (!elem->best_match) { | ||
| 480 | size_t name_len = strlen(elem->name); | ||
| 481 | struct mount_entry *best_match = NULL; | ||
| 482 | |||
| 483 | /* set best match if path name exactly matches a mounted device name */ | ||
| 484 | for (struct mount_entry *mount_entry = mount_list; mount_entry; | ||
| 485 | mount_entry = mount_entry->me_next) { | ||
| 486 | if (strcmp(mount_entry->me_devname, elem->name) == 0) { | ||
| 487 | struct fs_usage fsp; | ||
| 488 | if (get_fs_usage(mount_entry->me_mountdir, mount_entry->me_devname, &fsp) >= | ||
| 489 | 0) { | ||
| 490 | best_match = mount_entry; | ||
| 491 | } | ||
| 492 | } | ||
| 493 | } | ||
| 494 | |||
| 495 | /* set best match by directory name if no match was found by devname */ | ||
| 496 | if (!best_match) { | ||
| 497 | size_t best_match_len = 0; | ||
| 498 | for (struct mount_entry *mount_entry = mount_list; mount_entry; | ||
| 499 | mount_entry = mount_entry->me_next) { | ||
| 500 | size_t len = strlen(mount_entry->me_mountdir); | ||
| 501 | |||
| 502 | if ((!exact && | ||
| 503 | (best_match_len <= len && len <= name_len && | ||
| 504 | (len == 1 || strncmp(mount_entry->me_mountdir, elem->name, len) == 0))) || | ||
| 505 | (exact && strcmp(mount_entry->me_mountdir, elem->name) == 0)) { | ||
| 506 | struct fs_usage fsp; | ||
| 507 | |||
| 508 | if (get_fs_usage(mount_entry->me_mountdir, mount_entry->me_devname, &fsp) >= | ||
| 509 | 0) { | ||
| 510 | best_match = mount_entry; | ||
| 511 | best_match_len = len; | ||
| 512 | } | ||
| 513 | } | ||
| 514 | } | ||
| 515 | } | ||
| 516 | |||
| 517 | if (best_match) { | ||
| 518 | elem->best_match = best_match; | ||
| 519 | } else { | ||
| 520 | elem->best_match = | ||
| 521 | NULL; /* Not sure why this is needed as it should be null on initialisation */ | ||
| 522 | } | ||
| 523 | |||
| 524 | // No filesystem without a mount_entry! | ||
| 525 | // assert(elem->best_match != NULL); | ||
| 526 | } | ||
| 527 | } | ||
| 528 | } | ||
diff --git a/plugins/check_disk.d/utils_disk.h b/plugins/check_disk.d/utils_disk.h new file mode 100644 index 00000000..c96d4296 --- /dev/null +++ b/plugins/check_disk.d/utils_disk.h | |||
| @@ -0,0 +1,160 @@ | |||
| 1 | #pragma once | ||
| 2 | /* Header file for utils_disk */ | ||
| 3 | |||
| 4 | #include "../../config.h" | ||
| 5 | #include "../../gl/mountlist.h" | ||
| 6 | #include "../../lib/utils_base.h" | ||
| 7 | #include "../../lib/output.h" | ||
| 8 | #include "regex.h" | ||
| 9 | #include <stdint.h> | ||
| 10 | |||
| 11 | typedef unsigned long long byte_unit; | ||
| 12 | |||
| 13 | typedef enum { | ||
| 14 | Humanized, | ||
| 15 | Bytes, | ||
| 16 | KibiBytes, | ||
| 17 | MebiBytes, | ||
| 18 | GibiBytes, | ||
| 19 | TebiBytes, | ||
| 20 | PebiBytes, | ||
| 21 | ExbiBytes, | ||
| 22 | KiloBytes, | ||
| 23 | MegaBytes, | ||
| 24 | GigaBytes, | ||
| 25 | TeraBytes, | ||
| 26 | PetaBytes, | ||
| 27 | ExaBytes, | ||
| 28 | } byte_unit_enum; | ||
| 29 | |||
| 30 | typedef struct name_list string_list; | ||
| 31 | struct name_list { | ||
| 32 | char *name; | ||
| 33 | string_list *next; | ||
| 34 | }; | ||
| 35 | |||
| 36 | struct regex_list { | ||
| 37 | regex_t regex; | ||
| 38 | struct regex_list *next; | ||
| 39 | }; | ||
| 40 | |||
| 41 | typedef struct parameter_list parameter_list_elem; | ||
| 42 | struct parameter_list { | ||
| 43 | char *name; | ||
| 44 | char *group; | ||
| 45 | |||
| 46 | mp_thresholds freespace_units; | ||
| 47 | mp_thresholds freespace_percent; | ||
| 48 | mp_thresholds freeinodes_percent; | ||
| 49 | |||
| 50 | struct mount_entry *best_match; | ||
| 51 | |||
| 52 | uintmax_t inodes_free_to_root; | ||
| 53 | uintmax_t inodes_free; | ||
| 54 | uintmax_t inodes_used; | ||
| 55 | uintmax_t inodes_total; | ||
| 56 | |||
| 57 | uint64_t used_bytes; | ||
| 58 | uint64_t free_bytes; | ||
| 59 | uint64_t total_bytes; | ||
| 60 | |||
| 61 | parameter_list_elem *next; | ||
| 62 | parameter_list_elem *prev; | ||
| 63 | }; | ||
| 64 | |||
| 65 | typedef struct { | ||
| 66 | size_t length; | ||
| 67 | parameter_list_elem *first; | ||
| 68 | } filesystem_list; | ||
| 69 | |||
| 70 | filesystem_list filesystem_list_init(); | ||
| 71 | |||
| 72 | typedef struct { | ||
| 73 | char *name; | ||
| 74 | char *filesystem_type; | ||
| 75 | bool is_group; | ||
| 76 | |||
| 77 | mp_thresholds freespace_bytes_thresholds; | ||
| 78 | mp_thresholds freespace_percent_thresholds; | ||
| 79 | mp_thresholds freeinodes_percent_thresholds; | ||
| 80 | |||
| 81 | uintmax_t inodes_free_to_root; | ||
| 82 | uintmax_t inodes_free; | ||
| 83 | uintmax_t inodes_used; | ||
| 84 | uintmax_t inodes_total; | ||
| 85 | |||
| 86 | uintmax_t used_bytes; | ||
| 87 | uintmax_t free_bytes; | ||
| 88 | uintmax_t total_bytes; | ||
| 89 | } measurement_unit; | ||
| 90 | |||
| 91 | typedef struct measurement_unit_list measurement_unit_list; | ||
| 92 | struct measurement_unit_list { | ||
| 93 | measurement_unit unit; | ||
| 94 | measurement_unit_list *next; | ||
| 95 | }; | ||
| 96 | |||
| 97 | typedef struct { | ||
| 98 | // Output options | ||
| 99 | bool erronly; | ||
| 100 | bool display_mntp; | ||
| 101 | /* show only local filesystems. */ | ||
| 102 | bool show_local_fs; | ||
| 103 | /* show only local filesystems but call stat() on remote ones. */ | ||
| 104 | bool stat_remote_fs; | ||
| 105 | bool display_inodes_perfdata; | ||
| 106 | |||
| 107 | bool exact_match; | ||
| 108 | bool freespace_ignore_reserved; | ||
| 109 | |||
| 110 | bool ignore_missing; | ||
| 111 | bool path_ignored; | ||
| 112 | |||
| 113 | /* Linked list of filesystem types to omit. | ||
| 114 | If the list is empty, don't exclude any types. */ | ||
| 115 | struct regex_list *fs_exclude_list; | ||
| 116 | /* Linked list of filesystem types to check. | ||
| 117 | If the list is empty, include all types. */ | ||
| 118 | struct regex_list *fs_include_list; | ||
| 119 | struct name_list *device_path_exclude_list; | ||
| 120 | filesystem_list path_select_list; | ||
| 121 | /* Linked list of mounted filesystems. */ | ||
| 122 | struct mount_entry *mount_list; | ||
| 123 | struct name_list *seen; | ||
| 124 | |||
| 125 | byte_unit_enum display_unit; | ||
| 126 | // byte_unit unit; | ||
| 127 | |||
| 128 | bool output_format_is_set; | ||
| 129 | mp_output_format output_format; | ||
| 130 | } check_disk_config; | ||
| 131 | |||
| 132 | void np_add_name(struct name_list **list, const char *name); | ||
| 133 | bool np_find_name(struct name_list *list, const char *name); | ||
| 134 | bool np_seen_name(struct name_list *list, const char *name); | ||
| 135 | int np_add_regex(struct regex_list **list, const char *regex, int cflags); | ||
| 136 | bool np_find_regmatch(struct regex_list *list, const char *name); | ||
| 137 | |||
| 138 | parameter_list_elem parameter_list_init(const char *); | ||
| 139 | |||
| 140 | parameter_list_elem *mp_int_fs_list_append(filesystem_list *list, const char *name); | ||
| 141 | parameter_list_elem *mp_int_fs_list_find(filesystem_list list, const char *name); | ||
| 142 | parameter_list_elem *mp_int_fs_list_del(filesystem_list *list, parameter_list_elem *item); | ||
| 143 | parameter_list_elem *mp_int_fs_list_get_next(parameter_list_elem *current); | ||
| 144 | void mp_int_fs_list_set_best_match(filesystem_list list, struct mount_entry *mount_list, | ||
| 145 | bool exact); | ||
| 146 | |||
| 147 | measurement_unit measurement_unit_init(); | ||
| 148 | measurement_unit_list *add_measurement_list(measurement_unit_list *list, measurement_unit elem); | ||
| 149 | measurement_unit add_filesystem_to_measurement_unit(measurement_unit unit, | ||
| 150 | parameter_list_elem filesystem); | ||
| 151 | measurement_unit create_measurement_unit_from_filesystem(parameter_list_elem filesystem, | ||
| 152 | bool display_mntp); | ||
| 153 | |||
| 154 | int search_parameter_list(parameter_list_elem *list, const char *name); | ||
| 155 | bool np_regex_match_mount_entry(struct mount_entry *, regex_t *); | ||
| 156 | |||
| 157 | char *get_unit_string(byte_unit_enum); | ||
| 158 | check_disk_config check_disk_config_init(); | ||
| 159 | |||
| 160 | char *humanize_byte_value(unsigned long long value, bool use_si_units); | ||
