forked from smacker/opentracing-gorm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathotgorm.go
109 lines (98 loc) · 3.6 KB
/
otgorm.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
package otgorm
import (
"context"
"fmt"
"strings"
opentracing "github.com/opentracing/opentracing-go"
"github.com/opentracing/opentracing-go/ext"
"gorm.io/gorm"
)
const (
parentSpanGormKey = "opentracingParentSpan"
spanGormKey = "opentracingSpan"
)
// SetSpanToGorm sets span to gorm settings, returns cloned DB
func SetSpanToGorm(ctx context.Context, db *gorm.DB) *gorm.DB {
if ctx == nil {
return db
}
parentSpan := opentracing.SpanFromContext(ctx)
if parentSpan == nil {
return db
}
return db.Set(parentSpanGormKey, parentSpan)
}
// AddGormCallbacks adds callbacks for tracing, you should call SetSpanToGorm to make them work
func AddGormCallbacks(db *gorm.DB) {
callbacks := newCallbacks()
registerCallbacks(db, "create", callbacks)
registerCallbacks(db, "query", callbacks)
registerCallbacks(db, "update", callbacks)
registerCallbacks(db, "delete", callbacks)
registerCallbacks(db, "row_query", callbacks)
}
type callbacks struct{}
func newCallbacks() *callbacks {
return &callbacks{}
}
func (c *callbacks) beforeCreate(db *gorm.DB) { c.before(db) }
func (c *callbacks) afterCreate(db *gorm.DB) { c.after(db, "INSERT") }
func (c *callbacks) beforeQuery(db *gorm.DB) { c.before(db) }
func (c *callbacks) afterQuery(db *gorm.DB) { c.after(db, "SELECT") }
func (c *callbacks) beforeUpdate(db *gorm.DB) { c.before(db) }
func (c *callbacks) afterUpdate(db *gorm.DB) { c.after(db, "UPDATE") }
func (c *callbacks) beforeDelete(db *gorm.DB) { c.before(db) }
func (c *callbacks) afterDelete(db *gorm.DB) { c.after(db, "DELETE") }
func (c *callbacks) beforeRowQuery(db *gorm.DB) { c.before(db) }
func (c *callbacks) afterRowQuery(db *gorm.DB) { c.after(db, "") }
func (c *callbacks) before(db *gorm.DB) {
val, ok := db.Get(parentSpanGormKey)
if !ok {
return
}
parentSpan := val.(opentracing.Span)
tr := parentSpan.Tracer()
sp := tr.StartSpan("sql", opentracing.ChildOf(parentSpan.Context()))
ext.DBType.Set(sp, "sql")
db.Set(spanGormKey, sp)
}
func (c *callbacks) after(db *gorm.DB, operation string) {
val, ok := db.Get(spanGormKey)
if !ok {
return
}
sp := val.(opentracing.Span)
sql := db.Statement.SQL.String()
if operation == "" {
operation = strings.ToUpper(strings.Split(sql, " ")[0])
}
ext.Error.Set(sp, db.Error != nil)
ext.DBStatement.Set(sp, sql)
sp.SetTag("db.table", db.Statement.Table)
sp.SetTag("db.method", operation)
sp.SetTag("db.err", db.Error != nil)
sp.SetTag("db.count", db.RowsAffected)
sp.Finish()
}
func registerCallbacks(db *gorm.DB, name string, c *callbacks) {
beforeName := fmt.Sprintf("tracing:%v_before", name)
afterName := fmt.Sprintf("tracing:%v_after", name)
gormCallbackName := fmt.Sprintf("gorm:%v", name)
switch name {
case "create":
db.Callback().Create().Before(gormCallbackName).Register(beforeName, c.beforeCreate)
db.Callback().Create().After(gormCallbackName).Register(afterName, c.afterCreate)
case "query":
db.Callback().Query().Before(gormCallbackName).Register(beforeName, c.beforeQuery)
db.Callback().Query().After(gormCallbackName).Register(afterName, c.afterQuery)
case "update":
db.Callback().Update().Before(gormCallbackName).Register(beforeName, c.beforeUpdate)
db.Callback().Update().After(gormCallbackName).Register(afterName, c.afterUpdate)
case "delete":
db.Callback().Delete().Before(gormCallbackName).Register(beforeName, c.beforeDelete)
db.Callback().Delete().After(gormCallbackName).Register(afterName, c.afterDelete)
case "row_query":
db.Callback().Row().Before(gormCallbackName).Register(beforeName, c.beforeRowQuery)
db.Callback().Row().After(gormCallbackName).Register(afterName, c.afterRowQuery)
}
}