allmetrics_shell.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "allmetrics_shell.h"
  3. // ----------------------------------------------------------------------------
  4. // BASH
  5. // /api/v1/allmetrics?format=bash
  6. static inline size_t shell_name_copy(char *d, const char *s, size_t usable) {
  7. size_t n;
  8. for(n = 0; *s && n < usable ; d++, s++, n++) {
  9. register char c = *s;
  10. if(unlikely(!isalnum(c))) *d = '_';
  11. else *d = (char)toupper(c);
  12. }
  13. *d = '\0';
  14. return n;
  15. }
  16. #define SHELL_ELEMENT_MAX 100
  17. void rrd_stats_api_v1_charts_allmetrics_shell(RRDHOST *host, const char *filter_string, BUFFER *wb) {
  18. analytics_log_shell();
  19. SIMPLE_PATTERN *filter = simple_pattern_create(filter_string, NULL, SIMPLE_PATTERN_EXACT, true);
  20. // for each chart
  21. RRDSET *st;
  22. rrdset_foreach_read(st, host) {
  23. if (filter && !simple_pattern_matches_string(filter, st->name))
  24. continue;
  25. NETDATA_DOUBLE total = 0.0;
  26. char chart[SHELL_ELEMENT_MAX + 1];
  27. shell_name_copy(chart, st->name?rrdset_name(st):rrdset_id(st), SHELL_ELEMENT_MAX);
  28. buffer_sprintf(wb, "\n# chart: %s (name: %s)\n", rrdset_id(st), rrdset_name(st));
  29. if(rrdset_is_available_for_viewers(st)) {
  30. // for each dimension
  31. RRDDIM *rd;
  32. rrddim_foreach_read(rd, st) {
  33. if(rd->collections_counter && !rrddim_flag_check(rd, RRDDIM_FLAG_OBSOLETE)) {
  34. char dimension[SHELL_ELEMENT_MAX + 1];
  35. shell_name_copy(dimension, rd->name?rrddim_name(rd):rrddim_id(rd), SHELL_ELEMENT_MAX);
  36. NETDATA_DOUBLE n = rd->last_stored_value;
  37. if(isnan(n) || isinf(n))
  38. buffer_sprintf(wb, "NETDATA_%s_%s=\"\" # %s\n", chart, dimension, rrdset_units(st));
  39. else {
  40. if(rd->multiplier < 0 || rd->divisor < 0) n = -n;
  41. n = roundndd(n);
  42. if(!rrddim_option_check(rd, RRDDIM_OPTION_HIDDEN)) total += n;
  43. buffer_sprintf(wb, "NETDATA_%s_%s=\"" NETDATA_DOUBLE_FORMAT_ZERO "\" # %s\n", chart, dimension, n, rrdset_units(st));
  44. }
  45. }
  46. }
  47. rrddim_foreach_done(rd);
  48. total = roundndd(total);
  49. buffer_sprintf(wb, "NETDATA_%s_VISIBLETOTAL=\"" NETDATA_DOUBLE_FORMAT_ZERO "\" # %s\n", chart, total, rrdset_units(st));
  50. }
  51. }
  52. rrdset_foreach_done(st);
  53. buffer_strcat(wb, "\n# NETDATA ALARMS RUNNING\n");
  54. RRDCALC *rc;
  55. foreach_rrdcalc_in_rrdhost_read(host, rc) {
  56. if(!rc->rrdset) continue;
  57. char chart[SHELL_ELEMENT_MAX + 1];
  58. shell_name_copy(chart, rc->rrdset->name?rrdset_name(rc->rrdset):rrdset_id(rc->rrdset), SHELL_ELEMENT_MAX);
  59. char alarm[SHELL_ELEMENT_MAX + 1];
  60. shell_name_copy(alarm, rrdcalc_name(rc), SHELL_ELEMENT_MAX);
  61. NETDATA_DOUBLE n = rc->value;
  62. if(isnan(n) || isinf(n))
  63. buffer_sprintf(wb, "NETDATA_ALARM_%s_%s_VALUE=\"\" # %s\n", chart, alarm, rrdcalc_units(rc));
  64. else {
  65. n = roundndd(n);
  66. buffer_sprintf(wb, "NETDATA_ALARM_%s_%s_VALUE=\"" NETDATA_DOUBLE_FORMAT_ZERO "\" # %s\n", chart, alarm, n, rrdcalc_units(rc));
  67. }
  68. buffer_sprintf(wb, "NETDATA_ALARM_%s_%s_STATUS=\"%s\"\n", chart, alarm, rrdcalc_status2string(rc->status));
  69. }
  70. foreach_rrdcalc_in_rrdhost_done(rc);
  71. simple_pattern_free(filter);
  72. }
  73. // ----------------------------------------------------------------------------
  74. void rrd_stats_api_v1_charts_allmetrics_json(RRDHOST *host, const char *filter_string, BUFFER *wb) {
  75. analytics_log_json();
  76. SIMPLE_PATTERN *filter = simple_pattern_create(filter_string, NULL, SIMPLE_PATTERN_EXACT, true);
  77. buffer_strcat(wb, "{");
  78. size_t chart_counter = 0;
  79. size_t dimension_counter = 0;
  80. // for each chart
  81. RRDSET *st;
  82. rrdset_foreach_read(st, host) {
  83. if (filter && !(simple_pattern_matches_string(filter, st->id) || simple_pattern_matches_string(filter, st->name)))
  84. continue;
  85. if(rrdset_is_available_for_viewers(st)) {
  86. buffer_sprintf(
  87. wb,
  88. "%s\n"
  89. "\t\"%s\": {\n"
  90. "\t\t\"name\":\"%s\",\n"
  91. "\t\t\"family\":\"%s\",\n"
  92. "\t\t\"context\":\"%s\",\n"
  93. "\t\t\"units\":\"%s\",\n"
  94. "\t\t\"last_updated\": %"PRId64",\n"
  95. "\t\t\"dimensions\": {",
  96. chart_counter ? "," : "",
  97. rrdset_id(st),
  98. rrdset_name(st),
  99. rrdset_family(st),
  100. rrdset_context(st),
  101. rrdset_units(st),
  102. (int64_t) rrdset_last_entry_s(st));
  103. chart_counter++;
  104. dimension_counter = 0;
  105. // for each dimension
  106. RRDDIM *rd;
  107. rrddim_foreach_read(rd, st) {
  108. if(rd->collections_counter && !rrddim_flag_check(rd, RRDDIM_FLAG_OBSOLETE)) {
  109. buffer_sprintf(
  110. wb,
  111. "%s\n"
  112. "\t\t\t\"%s\": {\n"
  113. "\t\t\t\t\"name\": \"%s\",\n"
  114. "\t\t\t\t\"value\": ",
  115. dimension_counter ? "," : "",
  116. rrddim_id(rd),
  117. rrddim_name(rd));
  118. if(isnan(rd->last_stored_value))
  119. buffer_strcat(wb, "null");
  120. else
  121. buffer_sprintf(wb, NETDATA_DOUBLE_FORMAT, rd->last_stored_value);
  122. buffer_strcat(wb, "\n\t\t\t}");
  123. dimension_counter++;
  124. }
  125. }
  126. rrddim_foreach_done(rd);
  127. buffer_strcat(wb, "\n\t\t}\n\t}");
  128. }
  129. }
  130. rrdset_foreach_done(st);
  131. buffer_strcat(wb, "\n}");
  132. simple_pattern_free(filter);
  133. }