forked from goadesign/goa
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlogging.go
123 lines (110 loc) · 3.39 KB
/
logging.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
package goa
import (
"bytes"
"fmt"
"log"
"context"
)
// ErrMissingLogValue is the value used to log keys with missing values
const ErrMissingLogValue = "MISSING"
type (
// LogAdapter is the logger interface used by goa to log informational and error messages.
// Adapters to different logging backends are provided in the logging sub-packages.
// goa takes care of initializing the logging context with the service, controller and
// action names.
LogAdapter interface {
// Info logs an informational message.
Info(msg string, keyvals ...interface{})
// Error logs an error.
Error(msg string, keyvals ...interface{})
// New appends to the logger context and returns the updated logger logger.
New(keyvals ...interface{}) LogAdapter
}
// adapter is the stdlib logger adapter.
adapter struct {
*log.Logger
keyvals []interface{}
}
)
// NewLogger returns a goa log adpater backed by a log logger.
func NewLogger(logger *log.Logger) LogAdapter {
return &adapter{Logger: logger}
}
// Logger returns the logger stored in the context if any, nil otherwise.
func Logger(ctx context.Context) *log.Logger {
logger := ContextLogger(ctx)
if a, ok := logger.(*adapter); ok {
return a.Logger
}
return nil
}
func (a *adapter) Info(msg string, keyvals ...interface{}) {
a.logit(msg, keyvals, false)
}
func (a *adapter) Error(msg string, keyvals ...interface{}) {
a.logit(msg, keyvals, true)
}
func (a *adapter) New(keyvals ...interface{}) LogAdapter {
if len(keyvals) == 0 {
return a
}
kvs := append(a.keyvals, keyvals...)
if len(kvs)%2 != 0 {
kvs = append(kvs, ErrMissingLogValue)
}
return &adapter{
Logger: a.Logger,
// Limiting the capacity of the stored keyvals ensures that a new
// backing array is created if the slice must grow.
keyvals: kvs[:len(kvs):len(kvs)],
}
}
func (a *adapter) logit(msg string, keyvals []interface{}, iserror bool) {
n := (len(keyvals) + 1) / 2
if len(keyvals)%2 != 0 {
keyvals = append(keyvals, ErrMissingLogValue)
}
m := (len(a.keyvals) + 1) / 2
n += m
var fm bytes.Buffer
lvl := "INFO"
if iserror {
lvl = "EROR" // Not a typo. It ensures all level strings are 4-chars long.
}
fm.WriteString(fmt.Sprintf("[%s] %s", lvl, msg))
vals := make([]interface{}, n)
offset := len(a.keyvals)
for i := 0; i < offset; i += 2 {
k := a.keyvals[i]
v := a.keyvals[i+1]
vals[i/2] = v
fm.WriteString(fmt.Sprintf(" %s=%%+v", k))
}
for i := 0; i < len(keyvals); i += 2 {
k := keyvals[i]
v := keyvals[i+1]
vals[i/2+offset/2] = v
fm.WriteString(fmt.Sprintf(" %s=%%+v", k))
}
a.Logger.Printf(fm.String(), vals...)
}
// LogInfo extracts the logger from the given context and calls Info on it.
// This is intended for code that needs portable logging such as the internal code of goa and
// middleware. User code should use the log adapters instead.
func LogInfo(ctx context.Context, msg string, keyvals ...interface{}) {
if l := ctx.Value(logKey); l != nil {
if logger, ok := l.(LogAdapter); ok {
logger.Info(msg, keyvals...)
}
}
}
// LogError extracts the logger from the given context and calls Error on it.
// This is intended for code that needs portable logging such as the internal code of goa and
// middleware. User code should use the log adapters instead.
func LogError(ctx context.Context, msg string, keyvals ...interface{}) {
if l := ctx.Value(logKey); l != nil {
if logger, ok := l.(LogAdapter); ok {
logger.Error(msg, keyvals...)
}
}
}