mirror of
https://git.mirrors.martin98.com/https://github.com/SigNoz/signoz
synced 2025-06-22 13:32:31 +08:00

* chore: add test validating collector doesn't spam logs for string ops on missing attributes * fix: pipelines filter: check if attrib is not nil before running string operator * chore: do a nil check for all but == and != ops
485 lines
10 KiB
Go
485 lines
10 KiB
Go
package logparsingpipeline
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/open-telemetry/opentelemetry-collector-contrib/pkg/stanza/entry"
|
|
. "github.com/smartystreets/goconvey/convey"
|
|
"github.com/stretchr/testify/require"
|
|
"go.signoz.io/signoz/pkg/query-service/model"
|
|
v3 "go.signoz.io/signoz/pkg/query-service/model/v3"
|
|
"go.signoz.io/signoz/pkg/query-service/utils"
|
|
)
|
|
|
|
var prepareProcessorTestData = []struct {
|
|
Name string
|
|
Operators []PipelineOperator
|
|
Output []PipelineOperator
|
|
}{
|
|
{
|
|
Name: "Last operator disabled",
|
|
Operators: []PipelineOperator{
|
|
{
|
|
ID: "t1",
|
|
Name: "t1",
|
|
Output: "t2",
|
|
Enabled: true,
|
|
},
|
|
{
|
|
ID: "t2",
|
|
Name: "t2",
|
|
Enabled: false,
|
|
},
|
|
},
|
|
Output: []PipelineOperator{
|
|
{
|
|
ID: "t1",
|
|
Name: "t1",
|
|
Enabled: true,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "Operator in middle disabled",
|
|
Operators: []PipelineOperator{
|
|
{
|
|
ID: "t1",
|
|
Name: "t1",
|
|
Output: "t2",
|
|
Enabled: true,
|
|
},
|
|
{
|
|
ID: "t2",
|
|
Name: "t2",
|
|
Output: "t3",
|
|
Enabled: false,
|
|
},
|
|
{
|
|
ID: "t3",
|
|
Name: "t3",
|
|
Enabled: true,
|
|
},
|
|
},
|
|
Output: []PipelineOperator{
|
|
{
|
|
ID: "t1",
|
|
Name: "t1",
|
|
Output: "t3",
|
|
Enabled: true,
|
|
},
|
|
{
|
|
ID: "t3",
|
|
Name: "t3",
|
|
Enabled: true,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "Single operator disabled",
|
|
Operators: []PipelineOperator{
|
|
{
|
|
ID: "t1",
|
|
Name: "t1",
|
|
Output: "t2",
|
|
Enabled: false,
|
|
},
|
|
},
|
|
Output: []PipelineOperator{},
|
|
},
|
|
{
|
|
Name: "Single operator enabled",
|
|
Operators: []PipelineOperator{
|
|
{
|
|
ID: "t1",
|
|
Name: "t1",
|
|
Enabled: true,
|
|
},
|
|
},
|
|
Output: []PipelineOperator{
|
|
{
|
|
ID: "t1",
|
|
Name: "t1",
|
|
Enabled: true,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "Empty operator",
|
|
Operators: []PipelineOperator{},
|
|
Output: []PipelineOperator{},
|
|
},
|
|
{
|
|
Name: "new test",
|
|
Operators: []PipelineOperator{
|
|
{
|
|
ID: "move_filename",
|
|
Output: "move_function",
|
|
Enabled: true,
|
|
Name: "move_filename",
|
|
},
|
|
{
|
|
ID: "move_function",
|
|
Output: "move_line",
|
|
Enabled: false,
|
|
Name: "move_function",
|
|
},
|
|
{
|
|
ID: "move_line",
|
|
Output: "move_lwp",
|
|
Enabled: true,
|
|
Name: "move_line",
|
|
},
|
|
{
|
|
ID: "move_lwp",
|
|
Output: "move_callid",
|
|
Enabled: true,
|
|
Name: "move_lwp",
|
|
},
|
|
{
|
|
ID: "move_callid",
|
|
Enabled: true,
|
|
Name: "move_lwp",
|
|
},
|
|
},
|
|
Output: []PipelineOperator{
|
|
{
|
|
ID: "move_filename",
|
|
Output: "move_line",
|
|
Enabled: true,
|
|
Name: "move_filename",
|
|
},
|
|
{
|
|
ID: "move_line",
|
|
Output: "move_lwp",
|
|
Enabled: true,
|
|
Name: "move_line",
|
|
},
|
|
{
|
|
ID: "move_lwp",
|
|
Output: "move_callid",
|
|
Enabled: true,
|
|
Name: "move_lwp",
|
|
},
|
|
{
|
|
ID: "move_callid",
|
|
Enabled: true,
|
|
Name: "move_lwp",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Name: "first op disabled",
|
|
Operators: []PipelineOperator{
|
|
{
|
|
ID: "move_filename",
|
|
Output: "move_function",
|
|
Enabled: false,
|
|
Name: "move_filename",
|
|
},
|
|
{
|
|
ID: "move_function",
|
|
Enabled: true,
|
|
Name: "move_function",
|
|
},
|
|
},
|
|
Output: []PipelineOperator{
|
|
{
|
|
ID: "move_function",
|
|
Enabled: true,
|
|
Name: "move_function",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
func TestPreparePipelineProcessor(t *testing.T) {
|
|
for _, test := range prepareProcessorTestData {
|
|
Convey(test.Name, t, func() {
|
|
res := getOperators(test.Operators)
|
|
So(res, ShouldResemble, test.Output)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestNoCollectorErrorsFromProcessorsForMismatchedLogs(t *testing.T) {
|
|
require := require.New(t)
|
|
|
|
testPipelineFilter := &v3.FilterSet{
|
|
Operator: "AND",
|
|
Items: []v3.FilterItem{
|
|
{
|
|
Key: v3.AttributeKey{
|
|
Key: "method",
|
|
DataType: v3.AttributeKeyDataTypeString,
|
|
Type: v3.AttributeKeyTypeTag,
|
|
},
|
|
Operator: "=",
|
|
Value: "GET",
|
|
},
|
|
},
|
|
}
|
|
makeTestPipeline := func(config []PipelineOperator) Pipeline {
|
|
return Pipeline{
|
|
OrderId: 1,
|
|
Name: "pipeline1",
|
|
Alias: "pipeline1",
|
|
Enabled: true,
|
|
Filter: testPipelineFilter,
|
|
Config: config,
|
|
}
|
|
}
|
|
|
|
makeTestLog := func(
|
|
body string,
|
|
attributes map[string]string,
|
|
) model.SignozLog {
|
|
attributes["method"] = "GET"
|
|
|
|
testTraceId, err := utils.RandomHex(16)
|
|
require.Nil(err)
|
|
|
|
testSpanId, err := utils.RandomHex(8)
|
|
require.Nil(err)
|
|
|
|
return model.SignozLog{
|
|
Timestamp: uint64(time.Now().UnixNano()),
|
|
Body: body,
|
|
Attributes_string: attributes,
|
|
Resources_string: attributes,
|
|
SeverityText: entry.Info.String(),
|
|
SeverityNumber: uint8(entry.Info),
|
|
SpanID: testSpanId,
|
|
TraceID: testTraceId,
|
|
}
|
|
}
|
|
|
|
testCases := []struct {
|
|
Name string
|
|
Operator PipelineOperator
|
|
NonMatchingLog model.SignozLog
|
|
}{
|
|
{
|
|
"regex processor should ignore log with missing field",
|
|
PipelineOperator{
|
|
ID: "regex",
|
|
Type: "regex_parser",
|
|
Enabled: true,
|
|
Name: "regex parser",
|
|
ParseFrom: "attributes.test_regex_target",
|
|
ParseTo: "attributes",
|
|
Regex: `^\s*(?P<json_data>{.*})\s*$`,
|
|
},
|
|
makeTestLog("mismatching log", map[string]string{}),
|
|
}, {
|
|
"regex processor should ignore non-matching log",
|
|
PipelineOperator{
|
|
ID: "regex",
|
|
Type: "regex_parser",
|
|
Enabled: true,
|
|
Name: "regex parser",
|
|
ParseFrom: "body",
|
|
ParseTo: "attributes",
|
|
Regex: `^\s*(?P<body_json>{.*})\s*$`,
|
|
},
|
|
makeTestLog("mismatching log", map[string]string{}),
|
|
}, {
|
|
"json parser should ignore logs with missing field.",
|
|
PipelineOperator{
|
|
ID: "json",
|
|
Type: "json_parser",
|
|
Enabled: true,
|
|
Name: "json parser",
|
|
ParseFrom: "attributes.test_json",
|
|
ParseTo: "attributes",
|
|
},
|
|
makeTestLog("mismatching log", map[string]string{}),
|
|
},
|
|
{
|
|
"json parser should ignore log with non JSON target field value",
|
|
PipelineOperator{
|
|
ID: "json",
|
|
Type: "json_parser",
|
|
Enabled: true,
|
|
Name: "json parser",
|
|
ParseFrom: "attributes.test_json",
|
|
ParseTo: "attributes",
|
|
},
|
|
makeTestLog("mismatching log", map[string]string{
|
|
"test_json": "bad json",
|
|
}),
|
|
}, {
|
|
"move parser should ignore non matching logs",
|
|
PipelineOperator{
|
|
ID: "move",
|
|
Type: "move",
|
|
Enabled: true,
|
|
Name: "move",
|
|
From: "attributes.test1",
|
|
To: "attributes.test2",
|
|
},
|
|
makeTestLog("mismatching log", map[string]string{}),
|
|
}, {
|
|
"copy parser should ignore non matching logs",
|
|
PipelineOperator{
|
|
ID: "copy",
|
|
Type: "copy",
|
|
Enabled: true,
|
|
Name: "copy",
|
|
From: "attributes.test1",
|
|
To: "attributes.test2",
|
|
},
|
|
makeTestLog("mismatching log", map[string]string{}),
|
|
}, {
|
|
"remove parser should ignore non matching logs",
|
|
PipelineOperator{
|
|
ID: "remove",
|
|
Type: "remove",
|
|
Enabled: true,
|
|
Name: "remove",
|
|
Field: "attributes.test",
|
|
},
|
|
makeTestLog("mismatching log", map[string]string{}),
|
|
},
|
|
// TODO(Raj): see if there is an error scenario for grok parser.
|
|
// TODO(Raj): see if there is an error scenario for trace parser.
|
|
// TODO(Raj): see if there is an error scenario for Add operator.
|
|
}
|
|
|
|
for _, testCase := range testCases {
|
|
testPipelines := []Pipeline{makeTestPipeline([]PipelineOperator{testCase.Operator})}
|
|
|
|
result, collectorWarnAndErrorLogs, err := SimulatePipelinesProcessing(
|
|
context.Background(),
|
|
testPipelines,
|
|
[]model.SignozLog{testCase.NonMatchingLog},
|
|
)
|
|
require.Nil(err)
|
|
require.Equal(0, len(collectorWarnAndErrorLogs), strings.Join(collectorWarnAndErrorLogs, "\n"))
|
|
require.Equal(1, len(result))
|
|
}
|
|
}
|
|
|
|
func TestResourceFiltersWork(t *testing.T) {
|
|
require := require.New(t)
|
|
|
|
testPipeline := Pipeline{
|
|
OrderId: 1,
|
|
Name: "pipeline1",
|
|
Alias: "pipeline1",
|
|
Enabled: true,
|
|
Filter: &v3.FilterSet{
|
|
Operator: "AND",
|
|
Items: []v3.FilterItem{
|
|
{
|
|
Key: v3.AttributeKey{
|
|
Key: "service",
|
|
DataType: v3.AttributeKeyDataTypeString,
|
|
Type: v3.AttributeKeyTypeResource,
|
|
},
|
|
Operator: "=",
|
|
Value: "nginx",
|
|
},
|
|
},
|
|
},
|
|
Config: []PipelineOperator{
|
|
{
|
|
ID: "add",
|
|
Type: "add",
|
|
Enabled: true,
|
|
Name: "add",
|
|
Field: "attributes.test",
|
|
Value: "test-value",
|
|
},
|
|
},
|
|
}
|
|
|
|
testLog := model.SignozLog{
|
|
Timestamp: uint64(time.Now().UnixNano()),
|
|
Body: "test log",
|
|
Attributes_string: map[string]string{},
|
|
Resources_string: map[string]string{
|
|
"service": "nginx",
|
|
},
|
|
SeverityText: entry.Info.String(),
|
|
SeverityNumber: uint8(entry.Info),
|
|
SpanID: "",
|
|
TraceID: "",
|
|
}
|
|
|
|
result, collectorWarnAndErrorLogs, err := SimulatePipelinesProcessing(
|
|
context.Background(),
|
|
[]Pipeline{testPipeline},
|
|
[]model.SignozLog{testLog},
|
|
)
|
|
require.Nil(err)
|
|
require.Equal(0, len(collectorWarnAndErrorLogs), strings.Join(collectorWarnAndErrorLogs, "\n"))
|
|
require.Equal(1, len(result))
|
|
|
|
require.Equal(result[0].Attributes_string["test"], "test-value")
|
|
}
|
|
|
|
func TestPipelineFilterWithStringOpsShouldNotSpamWarningsIfAttributeIsMissing(t *testing.T) {
|
|
require := require.New(t)
|
|
|
|
for _, operator := range []v3.FilterOperator{
|
|
v3.FilterOperatorContains,
|
|
v3.FilterOperatorNotContains,
|
|
v3.FilterOperatorRegex,
|
|
v3.FilterOperatorNotRegex,
|
|
} {
|
|
testPipeline := Pipeline{
|
|
OrderId: 1,
|
|
Name: "pipeline1",
|
|
Alias: "pipeline1",
|
|
Enabled: true,
|
|
Filter: &v3.FilterSet{
|
|
Operator: "AND",
|
|
Items: []v3.FilterItem{
|
|
{
|
|
Key: v3.AttributeKey{
|
|
Key: "service",
|
|
DataType: v3.AttributeKeyDataTypeString,
|
|
Type: v3.AttributeKeyTypeResource,
|
|
},
|
|
Operator: operator,
|
|
Value: "nginx",
|
|
},
|
|
},
|
|
},
|
|
Config: []PipelineOperator{
|
|
{
|
|
ID: "add",
|
|
Type: "add",
|
|
Enabled: true,
|
|
Name: "add",
|
|
Field: "attributes.test",
|
|
Value: "test-value",
|
|
},
|
|
},
|
|
}
|
|
|
|
testLog := model.SignozLog{
|
|
Timestamp: uint64(time.Now().UnixNano()),
|
|
Body: "test log",
|
|
Attributes_string: map[string]string{},
|
|
Resources_string: map[string]string{},
|
|
SeverityText: entry.Info.String(),
|
|
SeverityNumber: uint8(entry.Info),
|
|
SpanID: "",
|
|
TraceID: "",
|
|
}
|
|
|
|
result, collectorWarnAndErrorLogs, err := SimulatePipelinesProcessing(
|
|
context.Background(),
|
|
[]Pipeline{testPipeline},
|
|
[]model.SignozLog{testLog},
|
|
)
|
|
require.Nil(err)
|
|
require.Equal(0, len(collectorWarnAndErrorLogs), strings.Join(collectorWarnAndErrorLogs, "\n"))
|
|
require.Equal(1, len(result))
|
|
}
|
|
}
|