-
Notifications
You must be signed in to change notification settings - Fork 305
/
Copy pathnrzap.go
214 lines (189 loc) · 6.98 KB
/
nrzap.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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
package nrzap
import (
"errors"
"math"
"time"
"github.com/newrelic/go-agent/v3/internal"
"github.com/newrelic/go-agent/v3/newrelic"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
)
func init() { internal.TrackUsage("integration", "logcontext-v2", "zap") }
// NewRelicZapCore implements zap.Core
type NewRelicZapCore struct {
fields []zap.Field
core zapcore.Core
nr newrelicApplicationState
}
// newrelicApplicationState is a private struct that stores newrelic application data
// for automatic behind the scenes log collection logic.
type newrelicApplicationState struct {
app *newrelic.Application
txn *newrelic.Transaction
}
// Helper function that converts zap fields to a map of string interface
func convertFieldWithMapEncoder(fields []zap.Field) map[string]interface{} {
attributes := make(map[string]interface{})
for _, field := range fields {
enc := zapcore.NewMapObjectEncoder()
field.AddTo(enc)
for key, value := range enc.Fields {
// Format time.Duration values as strings
if durationVal, ok := value.(time.Duration); ok {
attributes[key] = durationVal.String()
} else {
attributes[key] = value
}
}
}
return attributes
}
func convertFieldsAtHarvestTime(fields []zap.Field) map[string]interface{} {
attributes := make(map[string]interface{})
for _, field := range fields {
if field.Interface != nil {
// Handles ErrorType fields
if field.Type == zapcore.ErrorType {
attributes[field.Key] = field.Interface.(error).Error()
} else {
// Handles all interface types
attributes[field.Key] = field.Interface
}
} else if field.String != "" { // Check if the field is a string and doesn't contain an interface
attributes[field.Key] = field.String
} else {
// Float Types
if field.Type == zapcore.Float32Type {
attributes[field.Key] = math.Float32frombits(uint32(field.Integer))
continue
} else if field.Type == zapcore.Float64Type {
attributes[field.Key] = math.Float64frombits(uint64(field.Integer))
continue
}
// Bool Type
if field.Type == zapcore.BoolType {
field.Interface = field.Integer == 1
attributes[field.Key] = field.Interface
} else {
// Integer Types
attributes[field.Key] = field.Integer
}
}
}
return attributes
}
// internal handler function to manage writing a log to the new relic application
func (nr *newrelicApplicationState) recordLog(entry zapcore.Entry, fields []zap.Field) {
attributes := map[string]interface{}{}
cfg, _ := nr.app.Config()
// Check if the attributes should be frontloaded or marshalled at harvest time
if cfg.ApplicationLogging.ZapLogger.AttributesFrontloaded {
attributes = convertFieldWithMapEncoder(fields)
} else {
attributes = convertFieldsAtHarvestTime(fields)
}
data := newrelic.LogData{
Timestamp: entry.Time.UnixMilli(),
Severity: entry.Level.String(),
Message: entry.Message,
Attributes: attributes,
}
if nr.txn != nil {
nr.txn.RecordLog(data)
} else if nr.app != nil {
nr.app.RecordLog(data)
}
}
var (
// ErrNilZapcore is an error caused by calling a WrapXCore function on a nil zapcore.Core object
ErrNilZapcore = errors.New("cannot wrap nil zapcore.Core object")
// ErrNilApp is an error caused by calling WrapBackgroundCore with a nil newrelic.Application
ErrNilApp = errors.New("wrapped a zapcore.Core with a nil New Relic application; logs will not be captured")
// ErrNilTxn is an error caused by calling WrapTransactionCore with a nil newrelic.Transaction
ErrNilTxn = errors.New("wrapped a zapcore.Core with a nil New Relic transaction; logs will not be captured")
)
// NewBackgroundCore creates a new NewRelicZapCore object, which is a wrapped zapcore.Core object. This wrapped object
// captures background logs in context and sends them to New Relic.
//
// Errors will be returned if the zapcore object is nil, or if the application is nil. It is up to the user to decide
// how to handle the case where the newrelic.Application is nil.
// In the case that the newrelic.Application is nil, a valid NewRelicZapCore object will still be returned.
//
// Please note that, while enriched context is added to log data forwarded to New Relic telemetry,
// it is not added to the local log data itself. This is due to the specific way zap logs are
// efficiently integrated into the New Relic agent logs in context. Local log decoration for zap
// logs requires additional custom code.
func WrapBackgroundCore(core zapcore.Core, app *newrelic.Application) (*NewRelicZapCore, error) {
if core == nil {
return nil, ErrNilZapcore
}
var err error
if app == nil {
err = ErrNilApp
}
return &NewRelicZapCore{
core: core,
nr: newrelicApplicationState{
app: app,
},
}, err
}
// WrapTransactionCore creates a new NewRelicZapCore object, which is a wrapped zapcore.Core object. This wrapped object
// captures logs in context of a transaction and sends them to New Relic.
//
// Errors will be returned if the zapcore object is nil, or if the application is nil. It is up to the user to decide
// how to handle the case where the newrelic.Transaction is nil.
// In the case that the newrelic.Application is nil, a valid NewRelicZapCore object will still be returned.
//
// Please note that, while enriched context is added to log data forwarded to New Relic telemetry,
// it is not added to the local log data itself. This is due to the specific way zap logs are
// efficiently integrated into the New Relic agent logs in context. Local log decoration for zap
// logs requires additional custom code.
func WrapTransactionCore(core zapcore.Core, txn *newrelic.Transaction) (zapcore.Core, error) {
if core == nil {
return nil, ErrNilZapcore
}
var err error
if txn == nil {
err = ErrNilTxn
}
return &NewRelicZapCore{
core: core,
nr: newrelicApplicationState{
txn: txn,
},
}, err
}
// With makes a copy of a NewRelicZapCore with new zap.Fields. It calls zapcore.With() on the zap core object
// then makes a deepcopy of the NewRelicApplicationState object so the original
// object can be deallocated when it's no longer in scope.
func (c *NewRelicZapCore) With(fields []zap.Field) zapcore.Core {
return &NewRelicZapCore{
core: c.core.With(fields),
fields: append(fields, c.fields...),
nr: newrelicApplicationState{
c.nr.app,
c.nr.txn,
},
}
}
// Check simply calls zapcore.Check on the Core object.
func (c *NewRelicZapCore) Check(entry zapcore.Entry, checkedEntry *zapcore.CheckedEntry) *zapcore.CheckedEntry {
ce := c.core.Check(entry, checkedEntry)
ce.AddCore(entry, c)
return ce
}
// Write wraps zapcore.Write and captures the log entry and sends that data to New Relic.
func (c *NewRelicZapCore) Write(entry zapcore.Entry, fields []zap.Field) error {
allFields := append(fields, c.fields...)
c.nr.recordLog(entry, allFields)
return nil
}
// Sync simply calls zapcore.Sync on the Core object.
func (c *NewRelicZapCore) Sync() error {
return c.core.Sync()
}
// Enabled simply calls zapcore.Enabled on the zapcore.Level passed to it.
func (c *NewRelicZapCore) Enabled(level zapcore.Level) bool {
return c.core.Enabled(level)
}