pkg/rules/gorm/setup.go (91 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 gorm
import (
"context"
"os"
_ "unsafe"
"github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/api"
driver "github.com/go-sql-driver/mysql"
"gorm.io/driver/mysql"
"gorm.io/gorm"
)
var contextKey = "otel-context"
var requestKey = "otel-request"
type gormInnerEnabler struct {
enabled bool
}
func (g gormInnerEnabler) Enable() bool {
return g.enabled
}
var gormEnabler = gormInnerEnabler{os.Getenv("OTEL_INSTRUMENTATION_GORM_ENABLED") != "false"}
var gormInstrumenter = BuildGormInstrumenter()
//go:linkname afterGormOpen gorm.io/gorm.afterGormOpen
func afterGormOpen(call api.CallContext, db *gorm.DB, err error) {
if !gormEnabler.Enable() {
return
}
if err != nil || db == nil {
return
}
// add the callback
_ = db.Callback().Create().Before("gorm:create").Register("otel_create_create_span", beforeCallback("", "create"))
_ = db.Callback().Query().Before("gorm:query").Register("otel_create_query_span", beforeCallback("", "query"))
_ = db.Callback().Update().Before("gorm:update").Register("otel_create_update_span", beforeCallback("", "update"))
_ = db.Callback().Delete().Before("gorm:delete").Register("otel_create_delete_span", beforeCallback("", "delete"))
_ = db.Callback().Row().Before("gorm:row").Register("otel_create_row_span", beforeCallback("", "row"))
_ = db.Callback().Raw().Before("gorm:raw").Register("otel_create_raw_span", beforeCallback("", "raw"))
// after database operation
_ = db.Callback().Create().After("gorm:create").Register("otel_end_create_span", afterCallback(""))
_ = db.Callback().Query().After("gorm:query").Register("otel_end_query_span", afterCallback(""))
_ = db.Callback().Update().After("gorm:update").Register("otel_end_update_span", afterCallback(""))
_ = db.Callback().Delete().After("gorm:delete").Register("otel_end_delete_span", afterCallback(""))
_ = db.Callback().Row().After("gorm:row").Register("otel_end_row_span", afterCallback(""))
_ = db.Callback().Raw().After("gorm:raw").Register("otel_end_raw_span", afterCallback(""))
}
func beforeCallback(endpoint string, op string) func(db *gorm.DB) {
return func(db *gorm.DB) {
dbName, addr, system, user := getDbInfo(db.Config.Dialector)
request := gormRequest{
DbName: dbName,
Endpoint: addr,
Operation: op,
User: user,
System: system,
}
ctx := gormInstrumenter.Start(context.Background(), request)
db.Set(contextKey, ctx)
db.Set(requestKey, request)
}
}
func afterCallback(endpoint string) func(db *gorm.DB) {
return func(db *gorm.DB) {
iCtx, ok := db.Get(contextKey)
if !ok {
return
}
ctx, ok := iCtx.(context.Context)
if !ok {
return
}
iRequest, ok := db.Get(requestKey)
if !ok {
return
}
request, ok := iRequest.(gormRequest)
if !ok {
return
}
gormInstrumenter.End(ctx, request, nil, db.Statement.Error)
}
}
func getDbInfo(dial gorm.Dialector) (string, string, string, string) {
// TODO: support other database
res, ok := dial.(*mysql.Dialector)
if !ok {
return "", "", "", ""
}
if cfg, ok := res.DbInfo.(*driver.Config); ok {
return cfg.DBName, cfg.Addr, "mysql", cfg.User
}
cfg, err := driver.ParseDSN(res.Config.DSN)
if err != nil {
return "", "", "", ""
}
res.DbInfo = cfg
return cfg.DBName, cfg.Addr, "mysql", cfg.User
}