From 7329eebc9290f8d414cfeb0cec9cf531560562e6 Mon Sep 17 00:00:00 2001 From: Jiawei Tan Date: Mon, 18 Nov 2024 10:16:27 -0500 Subject: [PATCH] fix(condition): handling of missing source and/or target keys --- condition/number_equal_to.go | 17 +++++--- condition/number_equal_to_test.go | 64 +++++++++++++++++++++++++++++++ condition/string_equal_to.go | 17 +++++--- condition/string_equal_to_test.go | 64 +++++++++++++++++++++++++++++++ 4 files changed, 152 insertions(+), 10 deletions(-) diff --git a/condition/number_equal_to.go b/condition/number_equal_to.go index f5accfff..84af552b 100644 --- a/condition/number_equal_to.go +++ b/condition/number_equal_to.go @@ -38,14 +38,21 @@ func (insp *numberEqualTo) Condition(ctx context.Context, msg *message.Message) return insp.match(f, compare), nil } + source_value := msg.GetValue(insp.conf.Object.SourceKey) + target_value := msg.GetValue(insp.conf.Object.TargetKey) - target := msg.GetValue(insp.conf.Object.TargetKey) - if target.Exists() { - compare = target.Float() + // for gjson's GetValue, if the path is empty string (indicating source key or target key is not present), + // the Result.Exists() will return false + // If source or target key is present but value cannot be found, always return false + if !source_value.Exists() || insp.conf.Object.TargetKey != "" && !target_value.Exists() { + return false, nil + } + + if target_value.Exists() { + compare = target_value.Float() } - v := msg.GetValue(insp.conf.Object.SourceKey) - return insp.match(v.Float(), compare), nil + return insp.match(source_value.Float(), compare), nil } func (c *numberEqualTo) match(f float64, t float64) bool { diff --git a/condition/number_equal_to_test.go b/condition/number_equal_to_test.go index 91f264b0..588a9337 100644 --- a/condition/number_equal_to_test.go +++ b/condition/number_equal_to_test.go @@ -134,6 +134,70 @@ var numberEqualToTests = []struct { []byte(`{"foo": 100, "bar": 200}`), false, }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "baz", + }, + "value": 0, + }, + }, + []byte(`{"foo": 100, "bar": 200}`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "baz", + "target_key": "bar", + }, + }, + }, + []byte(`{"foo": 100, "bar": 200}`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "target_key": "abc", + }, + }, + }, + []byte(`100`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "foo", + "target_key": "baz", + }, + }, + }, + []byte(`{"foo": 100, "bar": 200}`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "baz", + "target_key": "abc", + }, + }, + }, + []byte(`{"foo": 100, "bar": 200}`), + false, + }, } func TestNumberEqualTo(t *testing.T) { diff --git a/condition/string_equal_to.go b/condition/string_equal_to.go index 562ee614..d5c03993 100644 --- a/condition/string_equal_to.go +++ b/condition/string_equal_to.go @@ -40,14 +40,21 @@ func (insp *stringEqualTo) Condition(ctx context.Context, msg *message.Message) return bytes.Equal(msg.Data(), compare), nil } - target := msg.GetValue(insp.conf.Object.TargetKey) + source_value := msg.GetValue(insp.conf.Object.SourceKey) + target_value := msg.GetValue(insp.conf.Object.TargetKey) - if target.Exists() { - compare = target.Bytes() + // for gjson's GetValue, if the path is empty string (indicating source key or target key is not present), + // the Result.Exists() will return false + // If source or target key is present but value cannot be found, always return false + if !source_value.Exists() || insp.conf.Object.TargetKey != "" && !target_value.Exists() { + return false, nil + } + + if target_value.Exists() { + compare = target_value.Bytes() } - value := msg.GetValue(insp.conf.Object.SourceKey) - return bytes.Equal(value.Bytes(), compare), nil + return bytes.Equal(source_value.Bytes(), compare), nil } func (c *stringEqualTo) String() string { diff --git a/condition/string_equal_to_test.go b/condition/string_equal_to_test.go index 33785ca3..93a7cc2b 100644 --- a/condition/string_equal_to_test.go +++ b/condition/string_equal_to_test.go @@ -73,6 +73,70 @@ var stringEqualToTests = []struct { []byte(`{"foo":"abc", "bar":"def"}`), false, }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "foo", + }, + "value": "abc", + }, + }, + []byte(`{"bar": "abc", "baz": "0"}`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "foo", + }, + "value": "", + }, + }, + []byte(`{"bar": "abc", "baz": "0"}`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "foo", + "target_key": "baz", + }, + }, + }, + []byte(`{"bar": "abc", "baz": "0"}`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "target_key": "foo", + }, + }, + }, + []byte(`{"bar": "abc", "baz": "0"}`), + false, + }, + { + "pass", + config.Config{ + Settings: map[string]interface{}{ + "object": map[string]interface{}{ + "source_key": "bar", + "target_key": "foo", + }, + }, + }, + []byte(`{"bar": "abc", "baz": "0"}`), + false, + }, } func TestStringEqualTo(t *testing.T) {