allmetrics_shell.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  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);
  20. rrdhost_rdlock(host);
  21. // for each chart
  22. RRDSET *st;
  23. rrdset_foreach_read(st, host) {
  24. if (filter && !simple_pattern_matches(filter, st->name))
  25. continue;
  26. NETDATA_DOUBLE total = 0.0;
  27. char chart[SHELL_ELEMENT_MAX + 1];
  28. shell_name_copy(chart, st->name?st->name:st->id, SHELL_ELEMENT_MAX);
  29. buffer_sprintf(wb, "\n# chart: %s (name: %s)\n", st->id, st->name);
  30. if(rrdset_is_available_for_viewers(st)) {
  31. rrdset_rdlock(st);
  32. // for each dimension
  33. RRDDIM *rd;
  34. rrddim_foreach_read(rd, st) {
  35. if(rd->collections_counter && !rrddim_flag_check(rd, RRDDIM_FLAG_OBSOLETE)) {
  36. char dimension[SHELL_ELEMENT_MAX + 1];
  37. shell_name_copy(dimension, rd->name?rd->name:rd->id, SHELL_ELEMENT_MAX);
  38. NETDATA_DOUBLE n = rd->last_stored_value;
  39. if(isnan(n) || isinf(n))
  40. buffer_sprintf(wb, "NETDATA_%s_%s=\"\" # %s\n", chart, dimension, st->units);
  41. else {
  42. if(rd->multiplier < 0 || rd->divisor < 0) n = -n;
  43. n = roundndd(n);
  44. if(!rrddim_flag_check(rd, RRDDIM_FLAG_HIDDEN)) total += n;
  45. buffer_sprintf(wb, "NETDATA_%s_%s=\"" NETDATA_DOUBLE_FORMAT_ZERO "\" # %s\n", chart, dimension, n, st->units);
  46. }
  47. }
  48. }
  49. total = roundndd(total);
  50. buffer_sprintf(wb, "NETDATA_%s_VISIBLETOTAL=\"" NETDATA_DOUBLE_FORMAT_ZERO "\" # %s\n", chart, total, st->units);
  51. rrdset_unlock(st);
  52. }
  53. }
  54. buffer_strcat(wb, "\n# NETDATA ALARMS RUNNING\n");
  55. RRDCALC *rc;
  56. for(rc = host->alarms; rc ;rc = rc->next) {
  57. if(!rc->rrdset) continue;
  58. char chart[SHELL_ELEMENT_MAX + 1];
  59. shell_name_copy(chart, rc->rrdset->name?rc->rrdset->name:rc->rrdset->id, SHELL_ELEMENT_MAX);
  60. char alarm[SHELL_ELEMENT_MAX + 1];
  61. shell_name_copy(alarm, rc->name, SHELL_ELEMENT_MAX);
  62. NETDATA_DOUBLE n = rc->value;
  63. if(isnan(n) || isinf(n))
  64. buffer_sprintf(wb, "NETDATA_ALARM_%s_%s_VALUE=\"\" # %s\n", chart, alarm, rc->units);
  65. else {
  66. n = roundndd(n);
  67. buffer_sprintf(wb, "NETDATA_ALARM_%s_%s_VALUE=\"" NETDATA_DOUBLE_FORMAT_ZERO "\" # %s\n", chart, alarm, n, rc->units);
  68. }
  69. buffer_sprintf(wb, "NETDATA_ALARM_%s_%s_STATUS=\"%s\"\n", chart, alarm, rrdcalc_status2string(rc->status));
  70. }
  71. rrdhost_unlock(host);
  72. simple_pattern_free(filter);
  73. }
  74. // ----------------------------------------------------------------------------
  75. void rrd_stats_api_v1_charts_allmetrics_json(RRDHOST *host, const char *filter_string, BUFFER *wb) {
  76. analytics_log_json();
  77. SIMPLE_PATTERN *filter = simple_pattern_create(filter_string, NULL, SIMPLE_PATTERN_EXACT);
  78. rrdhost_rdlock(host);
  79. buffer_strcat(wb, "{");
  80. size_t chart_counter = 0;
  81. size_t dimension_counter = 0;
  82. // for each chart
  83. RRDSET *st;
  84. rrdset_foreach_read(st, host) {
  85. if (filter && !(simple_pattern_matches(filter, st->id) || simple_pattern_matches(filter, st->name)))
  86. continue;
  87. if(rrdset_is_available_for_viewers(st)) {
  88. rrdset_rdlock(st);
  89. buffer_sprintf(
  90. wb,
  91. "%s\n"
  92. "\t\"%s\": {\n"
  93. "\t\t\"name\":\"%s\",\n"
  94. "\t\t\"family\":\"%s\",\n"
  95. "\t\t\"context\":\"%s\",\n"
  96. "\t\t\"units\":\"%s\",\n"
  97. "\t\t\"last_updated\": %"PRId64",\n"
  98. "\t\t\"dimensions\": {",
  99. chart_counter ? "," : "",
  100. st->id,
  101. st->name,
  102. st->family,
  103. st->context,
  104. st->units,
  105. (int64_t)rrdset_last_entry_t_nolock(st));
  106. chart_counter++;
  107. dimension_counter = 0;
  108. // for each dimension
  109. RRDDIM *rd;
  110. rrddim_foreach_read(rd, st) {
  111. if(rd->collections_counter && !rrddim_flag_check(rd, RRDDIM_FLAG_OBSOLETE)) {
  112. buffer_sprintf(
  113. wb,
  114. "%s\n"
  115. "\t\t\t\"%s\": {\n"
  116. "\t\t\t\t\"name\": \"%s\",\n"
  117. "\t\t\t\t\"value\": ",
  118. dimension_counter ? "," : "",
  119. rd->id,
  120. rd->name);
  121. if(isnan(rd->last_stored_value))
  122. buffer_strcat(wb, "null");
  123. else
  124. buffer_sprintf(wb, NETDATA_DOUBLE_FORMAT, rd->last_stored_value);
  125. buffer_strcat(wb, "\n\t\t\t}");
  126. dimension_counter++;
  127. }
  128. }
  129. buffer_strcat(wb, "\n\t\t}\n\t}");
  130. rrdset_unlock(st);
  131. }
  132. }
  133. buffer_strcat(wb, "\n}");
  134. rrdhost_unlock(host);
  135. simple_pattern_free(filter);
  136. }