mirror of
https://github.com/SigNoz/signoz.git
synced 2026-06-25 09:30:31 +01:00
Some checks failed
build-staging / prepare (push) Has been cancelled
build-staging / js-build (push) Has been cancelled
build-staging / go-build (push) Has been cancelled
build-staging / staging (push) Has been cancelled
Release Drafter / update_release_draft (push) Has been cancelled
* chore: send warning instead of error for unseen metrics and missing (metric, key) * chore: update integration test * chore: fix integration test * chore: fix test * chore: add unit test for missing key
156 lines
5.3 KiB
Go
156 lines
5.3 KiB
Go
package querier
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
"time"
|
|
|
|
cmock "github.com/SigNoz/clickhouse-go-mock"
|
|
|
|
"github.com/SigNoz/signoz/pkg/flagger/flaggertest"
|
|
"github.com/SigNoz/signoz/pkg/instrumentation/instrumentationtest"
|
|
"github.com/SigNoz/signoz/pkg/telemetrystore"
|
|
"github.com/SigNoz/signoz/pkg/telemetrystore/telemetrystoretest"
|
|
"github.com/SigNoz/signoz/pkg/types/metrictypes"
|
|
qbtypes "github.com/SigNoz/signoz/pkg/types/querybuildertypes/querybuildertypesv5"
|
|
"github.com/SigNoz/signoz/pkg/types/telemetrytypes"
|
|
"github.com/SigNoz/signoz/pkg/types/telemetrytypes/telemetrytypestest"
|
|
"github.com/SigNoz/signoz/pkg/valuer"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
type queryMatcherAny struct{}
|
|
|
|
func (m *queryMatcherAny) Match(string, string) error { return nil }
|
|
|
|
// mockMetricStmtBuilder implements qbtypes.StatementBuilder[qbtypes.MetricAggregation]
|
|
// and returns a fixed query string so the mock ClickHouse can match it.
|
|
type mockMetricStmtBuilder struct{}
|
|
|
|
func (m *mockMetricStmtBuilder) Build(_ context.Context, _, _ uint64, _ qbtypes.RequestType, _ qbtypes.QueryBuilderQuery[qbtypes.MetricAggregation], _ map[string]qbtypes.VariableItem) (*qbtypes.Statement, error) {
|
|
return &qbtypes.Statement{
|
|
Query: "SELECT ts, value FROM signoz_metrics",
|
|
Args: nil,
|
|
}, nil
|
|
}
|
|
|
|
func TestQueryRange_MetricTypeMissing(t *testing.T) {
|
|
// When a metric has UnspecifiedType and is not found in the metadata store,
|
|
// the querier should return an empty result with a warning instead of an error.
|
|
providerSettings := instrumentationtest.New().ToProviderSettings()
|
|
metadataStore := telemetrytypestest.NewMockMetadataStore()
|
|
|
|
q := New(
|
|
providerSettings,
|
|
nil, // telemetryStore
|
|
metadataStore,
|
|
nil, // prometheus
|
|
nil, // traceStmtBuilder
|
|
nil, // logStmtBuilder
|
|
nil, // auditStmtBuilder
|
|
nil, // metricStmtBuilder
|
|
nil, // meterStmtBuilder
|
|
nil, // traceOperatorStmtBuilder
|
|
nil, // bucketCache
|
|
flaggertest.New(t), // flagger
|
|
)
|
|
|
|
req := &qbtypes.QueryRangeRequest{
|
|
Start: uint64(time.Now().Add(-5 * time.Minute).UnixMilli()),
|
|
End: uint64(time.Now().UnixMilli()),
|
|
RequestType: qbtypes.RequestTypeTimeSeries,
|
|
CompositeQuery: qbtypes.CompositeQuery{
|
|
Queries: []qbtypes.QueryEnvelope{{
|
|
Type: qbtypes.QueryTypeBuilder,
|
|
Spec: qbtypes.QueryBuilderQuery[qbtypes.MetricAggregation]{
|
|
Name: "A",
|
|
StepInterval: qbtypes.Step{Duration: time.Minute},
|
|
Aggregations: []qbtypes.MetricAggregation{
|
|
{
|
|
MetricName: "unknown_metric",
|
|
Temporality: metrictypes.Cumulative,
|
|
TimeAggregation: metrictypes.TimeAggregationRate,
|
|
SpaceAggregation: metrictypes.SpaceAggregationSum,
|
|
},
|
|
},
|
|
Signal: telemetrytypes.SignalMetrics,
|
|
},
|
|
}},
|
|
},
|
|
}
|
|
|
|
resp, err := q.QueryRange(context.Background(), valuer.GenerateUUID(), req)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, resp)
|
|
require.NotNil(t, resp.Warning)
|
|
|
|
require.Len(t, resp.Warning.Warnings, 1)
|
|
assert.Contains(t, resp.Warning.Warnings[0].Message, "unknown_metric")
|
|
assert.Contains(t, resp.Warning.Warnings[0].Message, "has never been received")
|
|
}
|
|
|
|
func TestQueryRange_MetricTypeFromStore(t *testing.T) {
|
|
// When a metric has UnspecifiedType but the metadata store returns a valid type,
|
|
// the metric should not be treated as missing.
|
|
providerSettings := instrumentationtest.New().ToProviderSettings()
|
|
metadataStore := telemetrytypestest.NewMockMetadataStore()
|
|
metadataStore.TypeMap["my_metric"] = metrictypes.SumType
|
|
metadataStore.TemporalityMap["my_metric"] = metrictypes.Cumulative
|
|
|
|
telemetryStore := telemetrystoretest.New(telemetrystore.Config{}, &queryMatcherAny{})
|
|
|
|
cols := []cmock.ColumnType{
|
|
{Name: "ts", Type: "DateTime"},
|
|
{Name: "value", Type: "Float64"},
|
|
}
|
|
rows := cmock.NewRows(cols, [][]any{
|
|
{time.Now(), float64(42)},
|
|
})
|
|
telemetryStore.Mock().
|
|
ExpectQuery("SELECT any").
|
|
WillReturnRows(rows)
|
|
|
|
q := New(
|
|
providerSettings,
|
|
telemetryStore,
|
|
metadataStore,
|
|
nil, // prometheus
|
|
nil, // traceStmtBuilder
|
|
nil, // logStmtBuilder
|
|
nil, // auditStmtBuilder
|
|
&mockMetricStmtBuilder{}, // metricStmtBuilder
|
|
nil, // meterStmtBuilder
|
|
nil, // traceOperatorStmtBuilder
|
|
nil, // bucketCache
|
|
flaggertest.New(t), // flagger
|
|
)
|
|
|
|
req := &qbtypes.QueryRangeRequest{
|
|
Start: uint64(time.Now().Add(-5 * time.Minute).UnixMilli()),
|
|
End: uint64(time.Now().UnixMilli()),
|
|
RequestType: qbtypes.RequestTypeTimeSeries,
|
|
CompositeQuery: qbtypes.CompositeQuery{
|
|
Queries: []qbtypes.QueryEnvelope{{
|
|
Type: qbtypes.QueryTypeBuilder,
|
|
Spec: qbtypes.QueryBuilderQuery[qbtypes.MetricAggregation]{
|
|
Name: "A",
|
|
StepInterval: qbtypes.Step{Duration: time.Minute},
|
|
Aggregations: []qbtypes.MetricAggregation{
|
|
{
|
|
MetricName: "my_metric",
|
|
TimeAggregation: metrictypes.TimeAggregationRate,
|
|
SpaceAggregation: metrictypes.SpaceAggregationSum,
|
|
},
|
|
},
|
|
Signal: telemetrytypes.SignalMetrics,
|
|
},
|
|
}},
|
|
},
|
|
}
|
|
|
|
resp, err := q.QueryRange(context.Background(), valuer.GenerateUUID(), req)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, resp)
|
|
}
|