diff options
Diffstat (limited to 'src/fluent-bit/plugins/in_node_exporter_metrics/ne_meminfo_linux.c')
-rw-r--r-- | src/fluent-bit/plugins/in_node_exporter_metrics/ne_meminfo_linux.c | 283 |
1 files changed, 283 insertions, 0 deletions
diff --git a/src/fluent-bit/plugins/in_node_exporter_metrics/ne_meminfo_linux.c b/src/fluent-bit/plugins/in_node_exporter_metrics/ne_meminfo_linux.c new file mode 100644 index 000000000..3189d53c0 --- /dev/null +++ b/src/fluent-bit/plugins/in_node_exporter_metrics/ne_meminfo_linux.c @@ -0,0 +1,283 @@ +/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ + +/* Fluent Bit + * ========== + * Copyright (C) 2015-2022 The Fluent Bit Authors + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include <fluent-bit/flb_info.h> +#include <fluent-bit/flb_hash_table.h> +#include <fluent-bit/flb_sds.h> +#include <fluent-bit/flb_input_plugin.h> + +#include "ne.h" +#include "ne_utils.h" + +#include <unistd.h> + +static int meminfo_configure(struct flb_ne *ctx) +{ + int ret; + int parts; + int len; + char *p; + char desc[] = "Memory information field "; + struct cmt_gauge *g; + struct mk_list *head; + struct mk_list list; + struct mk_list split_list; + struct flb_slist_entry *entry; + struct flb_slist_entry *line; + flb_sds_t metric_name; + flb_sds_t metric_desc; + + /* Initialize hash table */ + ctx->meminfo_ht = flb_hash_table_create(FLB_HASH_TABLE_EVICT_NONE, 16, 0); + if (!ctx->meminfo_ht) { + return -1; + } + + mk_list_init(&list); + mk_list_init(&split_list); + + ret = ne_utils_file_read_lines(ctx->path_procfs, "/meminfo", &list); + if (ret == -1) { + return -1; + } + metric_name = flb_sds_create_size(128); + if (!metric_name) { + flb_hash_table_destroy(ctx->meminfo_ht); + flb_slist_destroy(&list); + return -1; + } + + metric_desc = flb_sds_create_size(256); + if (!metric_desc) { + flb_hash_table_destroy(ctx->meminfo_ht); + flb_slist_destroy(&list); + return -1; + } + + mk_list_foreach(head, &list) { + line = mk_list_entry(head, struct flb_slist_entry, _head); + + mk_list_init(&split_list); + ret = flb_slist_split_string(&split_list, line->str, ' ', -1); + if (ret == -1) { + continue; + } + parts = ret; + + /* set metric name */ + entry = mk_list_entry_first(&split_list, struct flb_slist_entry, _head); + + if ((p = strstr(entry->str, "(anon)")) || + (p = strstr(entry->str, "(file)"))) { + *p = '_'; + len = flb_sds_len(entry->str) - 2; + flb_sds_len_set(entry->str, len); + } + else { + len = flb_sds_len(entry->str) - 1; + flb_sds_len_set(entry->str, len); + } + entry->str[len] = '\0'; + + flb_sds_len_set(metric_name, 0); + flb_sds_cat(metric_name, entry->str, flb_sds_len(entry->str)); + + /* Metric description */ + flb_sds_len_set(metric_desc, 0); + flb_sds_cat(metric_desc, desc, sizeof(desc) - 1); + + if (parts == 2) { + /* No unit */ + flb_sds_cat(metric_desc, metric_name, flb_sds_len(metric_name)); + flb_sds_cat(metric_desc, ".", 1); + + g = cmt_gauge_create(ctx->cmt, "node", "memory", metric_name, + metric_desc, + 0, NULL); + if (!g) { + flb_slist_destroy(&split_list); + goto error; + } + } + else if (parts == 3) { + /* It has an extra 'kB' string in the line */ + flb_sds_cat(metric_name, "_bytes", 6); + flb_sds_cat(metric_desc, metric_name, flb_sds_len(metric_name)); + flb_sds_cat(metric_desc, ".", 1); + g = cmt_gauge_create(ctx->cmt, "node", "memory", metric_name, + metric_desc, + 0, NULL); + if (!g) { + flb_slist_destroy(&split_list); + goto error; + } + } + else { + flb_slist_destroy(&split_list); + continue; + } + flb_slist_destroy(&split_list); + + /* + * Register the gauge context into the hash table: note that depending + * of the number of parts in the list, if it contains the extra 'kB' + * the metric name gets appended the '_bytes' string. + */ + ret = flb_hash_table_add(ctx->meminfo_ht, + metric_name, flb_sds_len(metric_name), g, 0); + if (ret == -1) { + flb_plg_error(ctx->ins, + "could not add hash for metric: %s", metric_name); + goto error; + } + } + + flb_sds_destroy(metric_name); + flb_sds_destroy(metric_desc); + flb_slist_destroy(&list); + return 0; + + error: + flb_sds_destroy(metric_name); + flb_sds_destroy(metric_desc); + flb_slist_destroy(&list); + return -1; +} + +static int meminfo_update(struct flb_ne *ctx) +{ + int i = 0; + int ret; + int len; + int parts; + uint64_t ts; + double val; + size_t out_size; + char *p; + flb_sds_t tmp; + flb_sds_t metric_name = NULL; + struct cmt_gauge *g; + struct mk_list *head; + struct mk_list list; + struct mk_list split_list; + struct flb_slist_entry *line; + struct flb_slist_entry *entry; + + mk_list_init(&list); + ret = ne_utils_file_read_lines(ctx->path_procfs, "/meminfo", &list); + if (ret == -1) { + return -1; + } + + ts = cfl_time_now(); + + mk_list_foreach(head, &list) { + line = mk_list_entry(head, struct flb_slist_entry, _head); + + mk_list_init(&split_list); + ret = flb_slist_split_string(&split_list, line->str, ' ', -1); + if (ret == -1) { + continue; + } + parts = ret; + if (parts == 0) { + flb_slist_destroy(&split_list); + continue; + } + + /* Metric name */ + entry = mk_list_entry_first(&split_list, struct flb_slist_entry, _head); + metric_name = entry->str; + + if ((p = strstr(entry->str, "(anon)")) || + (p = strstr(entry->str, "(file)"))) { + *p = '_'; + len = flb_sds_len(metric_name) - 1; + flb_sds_len_set(metric_name, len); + } + + /* Metric value */ + entry = mk_list_entry_next(&split_list, struct flb_slist_entry, _head, + &entry->_head); + + ret = ne_utils_str_to_double(entry->str, &val); + if (ret == -1) { + i++; + flb_slist_destroy(&split_list); + } + + g = NULL; + if (parts == 2) { + /* Metric name is the same, no extra bytes */ + ret = flb_hash_table_get(ctx->meminfo_ht, + metric_name, flb_sds_len(metric_name) - 1, + (void *) &g, &out_size); + } + else if (parts == 3) { + /* Compose new metric name */ + tmp = flb_sds_create_size(256); + flb_sds_cat_safe(&tmp, metric_name, flb_sds_len(metric_name) - 1); + flb_sds_cat_safe(&tmp, "_bytes", 6); + + /* Get metric context */ + ret = flb_hash_table_get(ctx->meminfo_ht, + tmp, flb_sds_len(tmp), + (void *) &g, &out_size); + flb_sds_destroy(tmp); + + /* Value is in kB, convert to bytes */ + val *= 1024; + } + + if (!g) { + flb_plg_error(ctx->ins, + "gauge content for metric '%s' not found", + metric_name); + flb_slist_destroy(&split_list); + continue; + } + + /* Update metric */ + cmt_gauge_set(g, ts, val, 0, NULL); + flb_slist_destroy(&split_list); + } + + flb_slist_destroy(&list); + return 0; +} + +int ne_meminfo_init(struct flb_ne *ctx) +{ + meminfo_configure(ctx); + return 0; +} + +int ne_meminfo_update(struct flb_ne *ctx) +{ + meminfo_update(ctx); + return 0; +} + +int ne_meminfo_exit(struct flb_ne *ctx) +{ + if (ctx->meminfo_ht) { + flb_hash_table_destroy(ctx->meminfo_ht); + } + return 0; +} |