mirror of
https://git.mirrors.martin98.com/https://github.com/SigNoz/signoz
synced 2025-07-29 13:31:59 +08:00
117 lines
3.7 KiB
Go
117 lines
3.7 KiB
Go
package telemetrymetadata
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
schema "github.com/SigNoz/signoz-otel-collector/cmd/signozschemamigrator/schema_migrator"
|
|
"github.com/SigNoz/signoz/pkg/errors"
|
|
qbtypes "github.com/SigNoz/signoz/pkg/types/querybuildertypes/querybuildertypesv5"
|
|
"github.com/SigNoz/signoz/pkg/types/telemetrytypes"
|
|
"golang.org/x/exp/maps"
|
|
)
|
|
|
|
var (
|
|
attributeMetadataColumns = map[string]*schema.Column{
|
|
"resource_attributes": {Name: "resource_attributes", Type: schema.MapColumnType{
|
|
KeyType: schema.LowCardinalityColumnType{ElementType: schema.ColumnTypeString},
|
|
ValueType: schema.ColumnTypeString,
|
|
}},
|
|
"attributes": {Name: "attributes", Type: schema.MapColumnType{
|
|
KeyType: schema.LowCardinalityColumnType{ElementType: schema.ColumnTypeString},
|
|
ValueType: schema.ColumnTypeString,
|
|
}},
|
|
}
|
|
)
|
|
|
|
type fieldMapper struct {
|
|
}
|
|
|
|
func NewFieldMapper() qbtypes.FieldMapper {
|
|
return &fieldMapper{}
|
|
}
|
|
|
|
func (m *fieldMapper) getColumn(ctx context.Context, key *telemetrytypes.TelemetryFieldKey) (*schema.Column, error) {
|
|
switch key.FieldContext {
|
|
case telemetrytypes.FieldContextResource:
|
|
return attributeMetadataColumns["resource_attributes"], nil
|
|
case telemetrytypes.FieldContextAttribute:
|
|
return attributeMetadataColumns["attributes"], nil
|
|
}
|
|
return nil, qbtypes.ErrColumnNotFound
|
|
}
|
|
|
|
func (m *fieldMapper) ColumnFor(ctx context.Context, key *telemetrytypes.TelemetryFieldKey) (*schema.Column, error) {
|
|
column, err := m.getColumn(ctx, key)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return column, nil
|
|
}
|
|
|
|
func (m *fieldMapper) FieldFor(ctx context.Context, key *telemetrytypes.TelemetryFieldKey) (string, error) {
|
|
column, err := m.getColumn(ctx, key)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
switch column.Type {
|
|
case schema.MapColumnType{
|
|
KeyType: schema.LowCardinalityColumnType{ElementType: schema.ColumnTypeString},
|
|
ValueType: schema.ColumnTypeString,
|
|
}:
|
|
return fmt.Sprintf("%s['%s']", column.Name, key.Name), nil
|
|
}
|
|
return column.Name, nil
|
|
}
|
|
|
|
func (m *fieldMapper) ColumnExpressionFor(
|
|
ctx context.Context,
|
|
field *telemetrytypes.TelemetryFieldKey,
|
|
keys map[string][]*telemetrytypes.TelemetryFieldKey,
|
|
) (string, error) {
|
|
|
|
colName, err := m.FieldFor(ctx, field)
|
|
if errors.Is(err, qbtypes.ErrColumnNotFound) {
|
|
// the key didn't have the right context to be added to the query
|
|
// we try to use the context we know of
|
|
keysForField := keys[field.Name]
|
|
if len(keysForField) == 0 {
|
|
// is it a static field?
|
|
if _, ok := attributeMetadataColumns[field.Name]; ok {
|
|
// if it is, attach the column name directly
|
|
field.FieldContext = telemetrytypes.FieldContextSpan
|
|
colName, _ = m.FieldFor(ctx, field)
|
|
} else {
|
|
// - the context is not provided
|
|
// - there are not keys for the field
|
|
// - it is not a static field
|
|
// - the next best thing to do is see if there is a typo
|
|
// and suggest a correction
|
|
correction, found := telemetrytypes.SuggestCorrection(field.Name, maps.Keys(keys))
|
|
if found {
|
|
// we found a close match, in the error message send the suggestion
|
|
return "", errors.Wrapf(err, errors.TypeInvalidInput, errors.CodeInvalidInput, correction)
|
|
} else {
|
|
// not even a close match, return an error
|
|
return "", err
|
|
}
|
|
}
|
|
} else if len(keysForField) == 1 {
|
|
// we have a single key for the field, use it
|
|
colName, _ = m.FieldFor(ctx, keysForField[0])
|
|
} else {
|
|
// select any non-empty value from the keys
|
|
args := []string{}
|
|
for _, key := range keysForField {
|
|
colName, _ = m.FieldFor(ctx, key)
|
|
args = append(args, fmt.Sprintf("toString(%s) != '', toString(%s)", colName, colName))
|
|
}
|
|
colName = fmt.Sprintf("multiIf(%s)", strings.Join(args, ", "))
|
|
}
|
|
}
|
|
|
|
return fmt.Sprintf("%s AS `%s`", colName, field.Name), nil
|
|
}
|