mirror of
https://github.com/netdata/netdata.git
synced 2025-04-14 17:48:37 +00:00

* cleanup alerts * fix references * fix references * fix references * load alerts once and apply them to each node * simplify health_create_alarm_entry() * Compile without warnings with compiler flags: -Wall -Wextra -Wformat=2 -Wshadow -Wno-format-nonliteral -Winit-self * code re-organization and cleanup * generate patterns when applying prototypes; give unique dyncfg names to all alerts * eval expressions keep the source and the parsed_as as STRING pointers * renamed host to node in dyncfg ids * renamed host to node in dyncfg ids * add all cloud roles to the list of parsed X-Netdata-Role header and also default to member access level * working functionality * code re-organization: moved health event-loop to a new file, moved health globals to health.c * rrdcalctemplate is removed; alert_cfg is removed; foreach dimension is removed; RRDCALCs are now instanciated only when they are linked to RRDSETs * dyncfg alert prototypes initialization for alerts * health dyncfg split to separate file * cleanup not-needed code * normalize matches between parsing and json * also detect !* for disabled alerts * dyncfg capability disabled * Store alert config part1 * Add rrdlabels_common_count * wip health variables lookup without indexes * Improve rrdlabels_common_count by reusing rrdlabels_find_label_with_key_unsafe with an additional parameter * working variables with runtime lookup * working variables with runtime lookup * delete rrddimvar and rrdfamily index * remove rrdsetvar; now all variables are in RRDVARs inside hosts and charts * added /api/v1/variable that resolves a variable the same way alerts do * remove rrdcalc from eval * remove debug code * remove duplicate assignment * Fix memory leak * all alert variables are now handled by alert_variable_lookup() and EVAL is now independent of alerts * hide all internal structures of EVAL * Enable -Wformat flag Signed-off-by: Tasos Katsoulas <tasos@netdata.cloud> * Adjust binding for calculation, warning, critical * Remove unused macro * Update config hash id * use the right info and summary in alerts log * use synchronous queries for alerts * Handle cases when config_hash_id is missing from health_log * remove deadlock from health worker * parsing to json payload for health alert prototypes * cleaner parsing and avoiding memory leaks in case of duplicate members in json * fix left-over rename of function * Keep original lookup field to send to the cloud Cleanup / rename function to store config Remove unused DEFINEs, functions * Use ac->lookup * link jobs to the host when the template is registered; do not accept running a function without a host * full dyncfg support for health alerts, except action TEST * working dyncfg additions, updates, removals * fixed missing source, wrong status updates * add alerts by type, component, classification, recipient and module at the /api/v2/alerts endpoint * fix dyncfg unittest * rename functions * generalize the json-c parser macros and move them to libnetdata * report progress when enabling and disabling dyncfg templates * moved rrdcalc and rrdvar to health * update alarms * added schema for alerts; separated alert_action_options from rrdr_options; restructured the json payload for alerts * enable parsed json alerts; allow sending back accepted but disabled * added format_version for alerts payload; enables/disables status now is also inheritted by the status of the rules; fixed variable names in json output * remove the RRDHOST pointer from DYNCFG * Fix command field submitted to the cloud * do not send updates to creation requests, for DYNCFG jobs --------- Signed-off-by: Tasos Katsoulas <tasos@netdata.cloud> Co-authored-by: Stelios Fragkakis <52996999+stelfrag@users.noreply.github.com> Co-authored-by: Tasos Katsoulas <tasos@netdata.cloud> Co-authored-by: ilyam8 <ilya@netdata.cloud>
298 lines
9.3 KiB
C
298 lines
9.3 KiB
C
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
#include "../libnetdata.h"
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
static struct {
|
|
DYNCFG_TYPE type;
|
|
const char *name;
|
|
} dyncfg_types[] = {
|
|
{ .type = DYNCFG_TYPE_SINGLE, .name = "single" },
|
|
{ .type = DYNCFG_TYPE_TEMPLATE, .name = "template" },
|
|
{ .type = DYNCFG_TYPE_JOB, .name = "job" },
|
|
};
|
|
|
|
DYNCFG_TYPE dyncfg_type2id(const char *type) {
|
|
if(!type || !*type)
|
|
return DYNCFG_TYPE_SINGLE;
|
|
|
|
size_t entries = sizeof(dyncfg_types) / sizeof(dyncfg_types[0]);
|
|
for(size_t i = 0; i < entries ;i++) {
|
|
if(strcmp(dyncfg_types[i].name, type) == 0)
|
|
return dyncfg_types[i].type;
|
|
}
|
|
|
|
return DYNCFG_TYPE_SINGLE;
|
|
}
|
|
|
|
const char *dyncfg_id2type(DYNCFG_TYPE type) {
|
|
size_t entries = sizeof(dyncfg_types) / sizeof(dyncfg_types[0]);
|
|
for(size_t i = 0; i < entries ;i++) {
|
|
if(type == dyncfg_types[i].type)
|
|
return dyncfg_types[i].name;
|
|
}
|
|
|
|
return "single";
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
static struct {
|
|
DYNCFG_SOURCE_TYPE source_type;
|
|
const char *name;
|
|
} dyncfg_source_types[] = {
|
|
{ .source_type = DYNCFG_SOURCE_TYPE_INTERNAL, .name = "internal" },
|
|
{ .source_type = DYNCFG_SOURCE_TYPE_STOCK, .name = "stock" },
|
|
{ .source_type = DYNCFG_SOURCE_TYPE_USER, .name = "user" },
|
|
{ .source_type = DYNCFG_SOURCE_TYPE_DYNCFG, .name = "dyncfg" },
|
|
{ .source_type = DYNCFG_SOURCE_TYPE_DISCOVERED, .name = "discovered" },
|
|
};
|
|
|
|
DYNCFG_SOURCE_TYPE dyncfg_source_type2id(const char *source_type) {
|
|
if(!source_type || !*source_type)
|
|
return DYNCFG_SOURCE_TYPE_INTERNAL;
|
|
|
|
size_t entries = sizeof(dyncfg_source_types) / sizeof(dyncfg_source_types[0]);
|
|
for(size_t i = 0; i < entries ;i++) {
|
|
if(strcmp(dyncfg_source_types[i].name, source_type) == 0)
|
|
return dyncfg_source_types[i].source_type;
|
|
}
|
|
|
|
return DYNCFG_SOURCE_TYPE_INTERNAL;
|
|
}
|
|
|
|
const char *dyncfg_id2source_type(DYNCFG_SOURCE_TYPE source_type) {
|
|
size_t entries = sizeof(dyncfg_source_types) / sizeof(dyncfg_source_types[0]);
|
|
for(size_t i = 0; i < entries ;i++) {
|
|
if(source_type == dyncfg_source_types[i].source_type)
|
|
return dyncfg_source_types[i].name;
|
|
}
|
|
|
|
return "internal";
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
static struct {
|
|
DYNCFG_STATUS status;
|
|
const char *name;
|
|
} dyncfg_statuses[] = {
|
|
{ .status = DYNCFG_STATUS_NONE, .name = "none" },
|
|
{ .status = DYNCFG_STATUS_ACCEPTED, .name = "accepted" },
|
|
{ .status = DYNCFG_STATUS_RUNNING, .name = "running" },
|
|
{ .status = DYNCFG_STATUS_FAILED, .name = "failed" },
|
|
{ .status = DYNCFG_STATUS_DISABLED, .name = "disabled" },
|
|
{ .status = DYNCFG_STATUS_ORPHAN, .name = "orphan" },
|
|
{ .status = DYNCFG_STATUS_INCOMPLETE, .name = "incomplete" },
|
|
};
|
|
|
|
DYNCFG_STATUS dyncfg_status2id(const char *status) {
|
|
if(!status || !*status)
|
|
return DYNCFG_STATUS_NONE;
|
|
|
|
size_t entries = sizeof(dyncfg_statuses) / sizeof(dyncfg_statuses[0]);
|
|
for(size_t i = 0; i < entries ;i++) {
|
|
if(strcmp(dyncfg_statuses[i].name, status) == 0)
|
|
return dyncfg_statuses[i].status;
|
|
}
|
|
|
|
return DYNCFG_STATUS_NONE;
|
|
}
|
|
|
|
const char *dyncfg_id2status(DYNCFG_STATUS status) {
|
|
size_t entries = sizeof(dyncfg_statuses) / sizeof(dyncfg_statuses[0]);
|
|
for(size_t i = 0; i < entries ;i++) {
|
|
if(status == dyncfg_statuses[i].status)
|
|
return dyncfg_statuses[i].name;
|
|
}
|
|
|
|
return "none";
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
static struct {
|
|
DYNCFG_CMDS cmd;
|
|
const char *name;
|
|
} cmd_map[] = {
|
|
{ .cmd = DYNCFG_CMD_GET, .name = "get" },
|
|
{ .cmd = DYNCFG_CMD_SCHEMA, .name = "schema" },
|
|
{ .cmd = DYNCFG_CMD_UPDATE, .name = "update" },
|
|
{ .cmd = DYNCFG_CMD_ADD, .name = "add" },
|
|
{ .cmd = DYNCFG_CMD_TEST, .name = "test" },
|
|
{ .cmd = DYNCFG_CMD_REMOVE, .name = "remove" },
|
|
{ .cmd = DYNCFG_CMD_ENABLE, .name = "enable" },
|
|
{ .cmd = DYNCFG_CMD_DISABLE, .name = "disable" },
|
|
{ .cmd = DYNCFG_CMD_RESTART, .name = "restart" }
|
|
};
|
|
|
|
const char *dyncfg_id2cmd_one(DYNCFG_CMDS cmd) {
|
|
for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
|
|
if(cmd == cmd_map[i].cmd)
|
|
return cmd_map[i].name;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
DYNCFG_CMDS dyncfg_cmds2id(const char *cmds) {
|
|
if(!cmds || !*cmds)
|
|
return DYNCFG_CMD_NONE;
|
|
|
|
DYNCFG_CMDS result = DYNCFG_CMD_NONE;
|
|
const char *p = cmds;
|
|
size_t len, i;
|
|
|
|
while (*p) {
|
|
// Skip any leading spaces
|
|
while (*p == ' ') p++;
|
|
|
|
// Find the end of the current word
|
|
const char *end = p;
|
|
while (*end && *end != ' ') end++;
|
|
len = end - p;
|
|
|
|
// Compare with known commands
|
|
for (i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
|
|
if (strncmp(p, cmd_map[i].name, len) == 0 && cmd_map[i].name[len] == '\0') {
|
|
result |= cmd_map[i].cmd;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Move to the next word
|
|
p = end;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void dyncfg_cmds2fp(DYNCFG_CMDS cmds, FILE *fp) {
|
|
for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
|
|
if(cmds & cmd_map[i].cmd)
|
|
fprintf(fp, "%s ", cmd_map[i].name);
|
|
}
|
|
}
|
|
|
|
void dyncfg_cmds2json_array(DYNCFG_CMDS cmds, const char *key, BUFFER *wb) {
|
|
buffer_json_member_add_array(wb, key);
|
|
for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
|
|
if(cmds & cmd_map[i].cmd)
|
|
buffer_json_add_array_item_string(wb, cmd_map[i].name);
|
|
}
|
|
buffer_json_array_close(wb);
|
|
}
|
|
|
|
void dyncfg_cmds2buffer(DYNCFG_CMDS cmds, BUFFER *wb) {
|
|
size_t added = 0;
|
|
for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
|
|
if(cmds & cmd_map[i].cmd) {
|
|
if(added)
|
|
buffer_fast_strcat(wb, " ", 1);
|
|
|
|
buffer_strcat(wb, cmd_map[i].name);
|
|
added++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
bool dyncfg_is_valid_id(const char *id) {
|
|
const char *s = id;
|
|
|
|
while(*s) {
|
|
if(isspace(*s) || *s == '\'') return false;
|
|
s++;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
char *dyncfg_escape_id_for_filename(const char *id) {
|
|
if (id == NULL) return NULL;
|
|
|
|
// Allocate memory for the worst case, where every character is escaped.
|
|
char *escaped = mallocz(strlen(id) * 3 + 1); // Each char can become '%XX', plus '\0'
|
|
if (!escaped) return NULL;
|
|
|
|
const char *src = id;
|
|
char *dest = escaped;
|
|
|
|
while (*src) {
|
|
if (*src == '/' || isspace(*src) || !isprint(*src)) {
|
|
sprintf(dest, "%%%02X", (unsigned char)*src);
|
|
dest += 3;
|
|
} else {
|
|
*dest++ = *src;
|
|
}
|
|
src++;
|
|
}
|
|
|
|
*dest = '\0';
|
|
return escaped;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
int dyncfg_default_response(BUFFER *wb, int code, const char *msg) {
|
|
buffer_flush(wb);
|
|
wb->content_type = CT_APPLICATION_JSON;
|
|
wb->expires = now_realtime_sec();
|
|
|
|
buffer_json_initialize(wb, "\"", "\"", 0, true, BUFFER_JSON_OPTIONS_MINIFY);
|
|
buffer_json_member_add_uint64(wb, "status", code);
|
|
buffer_json_member_add_string(wb, "message", msg);
|
|
buffer_json_finalize(wb);
|
|
|
|
return code;
|
|
}
|
|
|
|
int dyncfg_node_find_and_call(DICTIONARY *dyncfg_nodes, const char *transaction, const char *function,
|
|
usec_t *stop_monotonic_ut, bool *cancelled,
|
|
BUFFER *payload, const char *source, BUFFER *result) {
|
|
if(!function || !*function)
|
|
return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "command received is empty");
|
|
|
|
char buf[strlen(function) + 1];
|
|
memcpy(buf, function, sizeof(buf));
|
|
|
|
char *words[MAX_FUNCTION_PARAMETERS]; // an array of pointers for the words in this line
|
|
size_t num_words = quoted_strings_splitter_pluginsd(buf, words, MAX_FUNCTION_PARAMETERS);
|
|
|
|
const char *id = get_word(words, num_words, 1);
|
|
const char *action = get_word(words, num_words, 2);
|
|
const char *add_name = get_word(words, num_words, 3);
|
|
|
|
if(!id || !*id)
|
|
return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "dyncfg node: id is missing from the request");
|
|
|
|
if(!action || !*action)
|
|
return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "dyncfg node: action is missing from the request");
|
|
|
|
DYNCFG_CMDS cmd = dyncfg_cmds2id(action);
|
|
if(cmd == DYNCFG_CMD_NONE)
|
|
return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "dyncfg node: action given in request is unknown");
|
|
|
|
const DICTIONARY_ITEM *item = dictionary_get_and_acquire_item(dyncfg_nodes, id);
|
|
if(!item)
|
|
return dyncfg_default_response(result, HTTP_RESP_NOT_FOUND, "dyncfg node: id is not found");
|
|
|
|
struct dyncfg_node *df = dictionary_acquired_item_value(item);
|
|
|
|
buffer_flush(result);
|
|
result->content_type = CT_APPLICATION_JSON;
|
|
|
|
int code = df->cb(transaction, id, cmd, add_name, payload, stop_monotonic_ut, cancelled, result, source, df->data);
|
|
|
|
if(!result->expires)
|
|
result->expires = now_realtime_sec();
|
|
|
|
if(!buffer_tostring(result))
|
|
dyncfg_default_response(result, code, "");
|
|
|
|
dictionary_acquired_item_release(dyncfg_nodes, item);
|
|
|
|
return code;
|
|
}
|