allmetrics_shell.c 5.3 KB

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