-
Notifications
You must be signed in to change notification settings - Fork 225
/
log.go
121 lines (103 loc) · 2.79 KB
/
log.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
// (c) 2019-2020, Ava Labs, Inc. All rights reserved.
// See the file LICENSE for licensing terms.
package evm
import (
"encoding/json"
"fmt"
"io"
"reflect"
"time"
"github.com/ethereum/go-ethereum/log"
)
const (
errorKey = "LOG15_ERROR"
timeFormat = "2006-01-02T15:04:05-0700"
)
type SubnetEVMLogger struct {
log.Handler
}
// InitLogger initializes logger with alias and sets the log level and format with the original [os.StdErr] interface
// along with the context logger.
func InitLogger(alias string, level string, jsonFormat bool, writer io.Writer) (SubnetEVMLogger, error) {
logFormat := SubnetEVMTermFormat(alias)
if jsonFormat {
logFormat = SubnetEVMJSONFormat(alias)
}
// Create handler
logHandler := log.StreamHandler(writer, logFormat)
c := SubnetEVMLogger{Handler: logHandler}
if err := c.SetLogLevel(level); err != nil {
return SubnetEVMLogger{}, err
}
return c, nil
}
// SetLogLevel sets the log level of initialized log handler.
func (c *SubnetEVMLogger) SetLogLevel(level string) error {
// Set log level
logLevel, err := log.LvlFromString(level)
if err != nil {
return err
}
log.Root().SetHandler(log.LvlFilterHandler(logLevel, c))
return nil
}
func SubnetEVMTermFormat(alias string) log.Format {
prefix := fmt.Sprintf("<%s Chain>", alias)
return log.FormatFunc(func(r *log.Record) []byte {
location := fmt.Sprintf("%+v", r.Call)
newMsg := fmt.Sprintf("%s %s: %s", prefix, location, r.Msg)
r.Msg = newMsg
return log.TerminalFormat(false).Format(r)
})
}
func SubnetEVMJSONFormat(alias string) log.Format {
prefix := fmt.Sprintf("%s Chain", alias)
return log.FormatFunc(func(r *log.Record) []byte {
props := make(map[string]interface{}, 5+len(r.Ctx)/2)
props["timestamp"] = r.Time
props["level"] = r.Lvl.String()
props[r.KeyNames.Msg] = r.Msg
props["logger"] = prefix
props["caller"] = fmt.Sprintf("%+v", r.Call)
for i := 0; i < len(r.Ctx); i += 2 {
k, ok := r.Ctx[i].(string)
if !ok {
props[errorKey] = fmt.Sprintf("%+v is not a string key", r.Ctx[i])
} else {
// The number of arguments is normalized from the geth logger
// to ensure that this will not cause an index out of bounds error
props[k] = formatJSONValue(r.Ctx[i+1])
}
}
b, err := json.Marshal(props)
if err != nil {
b, _ = json.Marshal(map[string]string{
errorKey: err.Error(),
})
return b
}
b = append(b, '\n')
return b
})
}
func formatJSONValue(value interface{}) (result interface{}) {
defer func() {
if err := recover(); err != nil {
if v := reflect.ValueOf(value); v.Kind() == reflect.Ptr && v.IsNil() {
result = "nil"
} else {
panic(err)
}
}
}()
switch v := value.(type) {
case time.Time:
return v.Format(timeFormat)
case error:
return v.Error()
case fmt.Stringer:
return v.String()
default:
return v
}
}