pkg/inst-api-semconv/instrumenter/db/db_client_extractor.go (96 lines of code) (raw):
// Copyright (c) 2024 Alibaba Group Holding Ltd.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package db
import (
"context"
"fmt"
"os"
"strconv"
"github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/inst-api/instrumenter"
"github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/inst-api/utils"
"go.opentelemetry.io/otel/attribute"
semconv "go.opentelemetry.io/otel/semconv/v1.30.0"
)
// TODO: remove server.address and put it into NetworkAttributesExtractor
const EnvDBExperimentalEnabled = "OTEL_INSTRUMENTATION_DB_EXPERIMENTAL_ENABLE"
type dbExperimentalEnabler struct {
enabled bool
}
func (d dbExperimentalEnabler) Enable() bool {
return d.enabled
}
var experimentalAttributesEnabler instrumenter.InstrumentEnabler = &dbExperimentalEnabler{
// SQL params parsing is not enabled by default.
enabled: func() bool {
val, err := strconv.ParseBool(os.Getenv(EnvDBExperimentalEnabled))
if err != nil {
return false
}
return val
}(),
}
type DbClientCommonAttrsExtractor[REQUEST any, RESPONSE any, GETTER DbClientCommonAttrsGetter[REQUEST]] struct {
Getter GETTER
AttributesFilter func(attrs []attribute.KeyValue) []attribute.KeyValue
}
func (d *DbClientCommonAttrsExtractor[REQUEST, RESPONSE, GETTER]) GetSpanKey() attribute.Key {
return utils.DB_CLIENT_KEY
}
func (d *DbClientCommonAttrsExtractor[REQUEST, RESPONSE, GETTER]) OnStart(attributes []attribute.KeyValue, parentContext context.Context, request REQUEST) ([]attribute.KeyValue, context.Context) {
return attributes, parentContext
}
func (d *DbClientCommonAttrsExtractor[REQUEST, RESPONSE, GETTER]) OnEnd(attrs []attribute.KeyValue, context context.Context, request REQUEST, response RESPONSE, err error) ([]attribute.KeyValue, context.Context) {
attrs = append(attrs, attribute.KeyValue{
Key: semconv.DBSystemNameKey,
Value: attribute.StringValue(d.Getter.GetSystem(request)),
})
if d.AttributesFilter != nil {
attrs = d.AttributesFilter(attrs)
}
return attrs, context
}
type DbClientAttrsExtractor[REQUEST any, RESPONSE any, GETTER DbClientAttrsGetter[REQUEST]] struct {
Base DbClientCommonAttrsExtractor[REQUEST, RESPONSE, GETTER]
}
func (d *DbClientAttrsExtractor[REQUEST, RESPONSE, GETTER]) OnStart(attrs []attribute.KeyValue, parentContext context.Context, request REQUEST) ([]attribute.KeyValue, context.Context) {
attrs, parentContext = d.Base.OnStart(attrs, parentContext, request)
if d.Base.AttributesFilter != nil {
attrs = d.Base.AttributesFilter(attrs)
}
return attrs, parentContext
}
func (d *DbClientAttrsExtractor[REQUEST, RESPONSE, GETTER]) OnEnd(attrs []attribute.KeyValue, context context.Context, request REQUEST, response RESPONSE, err error) ([]attribute.KeyValue, context.Context) {
attrs, context = d.Base.OnEnd(attrs, context, request, response, err)
attrs = append(attrs, attribute.KeyValue{
Key: semconv.DBQueryTextKey,
Value: attribute.StringValue(d.Base.Getter.GetStatement(request)),
}, attribute.KeyValue{
Key: semconv.DBOperationNameKey,
Value: attribute.StringValue(d.Base.Getter.GetOperation(request)),
}, attribute.KeyValue{
Key: semconv.ServerAddressKey,
Value: attribute.StringValue(d.Base.Getter.GetServerAddress(request)),
}, attribute.KeyValue{
Key: semconv.DBCollectionNameKey,
Value: attribute.StringValue(d.Base.Getter.GetCollection(request)),
})
batchSize := d.Base.Getter.GetBatchSize(request)
if batchSize > 0 {
attrs = append(attrs, attribute.KeyValue{Key: semconv.DBOperationBatchSizeKey, Value: attribute.IntValue(batchSize)})
}
dbNameSpace := d.Base.Getter.GetDbNamespace(request)
if dbNameSpace != "" {
attrs = append(attrs, attribute.KeyValue{Key: semconv.DBNamespaceKey, Value: attribute.StringValue(dbNameSpace)})
}
if d.Base.AttributesFilter != nil {
attrs = d.Base.AttributesFilter(attrs)
}
if experimentalAttributesEnabler.Enable() {
params := d.Base.Getter.GetParameters(request)
if len(params) > 0 {
for i, param := range params {
attrs = append(attrs, attribute.String("db.query.parameter."+strconv.Itoa(i), fmt.Sprintf("%v", param)))
}
}
}
return attrs, context
}
func (d *DbClientAttrsExtractor[REQUEST, RESPONSE, GETTER]) GetSpanKey() attribute.Key {
return utils.DB_CLIENT_KEY
}
// TODO: sanitize sql
// TODO: batch sql