123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944 |
- // SPDX-License-Identifier: GPL-3.0-or-later
- #include "prometheus.h"
- // ----------------------------------------------------------------------------
- // PROMETHEUS
- // /api/v1/allmetrics?format=prometheus and /api/v1/allmetrics?format=prometheus_all_hosts
- static int is_matches_rrdset(struct instance *instance, RRDSET *st, SIMPLE_PATTERN *filter) {
- if (instance->config.options & EXPORTING_OPTION_SEND_NAMES) {
- return simple_pattern_matches_string(filter, st->name);
- }
- return simple_pattern_matches_string(filter, st->id);
- }
- /**
- * Check if a chart can be sent to Prometheus
- *
- * @param instance an instance data structure.
- * @param st a chart.
- * @param filter a simple pattern to match against.
- * @return Returns 1 if the chart can be sent, 0 otherwise.
- */
- inline int can_send_rrdset(struct instance *instance, RRDSET *st, SIMPLE_PATTERN *filter)
- {
- #ifdef NETDATA_INTERNAL_CHECKS
- RRDHOST *host = st->rrdhost;
- #endif
- if (unlikely(rrdset_flag_check(st, RRDSET_FLAG_EXPORTING_IGNORE)))
- return 0;
- if (filter) {
- if (!is_matches_rrdset(instance, st, filter)) {
- return 0;
- }
- } else if (unlikely(!rrdset_flag_check(st, RRDSET_FLAG_EXPORTING_SEND))) {
- // we have not checked this chart
- if (is_matches_rrdset(instance, st, instance->config.charts_pattern)) {
- rrdset_flag_set(st, RRDSET_FLAG_EXPORTING_SEND);
- } else {
- rrdset_flag_set(st, RRDSET_FLAG_EXPORTING_IGNORE);
- netdata_log_debug(
- D_EXPORTING,
- "EXPORTING: not sending chart '%s' of host '%s', because it is disabled for exporting.",
- rrdset_id(st),
- rrdhost_hostname(host));
- return 0;
- }
- }
- if (unlikely(!rrdset_is_available_for_exporting_and_alarms(st))) {
- netdata_log_debug(
- D_EXPORTING,
- "EXPORTING: not sending chart '%s' of host '%s', because it is not available for exporting.",
- rrdset_id(st),
- rrdhost_hostname(host));
- return 0;
- }
- if (unlikely(
- st->rrd_memory_mode == RRD_MEMORY_MODE_NONE &&
- !(EXPORTING_OPTIONS_DATA_SOURCE(instance->config.options) == EXPORTING_SOURCE_DATA_AS_COLLECTED))) {
- netdata_log_debug(
- D_EXPORTING,
- "EXPORTING: not sending chart '%s' of host '%s' because its memory mode is '%s' and the exporting connector requires database access.",
- rrdset_id(st),
- rrdhost_hostname(host),
- rrd_memory_mode_name(host->rrd_memory_mode));
- return 0;
- }
- return 1;
- }
- static struct prometheus_server {
- const char *server;
- uint32_t hash;
- RRDHOST *host;
- time_t last_access;
- struct prometheus_server *next;
- } *prometheus_server_root = NULL;
- static netdata_mutex_t prometheus_server_root_mutex = NETDATA_MUTEX_INITIALIZER;
- /**
- * Clean server root local structure
- */
- void prometheus_clean_server_root()
- {
- if (prometheus_server_root) {
- netdata_mutex_lock(&prometheus_server_root_mutex);
- struct prometheus_server *ps;
- for (ps = prometheus_server_root; ps; ) {
- struct prometheus_server *current = ps;
- ps = ps->next;
- if(current->server)
- freez((void *)current->server);
- freez(current);
- }
- prometheus_server_root = NULL;
- netdata_mutex_unlock(&prometheus_server_root_mutex);
- }
- }
- /**
- * Get the last time when a Prometheus server scraped the Netdata Prometheus exporter.
- *
- * @param server the name of the Prometheus server.
- * @param host a data collecting host.
- * @param now actual time.
- * @return Returns the last time when the server accessed Netdata, or 0 if it is the first occurrence.
- */
- static inline time_t prometheus_server_last_access(const char *server, RRDHOST *host, time_t now)
- {
- #ifdef UNIT_TESTING
- return 0;
- #endif
- uint32_t hash = simple_hash(server);
- netdata_mutex_lock(&prometheus_server_root_mutex);
- struct prometheus_server *ps;
- for (ps = prometheus_server_root; ps; ps = ps->next) {
- if (host == ps->host && hash == ps->hash && !strcmp(server, ps->server)) {
- time_t last = ps->last_access;
- ps->last_access = now;
- netdata_mutex_unlock(&prometheus_server_root_mutex);
- return last;
- }
- }
- ps = callocz(1, sizeof(struct prometheus_server));
- ps->server = strdupz(server);
- ps->hash = hash;
- ps->host = host;
- ps->last_access = now;
- ps->next = prometheus_server_root;
- prometheus_server_root = ps;
- netdata_mutex_unlock(&prometheus_server_root_mutex);
- return 0;
- }
- /**
- * Copy and sanitize name.
- *
- * @param d a destination string.
- * @param s a source string.
- * @param usable the number of characters to copy.
- * @return Returns the length of the copied string.
- */
- inline size_t prometheus_name_copy(char *d, const char *s, size_t usable)
- {
- size_t n;
- for (n = 0; *s && n < usable; d++, s++, n++) {
- register char c = *s;
- if (!isalnum(c))
- *d = '_';
- else
- *d = c;
- }
- *d = '\0';
- return n;
- }
- /**
- * Copy and sanitize label.
- *
- * @param d a destination string.
- * @param s a source string.
- * @param usable the number of characters to copy.
- * @return Returns the length of the copied string.
- */
- inline size_t prometheus_label_copy(char *d, const char *s, size_t usable)
- {
- size_t n;
- // make sure we can escape one character without overflowing the buffer
- usable--;
- for (n = 0; *s && n < usable; d++, s++, n++) {
- register char c = *s;
- if (unlikely(c == '"' || c == '\\' || c == '\n')) {
- *d++ = '\\';
- n++;
- }
- *d = c;
- }
- *d = '\0';
- return n;
- }
- /**
- * Copy and sanitize units.
- *
- * @param d a destination string.
- * @param s a source string.
- * @param usable the number of characters to copy.
- * @param showoldunits set this flag to 1 to show old (before v1.12) units.
- * @return Returns the destination string.
- */
- inline char *prometheus_units_copy(char *d, const char *s, size_t usable, int showoldunits)
- {
- const char *sorig = s;
- char *ret = d;
- size_t n;
- // Fix for issue 5227
- if (unlikely(showoldunits)) {
- static struct {
- const char *newunit;
- uint32_t hash;
- const char *oldunit;
- } units[] = { { "KiB/s", 0, "kilobytes/s" },
- { "MiB/s", 0, "MB/s" },
- { "GiB/s", 0, "GB/s" },
- { "KiB", 0, "KB" },
- { "MiB", 0, "MB" },
- { "GiB", 0, "GB" },
- { "inodes", 0, "Inodes" },
- { "percentage", 0, "percent" },
- { "faults/s", 0, "page faults/s" },
- { "KiB/operation", 0, "kilobytes per operation" },
- { "milliseconds/operation", 0, "ms per operation" },
- { NULL, 0, NULL } };
- static int initialized = 0;
- int i;
- if (unlikely(!initialized)) {
- for (i = 0; units[i].newunit; i++)
- units[i].hash = simple_hash(units[i].newunit);
- initialized = 1;
- }
- uint32_t hash = simple_hash(s);
- for (i = 0; units[i].newunit; i++) {
- if (unlikely(hash == units[i].hash && !strcmp(s, units[i].newunit))) {
- // netdata_log_info("matched extension for filename '%s': '%s'", filename, last_dot);
- s = units[i].oldunit;
- sorig = s;
- break;
- }
- }
- }
- *d++ = '_';
- for (n = 1; *s && n < usable; d++, s++, n++) {
- register char c = *s;
- if (!isalnum(c))
- *d = '_';
- else
- *d = c;
- }
- if (n == 2 && sorig[0] == '%') {
- n = 0;
- d = ret;
- s = "_percent";
- for (; *s && n < usable; n++)
- *d++ = *s++;
- } else if (n > 3 && sorig[n - 3] == '/' && sorig[n - 2] == 's') {
- n = n - 2;
- d -= 2;
- s = "_persec";
- for (; *s && n < usable; n++)
- *d++ = *s++;
- }
- *d = '\0';
- return ret;
- }
- /**
- * Format host labels for the Prometheus exporter
- *
- * @param instance an instance data structure.
- * @param host a data collecting host.
- */
- struct format_prometheus_label_callback {
- struct instance *instance;
- size_t count;
- };
- static int format_prometheus_label_callback(const char *name, const char *value, RRDLABEL_SRC ls __maybe_unused, void *data) {
- struct format_prometheus_label_callback *d = (struct format_prometheus_label_callback *)data;
- if (!should_send_label(d->instance, ls)) return 0;
- char k[PROMETHEUS_ELEMENT_MAX + 1];
- char v[PROMETHEUS_ELEMENT_MAX + 1];
- prometheus_name_copy(k, name, PROMETHEUS_ELEMENT_MAX);
- prometheus_label_copy(v, value, PROMETHEUS_ELEMENT_MAX);
- if (*k && *v) {
- if (d->count > 0) buffer_strcat(d->instance->labels_buffer, ",");
- buffer_sprintf(d->instance->labels_buffer, "%s=\"%s\"", k, v);
- d->count++;
- }
- return 1;
- }
- void format_host_labels_prometheus(struct instance *instance, RRDHOST *host)
- {
- if (unlikely(!sending_labels_configured(instance)))
- return;
- if (!instance->labels_buffer)
- instance->labels_buffer = buffer_create(1024, &netdata_buffers_statistics.buffers_exporters);
- struct format_prometheus_label_callback tmp = {
- .instance = instance,
- .count = 0
- };
- rrdlabels_walkthrough_read(host->rrdlabels, format_prometheus_label_callback, &tmp);
- }
- /**
- * Format host labels for the Prometheus exporter
- * We are using a structure instead a direct buffer to expand options quickly.
- *
- * @param data is the buffer used to add labels.
- */
- static int format_prometheus_chart_label_callback(const char *name, const char *value, RRDLABEL_SRC ls __maybe_unused, void *data) {
- BUFFER *wb = data;
- if (name[0] == '_' )
- return 1;
- char k[PROMETHEUS_ELEMENT_MAX + 1];
- char v[PROMETHEUS_ELEMENT_MAX + 1];
- prometheus_name_copy(k, name, PROMETHEUS_ELEMENT_MAX);
- prometheus_label_copy(v, value, PROMETHEUS_ELEMENT_MAX);
- if (*k && *v)
- buffer_sprintf(wb, ",%s=\"%s\"", k, v);
- return 1;
- }
- struct host_variables_callback_options {
- RRDHOST *host;
- BUFFER *wb;
- EXPORTING_OPTIONS exporting_options;
- PROMETHEUS_OUTPUT_OPTIONS output_options;
- const char *prefix;
- const char *labels;
- time_t now;
- int host_header_printed;
- char name[PROMETHEUS_VARIABLE_MAX + 1];
- };
- /**
- * Print host variables.
- *
- * @param rv a variable.
- * @param data callback options.
- * @return Returns 1 if the chart can be sent, 0 otherwise.
- */
- static int print_host_variables_callback(const DICTIONARY_ITEM *item __maybe_unused, void *rv_ptr __maybe_unused, void *data) {
- const RRDVAR_ACQUIRED *rv = (const RRDVAR_ACQUIRED *)item;
- struct host_variables_callback_options *opts = data;
- if (rrdvar_flags(rv) & (RRDVAR_FLAG_CUSTOM_HOST_VAR | RRDVAR_FLAG_CUSTOM_CHART_VAR)) {
- if (!opts->host_header_printed) {
- opts->host_header_printed = 1;
- if (opts->output_options & PROMETHEUS_OUTPUT_HELP) {
- buffer_sprintf(opts->wb, "\n# COMMENT global host and chart variables\n");
- }
- }
- NETDATA_DOUBLE value = rrdvar2number(rv);
- if (isnan(value) || isinf(value)) {
- if (opts->output_options & PROMETHEUS_OUTPUT_HELP)
- buffer_sprintf(
- opts->wb, "# COMMENT variable \"%s\" is %s. Skipped.\n", rrdvar_name(rv), (isnan(value)) ? "NAN" : "INF");
- return 0;
- }
- char *label_pre = "";
- char *label_post = "";
- if (opts->labels && *opts->labels) {
- label_pre = "{";
- label_post = "}";
- }
- prometheus_name_copy(opts->name, rrdvar_name(rv), sizeof(opts->name));
- if (opts->output_options & PROMETHEUS_OUTPUT_TIMESTAMPS)
- buffer_sprintf(
- opts->wb,
- "%s_%s%s%s%s " NETDATA_DOUBLE_FORMAT " %llu\n",
- opts->prefix,
- opts->name,
- label_pre,
- opts->labels,
- label_post,
- value,
- opts->now * 1000ULL);
- else
- buffer_sprintf(
- opts->wb,
- "%s_%s%s%s%s " NETDATA_DOUBLE_FORMAT "\n",
- opts->prefix,
- opts->name,
- label_pre,
- opts->labels,
- label_post,
- value);
- return 1;
- }
- return 0;
- }
- struct gen_parameters {
- const char *prefix;
- const char *labels_prefix;
- char *context;
- char *suffix;
- char *chart;
- char *dimension;
- char *family;
- char *labels;
- PROMETHEUS_OUTPUT_OPTIONS output_options;
- RRDSET *st;
- RRDDIM *rd;
- const char *relation;
- const char *type;
- };
- /**
- * Write an as-collected help comment to a buffer.
- *
- * @param wb the buffer to write the comment to.
- * @param p parameters for generating the comment string.
- * @param homogeneous a flag for homogeneous charts.
- * @param prometheus_collector a flag for metrics from prometheus collector.
- */
- static void generate_as_collected_prom_help(BUFFER *wb, struct gen_parameters *p, int homogeneous, int prometheus_collector)
- {
- buffer_sprintf(wb, "# COMMENT %s_%s", p->prefix, p->context);
- if (!homogeneous)
- buffer_sprintf(wb, "_%s", p->dimension);
- buffer_sprintf(
- wb,
- "%s: chart \"%s\", context \"%s\", family \"%s\", dimension \"%s\", value * ",
- p->suffix,
- (p->output_options & PROMETHEUS_OUTPUT_NAMES && p->st->name) ? rrdset_name(p->st) : rrdset_id(p->st),
- rrdset_context(p->st),
- rrdset_family(p->st),
- (p->output_options & PROMETHEUS_OUTPUT_NAMES && p->rd->name) ? rrddim_name(p->rd) : rrddim_id(p->rd));
- if (prometheus_collector)
- buffer_sprintf(wb, "1 / 1");
- else
- buffer_sprintf(wb, "%d / %d", p->rd->multiplier, p->rd->divisor);
- buffer_sprintf(wb, " %s %s (%s)\n", p->relation, rrdset_units(p->st), p->type);
- }
- /**
- * Write an as-collected metric to a buffer.
- *
- * @param wb the buffer to write the metric to.
- * @param p parameters for generating the metric string.
- * @param homogeneous a flag for homogeneous charts.
- * @param prometheus_collector a flag for metrics from prometheus collector.
- * @param chart_labels the dictionary with chart labels
- */
- static void generate_as_collected_prom_metric(BUFFER *wb,
- struct gen_parameters *p,
- int homogeneous,
- int prometheus_collector,
- RRDLABELS *chart_labels)
- {
- buffer_sprintf(wb, "%s_%s", p->prefix, p->context);
- if (!homogeneous)
- buffer_sprintf(wb, "_%s", p->dimension);
- buffer_sprintf(wb, "%s{%schart=\"%s\"", p->suffix, p->labels_prefix, p->chart);
- if (homogeneous)
- buffer_sprintf(wb, ",%sdimension=\"%s\"", p->labels_prefix, p->dimension);
- buffer_sprintf(wb, ",%sfamily=\"%s\"", p->labels_prefix, p->family);
- rrdlabels_walkthrough_read(chart_labels, format_prometheus_chart_label_callback, wb);
- buffer_sprintf(wb, "%s} ", p->labels);
- if (prometheus_collector)
- buffer_sprintf(
- wb,
- NETDATA_DOUBLE_FORMAT,
- (NETDATA_DOUBLE)p->rd->collector.last_collected_value * (NETDATA_DOUBLE)p->rd->multiplier /
- (NETDATA_DOUBLE)p->rd->divisor);
- else
- buffer_sprintf(wb, COLLECTED_NUMBER_FORMAT, p->rd->collector.last_collected_value);
- if (p->output_options & PROMETHEUS_OUTPUT_TIMESTAMPS)
- buffer_sprintf(wb, " %"PRIu64"\n", timeval_msec(&p->rd->collector.last_collected_time));
- else
- buffer_sprintf(wb, "\n");
- }
- /**
- * Write metrics in Prometheus format to a buffer.
- *
- * @param instance an instance data structure.
- * @param host a data collecting host.
- * @param filter_string a simple pattern filter.
- * @param wb the buffer to fill with metrics.
- * @param prefix a prefix for every metric.
- * @param exporting_options options to configure what data is exported.
- * @param allhosts set to 1 if host instance should be in the output for tags.
- * @param output_options options to configure the format of the output.
- */
- static void rrd_stats_api_v1_charts_allmetrics_prometheus(
- struct instance *instance,
- RRDHOST *host,
- const char *filter_string,
- BUFFER *wb,
- const char *prefix,
- EXPORTING_OPTIONS exporting_options,
- int allhosts,
- PROMETHEUS_OUTPUT_OPTIONS output_options)
- {
- SIMPLE_PATTERN *filter = simple_pattern_create(filter_string, NULL, SIMPLE_PATTERN_EXACT, true);
- char hostname[PROMETHEUS_ELEMENT_MAX + 1];
- prometheus_label_copy(hostname, rrdhost_hostname(host), PROMETHEUS_ELEMENT_MAX);
- format_host_labels_prometheus(instance, host);
- buffer_sprintf(
- wb,
- "netdata_info{instance=\"%s\",application=\"%s\",version=\"%s\"",
- hostname,
- rrdhost_program_name(host),
- rrdhost_program_version(host));
- if (instance->labels_buffer && *buffer_tostring(instance->labels_buffer)) {
- buffer_sprintf(wb, ",%s", buffer_tostring(instance->labels_buffer));
- }
- if (output_options & PROMETHEUS_OUTPUT_TIMESTAMPS)
- buffer_sprintf(wb, "} 1 %llu\n", now_realtime_usec() / USEC_PER_MS);
- else
- buffer_sprintf(wb, "} 1\n");
- char labels[PROMETHEUS_LABELS_MAX + 1] = "";
- if (allhosts) {
- snprintfz(labels, PROMETHEUS_LABELS_MAX, ",%sinstance=\"%s\"", instance->config.label_prefix, hostname);
- }
- if (instance->labels_buffer)
- buffer_flush(instance->labels_buffer);
- // send custom variables set for the host
- if (output_options & PROMETHEUS_OUTPUT_VARIABLES) {
- struct host_variables_callback_options opts = {
- .host = host,
- .wb = wb,
- .labels = (labels[0] == ',') ? &labels[1] : labels,
- .exporting_options = exporting_options,
- .output_options = output_options,
- .prefix = prefix,
- .now = now_realtime_sec(),
- .host_header_printed = 0
- };
- rrdvar_walkthrough_read(host->rrdvars, print_host_variables_callback, &opts);
- }
- // for each chart
- RRDSET *st;
- BUFFER *plabels_buffer = buffer_create(0, NULL);
- const char *plabels_prefix = instance->config.label_prefix;
- STRING *prometheus = string_strdupz("prometheus");
- rrdset_foreach_read(st, host) {
- if (likely(can_send_rrdset(instance, st, filter))) {
- char chart[PROMETHEUS_ELEMENT_MAX + 1];
- char context[PROMETHEUS_ELEMENT_MAX + 1];
- char family[PROMETHEUS_ELEMENT_MAX + 1];
- char units[PROMETHEUS_ELEMENT_MAX + 1] = "";
- prometheus_label_copy(chart, (output_options & PROMETHEUS_OUTPUT_NAMES && st->name) ? rrdset_name(st) : rrdset_id(st), PROMETHEUS_ELEMENT_MAX);
- prometheus_label_copy(family, rrdset_family(st), PROMETHEUS_ELEMENT_MAX);
- prometheus_name_copy(context, rrdset_context(st), PROMETHEUS_ELEMENT_MAX);
- int as_collected = (EXPORTING_OPTIONS_DATA_SOURCE(exporting_options) == EXPORTING_SOURCE_DATA_AS_COLLECTED);
- int homogeneous = 1;
- int prometheus_collector = 0;
- RRDSET_FLAGS flags = rrdset_flag_get(st);
- if (as_collected) {
- if (flags & RRDSET_FLAG_HOMOGENEOUS_CHECK)
- rrdset_update_heterogeneous_flag(st);
- if (flags & RRDSET_FLAG_HETEROGENEOUS)
- homogeneous = 0;
- if (st->module_name == prometheus)
- prometheus_collector = 1;
- }
- else {
- if (EXPORTING_OPTIONS_DATA_SOURCE(exporting_options) == EXPORTING_SOURCE_DATA_AVERAGE &&
- !(output_options & PROMETHEUS_OUTPUT_HIDEUNITS))
- prometheus_units_copy(
- units, rrdset_units(st), PROMETHEUS_ELEMENT_MAX, output_options & PROMETHEUS_OUTPUT_OLDUNITS);
- }
- if (unlikely(output_options & PROMETHEUS_OUTPUT_HELP))
- buffer_sprintf(
- wb,
- "\n# COMMENT %s chart \"%s\", context \"%s\", family \"%s\", units \"%s\"\n",
- (homogeneous) ? "homogeneous" : "heterogeneous",
- (output_options & PROMETHEUS_OUTPUT_NAMES && st->name) ? rrdset_name(st) : rrdset_id(st),
- rrdset_context(st),
- rrdset_family(st),
- rrdset_units(st));
- // for each dimension
- RRDDIM *rd;
- rrddim_foreach_read(rd, st) {
- if (rd->collector.counter && !rrddim_flag_check(rd, RRDDIM_FLAG_OBSOLETE)) {
- char dimension[PROMETHEUS_ELEMENT_MAX + 1];
- char *suffix = "";
- if (as_collected) {
- // we need as-collected / raw data
- struct gen_parameters p;
- p.prefix = prefix;
- p.labels_prefix = instance->config.label_prefix;
- p.context = context;
- p.suffix = suffix;
- p.chart = chart;
- p.dimension = dimension;
- p.family = family;
- p.labels = labels;
- p.output_options = output_options;
- p.st = st;
- p.rd = rd;
- if (unlikely(rd->collector.last_collected_time.tv_sec < instance->after))
- continue;
- p.type = "gauge";
- p.relation = "gives";
- if (rd->algorithm == RRD_ALGORITHM_INCREMENTAL ||
- rd->algorithm == RRD_ALGORITHM_PCENT_OVER_DIFF_TOTAL) {
- p.type = "counter";
- p.relation = "delta gives";
- if (!prometheus_collector)
- p.suffix = "_total";
- }
- if (homogeneous) {
- // all the dimensions of the chart, has the same algorithm, multiplier and divisor
- // we add all dimensions as labels
- prometheus_label_copy(
- dimension,
- (output_options & PROMETHEUS_OUTPUT_NAMES && rd->name) ? rrddim_name(rd) : rrddim_id(rd),
- PROMETHEUS_ELEMENT_MAX);
- if (unlikely(output_options & PROMETHEUS_OUTPUT_HELP))
- generate_as_collected_prom_help(wb, &p, homogeneous, prometheus_collector);
- if (unlikely(output_options & PROMETHEUS_OUTPUT_TYPES))
- buffer_sprintf(wb, "# TYPE %s_%s%s %s\n", prefix, context, suffix, p.type);
- generate_as_collected_prom_metric(wb, &p, homogeneous, prometheus_collector, st->rrdlabels);
- }
- else {
- // the dimensions of the chart, do not have the same algorithm, multiplier or divisor
- // we create a metric per dimension
- prometheus_name_copy(
- dimension,
- (output_options & PROMETHEUS_OUTPUT_NAMES && rd->name) ? rrddim_name(rd) : rrddim_id(rd),
- PROMETHEUS_ELEMENT_MAX);
- if (unlikely(output_options & PROMETHEUS_OUTPUT_HELP))
- generate_as_collected_prom_help(wb, &p, homogeneous, prometheus_collector);
- if (unlikely(output_options & PROMETHEUS_OUTPUT_TYPES))
- buffer_sprintf(
- wb, "# TYPE %s_%s_%s%s %s\n", prefix, context, dimension, suffix, p.type);
- generate_as_collected_prom_metric(wb, &p, homogeneous, prometheus_collector, st->rrdlabels);
- }
- }
- else {
- // we need average or sum of the data
- time_t first_time = instance->after;
- time_t last_time = instance->before;
- NETDATA_DOUBLE value = exporting_calculate_value_from_stored_data(instance, rd, &last_time);
- if (!isnan(value) && !isinf(value)) {
- if (EXPORTING_OPTIONS_DATA_SOURCE(exporting_options) == EXPORTING_SOURCE_DATA_AVERAGE)
- suffix = "_average";
- else if (EXPORTING_OPTIONS_DATA_SOURCE(exporting_options) == EXPORTING_SOURCE_DATA_SUM)
- suffix = "_sum";
- prometheus_label_copy(
- dimension,
- (output_options & PROMETHEUS_OUTPUT_NAMES && rd->name) ? rrddim_name(rd) : rrddim_id(rd),
- PROMETHEUS_ELEMENT_MAX);
- buffer_flush(plabels_buffer);
- buffer_sprintf(plabels_buffer, "%1$schart=\"%2$s\",%1$sdimension=\"%3$s\",%1$sfamily=\"%4$s\"", plabels_prefix, chart, dimension, family);
- rrdlabels_walkthrough_read(st->rrdlabels, format_prometheus_chart_label_callback, plabels_buffer);
- if (unlikely(output_options & PROMETHEUS_OUTPUT_HELP))
- buffer_sprintf(
- wb,
- "# COMMENT %s_%s%s%s: dimension \"%s\", value is %s, gauge, dt %llu to %llu inclusive\n",
- prefix,
- context,
- units,
- suffix,
- (output_options & PROMETHEUS_OUTPUT_NAMES && rd->name) ? rrddim_name(rd) : rrddim_id(rd),
- rrdset_units(st),
- (unsigned long long)first_time,
- (unsigned long long)last_time);
- if (unlikely(output_options & PROMETHEUS_OUTPUT_TYPES))
- buffer_sprintf(wb, "# TYPE %s_%s%s%s gauge\n", prefix, context, units, suffix);
- if (output_options & PROMETHEUS_OUTPUT_TIMESTAMPS)
- buffer_sprintf(
- wb,
- "%s_%s%s%s{%s%s} " NETDATA_DOUBLE_FORMAT
- " %llu\n",
- prefix,
- context,
- units,
- suffix,
- buffer_tostring(plabels_buffer),
- labels,
- value,
- last_time * MSEC_PER_SEC);
- else
- buffer_sprintf(
- wb,
- "%s_%s%s%s{%s%s} " NETDATA_DOUBLE_FORMAT
- "\n",
- prefix,
- context,
- units,
- suffix,
- buffer_tostring(plabels_buffer),
- labels,
- value);
- }
- }
- }
- }
- rrddim_foreach_done(rd);
- }
- }
- rrdset_foreach_done(st);
- buffer_free(plabels_buffer);
- simple_pattern_free(filter);
- }
- /**
- * Get the last time time when a server accessed Netdata. Write information about an API request to a buffer.
- *
- * @param instance an instance data structure.
- * @param host a data collecting host.
- * @param wb the buffer to write to.
- * @param exporting_options options to configure what data is exported.
- * @param server the name of a Prometheus server..
- * @param now actual time.
- * @param output_options options to configure the format of the output.
- * @return Returns the last time when the server accessed Netdata.
- */
- static inline time_t prometheus_preparation(
- struct instance *instance,
- RRDHOST *host,
- BUFFER *wb,
- EXPORTING_OPTIONS exporting_options,
- const char *server,
- time_t now,
- PROMETHEUS_OUTPUT_OPTIONS output_options)
- {
- #ifndef UNIT_TESTING
- analytics_log_prometheus();
- #endif
- if (!server || !*server)
- server = "default";
- time_t after = prometheus_server_last_access(server, host, now);
- int first_seen = 0;
- if (!after) {
- after = now - instance->config.update_every;
- first_seen = 1;
- }
- if (after > now) {
- // oops! this should never happen
- after = now - instance->config.update_every;
- }
- if (output_options & PROMETHEUS_OUTPUT_HELP) {
- char *mode;
- if (EXPORTING_OPTIONS_DATA_SOURCE(exporting_options) == EXPORTING_SOURCE_DATA_AS_COLLECTED)
- mode = "as collected";
- else if (EXPORTING_OPTIONS_DATA_SOURCE(exporting_options) == EXPORTING_SOURCE_DATA_AVERAGE)
- mode = "average";
- else if (EXPORTING_OPTIONS_DATA_SOURCE(exporting_options) == EXPORTING_SOURCE_DATA_SUM)
- mode = "sum";
- else
- mode = "unknown";
- buffer_sprintf(
- wb,
- "# COMMENT netdata \"%s\" to %sprometheus \"%s\", source \"%s\", last seen %lu %s, time range %lu to %lu\n\n",
- rrdhost_hostname(host),
- (first_seen) ? "FIRST SEEN " : "",
- server,
- mode,
- (unsigned long)((first_seen) ? 0 : (now - after)),
- (first_seen) ? "never" : "seconds ago",
- (unsigned long)after,
- (unsigned long)now);
- }
- return after;
- }
- /**
- * Write metrics and auxiliary information for one host to a buffer.
- *
- * @param host a data collecting host.
- * @param filter_string a simple pattern filter.
- * @param wb the buffer to write to.
- * @param server the name of a Prometheus server.
- * @param prefix a prefix for every metric.
- * @param exporting_options options to configure what data is exported.
- * @param output_options options to configure the format of the output.
- */
- void rrd_stats_api_v1_charts_allmetrics_prometheus_single_host(
- RRDHOST *host,
- const char *filter_string,
- BUFFER *wb,
- const char *server,
- const char *prefix,
- EXPORTING_OPTIONS exporting_options,
- PROMETHEUS_OUTPUT_OPTIONS output_options)
- {
- if (unlikely(!prometheus_exporter_instance || !prometheus_exporter_instance->config.initialized))
- return;
- prometheus_exporter_instance->before = now_realtime_sec();
- // we start at the point we had stopped before
- prometheus_exporter_instance->after = prometheus_preparation(
- prometheus_exporter_instance,
- host,
- wb,
- exporting_options,
- server,
- prometheus_exporter_instance->before,
- output_options);
- rrd_stats_api_v1_charts_allmetrics_prometheus(
- prometheus_exporter_instance, host, filter_string, wb, prefix, exporting_options, 0, output_options);
- }
- /**
- * Write metrics and auxiliary information for all hosts to a buffer.
- *
- * @param host a data collecting host.
- * @param filter_string a simple pattern filter.
- * @param wb the buffer to write to.
- * @param server the name of a Prometheus server.
- * @param prefix a prefix for every metric.
- * @param exporting_options options to configure what data is exported.
- * @param output_options options to configure the format of the output.
- */
- void rrd_stats_api_v1_charts_allmetrics_prometheus_all_hosts(
- RRDHOST *host,
- const char *filter_string,
- BUFFER *wb,
- const char *server,
- const char *prefix,
- EXPORTING_OPTIONS exporting_options,
- PROMETHEUS_OUTPUT_OPTIONS output_options)
- {
- if (unlikely(!prometheus_exporter_instance || !prometheus_exporter_instance->config.initialized))
- return;
- prometheus_exporter_instance->before = now_realtime_sec();
- // we start at the point we had stopped before
- prometheus_exporter_instance->after = prometheus_preparation(
- prometheus_exporter_instance,
- host,
- wb,
- exporting_options,
- server,
- prometheus_exporter_instance->before,
- output_options);
- dfe_start_reentrant(rrdhost_root_index, host)
- {
- rrd_stats_api_v1_charts_allmetrics_prometheus(
- prometheus_exporter_instance, host, filter_string, wb, prefix, exporting_options, 1, output_options);
- }
- dfe_done(host);
- }
|