summaryrefslogtreecommitdiffstats
path: root/src/database/rrdlabels.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/database/rrdlabels.c')
-rw-r--r--src/database/rrdlabels.c125
1 files changed, 103 insertions, 22 deletions
diff --git a/src/database/rrdlabels.c b/src/database/rrdlabels.c
index 633287cba9..929d0064e9 100644
--- a/src/database/rrdlabels.c
+++ b/src/database/rrdlabels.c
@@ -977,7 +977,7 @@ int rrdlabels_walkthrough_read(RRDLABELS *labels, int (*callback)(const char *na
lfe_start_read(labels, lb, ls)
{
ret = callback(string2str(lb->index.key), string2str(lb->index.value), ls, data);
- if (ret < 0)
+ if (ret != 0)
break;
}
lfe_done(labels);
@@ -1125,9 +1125,12 @@ static int simple_pattern_match_name_only_callback(const char *name, const char
// we return -1 to stop the walkthrough on first match
t->searches++;
- if(simple_pattern_matches(t->pattern, name)) return -1;
+ SIMPLE_PATTERN_RESULT rc = simple_pattern_matches_extract(t->pattern, name, NULL, 0);
- return 0;
+ if(rc == SP_MATCHED_NEGATIVE)
+ return -1;
+
+ return rc == SP_MATCHED_POSITIVE;
}
static int simple_pattern_match_name_and_value_callback(const char *name, const char *value, RRDLABEL_SRC ls __maybe_unused, void *data) {
@@ -1154,13 +1157,15 @@ static int simple_pattern_match_name_and_value_callback(const char *name, const
*dst = '\0';
t->searches++;
- if(simple_pattern_matches_length_extract(t->pattern, tmp, dst - tmp, NULL, 0) == SP_MATCHED_POSITIVE)
+ SIMPLE_PATTERN_RESULT rc = simple_pattern_matches_length_extract(t->pattern, tmp, dst - tmp, NULL, 0);
+
+ if(rc == SP_MATCHED_NEGATIVE)
return -1;
- return 0;
+ return rc == SP_MATCHED_POSITIVE ? 1 : 0;
}
-bool rrdlabels_match_simple_pattern_parsed(RRDLABELS *labels, SIMPLE_PATTERN *pattern, char equal, size_t *searches) {
+SIMPLE_PATTERN_RESULT rrdlabels_match_simple_pattern_parsed(RRDLABELS *labels, SIMPLE_PATTERN *pattern, char equal, size_t *searches) {
if (!labels) return false;
struct simple_pattern_match_name_value t = {
@@ -1174,7 +1179,10 @@ bool rrdlabels_match_simple_pattern_parsed(RRDLABELS *labels, SIMPLE_PATTERN *pa
if(searches)
*searches = t.searches;
- return (ret == -1)?true:false;
+ if(ret < 0)
+ return SP_MATCHED_NEGATIVE;
+
+ return (ret > 0)?SP_MATCHED_POSITIVE:SP_NOT_MATCHED;
}
bool rrdlabels_match_simple_pattern(RRDLABELS *labels, const char *simple_pattern_txt) {
@@ -1191,11 +1199,11 @@ bool rrdlabels_match_simple_pattern(RRDLABELS *labels, const char *simple_patter
}
}
- bool ret = rrdlabels_match_simple_pattern_parsed(labels, pattern, equal, NULL);
+ SIMPLE_PATTERN_RESULT ret = rrdlabels_match_simple_pattern_parsed(labels, pattern, equal, NULL);
simple_pattern_free(pattern);
- return ret;
+ return ret == SP_MATCHED_POSITIVE;
}
@@ -1388,8 +1396,7 @@ bool pattern_array_label_match(
struct pattern_array *pa,
RRDLABELS *labels,
char eq,
- size_t *searches,
- bool (*callback_function)(RRDLABELS *, SIMPLE_PATTERN *, char, size_t *))
+ size_t *searches)
{
if (!pa || !labels)
return true;
@@ -1398,14 +1405,23 @@ bool pattern_array_label_match(
Word_t Index = 0;
bool first_then_next = true;
while ((Pvalue = JudyLFirstThenNext(pa->JudyL, &Index, &first_then_next))) {
+ // for each label key in the patterns array
+
struct pattern_array_item *pai = *Pvalue;
- bool match = false;
- for (Word_t i = 1; !match && i <= pai->size; i++) {
+ SIMPLE_PATTERN_RESULT match = SP_NOT_MATCHED ;
+ for (Word_t i = 1; i <= pai->size; i++) {
+ // for each pattern in the label key pattern list
+
if (!(Pvalue = JudyLGet(pai->JudyL, i, PJE0)) || !*Pvalue)
continue;
- match = callback_function(labels, (SIMPLE_PATTERN *)(*Pvalue), eq, searches);
+
+ match = rrdlabels_match_simple_pattern_parsed(labels, (SIMPLE_PATTERN *)(*Pvalue), eq, searches);
+
+ if(match != SP_NOT_MATCHED)
+ break;
}
- if (!match)
+
+ if (match != SP_MATCHED_POSITIVE)
return false;
}
return true;
@@ -1481,6 +1497,7 @@ void pattern_array_free(struct pattern_array *pa)
string_freez((STRING *)Index);
(void) JudyLDel(&(pa->JudyL), Index, PJE0);
+ freez(pai);
Index = 0;
}
freez(pa);
@@ -1669,7 +1686,7 @@ static int rrdlabels_walkthrough_index_read(RRDLABELS *labels, int (*callback)(c
lfe_start_read(labels, lb, ls)
{
ret = callback(string2str(lb->index.key), string2str(lb->index.value), ls, index, data);
- if (ret < 0)
+ if (ret != 0)
break;
index++;
}
@@ -1705,25 +1722,25 @@ static int rrdlabels_unittest_pattern_check()
bool match;
struct pattern_array *pa = pattern_array_add_key_value(NULL, "_module", "wrong_module", '=');
- match = pattern_array_label_match(pa, labels, '=', NULL, rrdlabels_match_simple_pattern_parsed);
+ match = pattern_array_label_match(pa, labels, '=', NULL);
// This should not match: _module in ("wrong_module")
if (match)
rc++;
pattern_array_add_key_value(pa, "_module", "disk_detection", '=');
- match = pattern_array_label_match(pa, labels, '=', NULL, rrdlabels_match_simple_pattern_parsed);
+ match = pattern_array_label_match(pa, labels, '=', NULL);
// This should match: _module in ("wrong_module","disk_detection")
if (!match)
rc++;
pattern_array_add_key_value(pa, "key1", "wrong_key1_value", '=');
- match = pattern_array_label_match(pa, labels, '=', NULL, rrdlabels_match_simple_pattern_parsed);
+ match = pattern_array_label_match(pa, labels, '=', NULL);
// This should not match: _module in ("wrong_module","disk_detection") AND key1 in ("wrong_key1_value")
if (match)
rc++;
pattern_array_add_key_value(pa, "key1", "value1", '=');
- match = pattern_array_label_match(pa, labels, '=', NULL, rrdlabels_match_simple_pattern_parsed);
+ match = pattern_array_label_match(pa, labels, '=', NULL);
// This should match: _module in ("wrong_module","disk_detection") AND key1 in ("wrong_key1_value", "value1")
if (!match)
rc++;
@@ -1734,13 +1751,13 @@ static int rrdlabels_unittest_pattern_check()
sp = simple_pattern_create("key3=*phant", SIMPLE_PATTERN_DEFAULT_WEB_SEPARATORS, SIMPLE_PATTERN_EXACT, true);
pattern_array_add_lblkey_with_sp(pa, "key3", sp);
- match = pattern_array_label_match(pa, labels, '=', NULL, rrdlabels_match_simple_pattern_parsed);
+ match = pattern_array_label_match(pa, labels, '=', NULL);
// This should match: _module in ("wrong_module","disk_detection") AND key1 in ("wrong_key1_value", "value1") AND key2 in ("cat* !d*") AND key3 in ("*phant")
if (!match)
rc++;
rrdlabels_add(labels, "key3", "now_fail", RRDLABEL_SRC_CONFIG);
- match = pattern_array_label_match(pa, labels, '=', NULL, rrdlabels_match_simple_pattern_parsed);
+ match = pattern_array_label_match(pa, labels, '=', NULL);
// This should not match: _module in ("wrong_module","disk_detection") AND key1 in ("wrong_key1_value", "value1") AND key2 in ("cat* !d*") AND key3 in ("*phant")
if (match)
rc++;
@@ -1832,6 +1849,69 @@ static int rrdlabels_unittest_migrate_check()
return rc;
}
+struct pattern_array *trim_and_add_key_to_values(struct pattern_array *pa, const char *key, STRING *input);
+static int rrdlabels_unittest_check_pattern_list(RRDLABELS *labels, const char *pattern, bool expected) {
+ fprintf(stderr, "rrdlabels_match_simple_pattern(labels, \"%s\") ... ", pattern);
+
+ STRING *str = string_strdupz(pattern);
+ struct pattern_array *pa = trim_and_add_key_to_values(NULL, NULL, str);
+
+ bool ret = pattern_array_label_match(pa, labels, '=', NULL);
+
+ fprintf(stderr, "%s, got %s expected %s\n", (ret == expected)?"OK":"FAILED", ret?"true":"false", expected?"true":"false");
+
+ string_freez(str);
+ pattern_array_free(pa);
+
+ return (ret == expected)?0:1;
+}
+
+static int rrdlabels_unittest_host_chart_labels() {
+ fprintf(stderr, "\n%s() tests\n", __FUNCTION__);
+
+ int errors = 0;
+
+ RRDLABELS *labels = rrdlabels_create();
+ rrdlabels_add(labels, "_hostname", "hostname1", RRDLABEL_SRC_CONFIG);
+ rrdlabels_add(labels, "_os", "linux", RRDLABEL_SRC_CONFIG);
+ rrdlabels_add(labels, "_distro", "ubuntu", RRDLABEL_SRC_CONFIG);
+
+ // match a single key
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=*", true);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=!*", false);
+
+ // conflicting keys (some positive, some negative)
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=* _os=!*", false);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=!* _os=*", false);
+
+ // the user uses a key that is not there
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_not_a_key=*", false);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_not_a_key=!*", false);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_not_a_key=* _hostname=* _os=*", false);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_not_a_key=!* _hostname=* _os=*", false);
+
+ // positive and negative matches on the same key
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=!*invalid* !*bad* *name*", true);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=*name* !*invalid* !*bad*", true);
+
+ // positive and negative matches on the same key with catch all
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=!*invalid* !*bad* *", true);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=* !*invalid* !*bad*", true);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=!*invalid* !*name* *", false);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=* !*invalid* !*name*", true);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=*name* !*", true);
+
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=!*name* _os=l*", false);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_os=l* hostname=!*name*", false);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=*name* _hostname=*", true);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_hostname=*name* _os=l*", true);
+ errors += rrdlabels_unittest_check_pattern_list(labels, "_os=l* _hostname=*name*", true);
+
+ rrdlabels_destroy(labels);
+
+ return errors;
+}
+
static int rrdlabels_unittest_check_simple_pattern(RRDLABELS *labels, const char *pattern, bool expected) {
fprintf(stderr, "rrdlabels_match_simple_pattern(labels, \"%s\") ... ", pattern);
@@ -1924,6 +2004,7 @@ int rrdlabels_unittest(void) {
errors += rrdlabels_unittest_sanitization();
errors += rrdlabels_unittest_add_pairs();
errors += rrdlabels_unittest_simple_pattern();
+ errors += rrdlabels_unittest_host_chart_labels();
errors += rrdlabels_unittest_double_check();
errors += rrdlabels_unittest_migrate_check();
errors += rrdlabels_unittest_pattern_check();