2015-09-18 02:49:06 -04:00
|
|
|
/*
|
2018-04-05 18:04:40 -04:00
|
|
|
* Minio Cloud Storage, (C) 2015, 2016, 2017, 2018 Minio, Inc.
|
2015-09-18 02:49:06 -04:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
package logger
|
2015-09-18 02:49:06 -04:00
|
|
|
|
|
|
|
import (
|
2018-04-05 18:04:40 -04:00
|
|
|
"context"
|
2018-01-17 10:24:46 -05:00
|
|
|
"encoding/json"
|
2016-10-11 03:50:27 -04:00
|
|
|
"fmt"
|
2018-03-30 22:13:25 -04:00
|
|
|
"go/build"
|
2018-01-17 10:24:46 -05:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2016-10-11 03:50:27 -04:00
|
|
|
"runtime"
|
2016-05-25 05:32:35 -04:00
|
|
|
"strings"
|
2018-01-17 10:24:46 -05:00
|
|
|
"time"
|
2015-09-18 02:49:06 -04:00
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
"github.com/fatih/color"
|
2017-03-23 19:36:00 -04:00
|
|
|
"github.com/minio/mc/pkg/console"
|
2015-09-18 02:49:06 -04:00
|
|
|
)
|
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
// global colors.
|
|
|
|
var (
|
|
|
|
colorBold = color.New(color.Bold).SprintFunc()
|
|
|
|
colorYellow = color.New(color.FgYellow).SprintfFunc()
|
|
|
|
colorRed = color.New(color.FgRed).SprintfFunc()
|
|
|
|
)
|
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
var trimStrings []string
|
2015-09-18 02:49:06 -04:00
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// Level type
|
|
|
|
type Level int8
|
2017-03-23 19:36:00 -04:00
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// Enumerated level types
|
|
|
|
const (
|
|
|
|
Error Level = iota + 1
|
|
|
|
Fatal
|
|
|
|
)
|
2017-03-23 19:36:00 -04:00
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
const loggerTimeFormat string = "15:04:05 MST 01/02/2006"
|
|
|
|
|
|
|
|
var matchingFuncNames = [...]string{
|
|
|
|
"http.HandlerFunc.ServeHTTP",
|
|
|
|
"cmd.serverMain",
|
|
|
|
"cmd.StartGateway",
|
|
|
|
// add more here ..
|
|
|
|
}
|
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
func (level Level) String() string {
|
|
|
|
var lvlStr string
|
|
|
|
switch level {
|
|
|
|
case Error:
|
|
|
|
lvlStr = "ERROR"
|
|
|
|
case Fatal:
|
|
|
|
lvlStr = "FATAL"
|
|
|
|
}
|
|
|
|
return lvlStr
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
type traceEntry struct {
|
|
|
|
Message string `json:"message"`
|
|
|
|
Source []string `json:"source"`
|
|
|
|
Variables map[string]string `json:"variables,omitempty"`
|
|
|
|
}
|
|
|
|
type args struct {
|
|
|
|
Bucket string `json:"bucket,omitempty"`
|
|
|
|
Object string `json:"object,omitempty"`
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
type api struct {
|
|
|
|
Name string `json:"name,omitempty"`
|
|
|
|
Args args `json:"args,omitempty"`
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
type logEntry struct {
|
|
|
|
Level string `json:"level"`
|
|
|
|
Time string `json:"time"`
|
|
|
|
API api `json:"api,omitempty"`
|
|
|
|
RemoteHost string `json:"remotehost,omitempty"`
|
|
|
|
RequestID string `json:"requestID,omitempty"`
|
|
|
|
UserAgent string `json:"userAgent,omitempty"`
|
|
|
|
Cause string `json:"cause,omitempty"`
|
|
|
|
Trace traceEntry `json:"error"`
|
2016-10-22 05:24:34 -04:00
|
|
|
}
|
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
// quiet: Hide startup messages if enabled
|
|
|
|
// jsonFlag: Display in JSON format, if enabled
|
|
|
|
var (
|
|
|
|
quiet, jsonFlag bool
|
|
|
|
)
|
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// EnableQuiet - turns quiet option on.
|
2018-04-05 18:04:40 -04:00
|
|
|
func EnableQuiet() {
|
|
|
|
quiet = true
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// EnableJSON - outputs logs in json format.
|
2018-04-05 18:04:40 -04:00
|
|
|
func EnableJSON() {
|
|
|
|
jsonFlag = true
|
|
|
|
quiet = true
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Println - wrapper to console.Println() with quiet flag.
|
2018-04-05 18:04:40 -04:00
|
|
|
func Println(args ...interface{}) {
|
|
|
|
if !quiet {
|
2017-03-23 19:36:00 -04:00
|
|
|
console.Println(args...)
|
2015-09-18 02:49:06 -04:00
|
|
|
}
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Printf - wrapper to console.Printf() with quiet flag.
|
2018-04-05 18:04:40 -04:00
|
|
|
func Printf(format string, args ...interface{}) {
|
|
|
|
if !quiet {
|
2017-03-23 19:36:00 -04:00
|
|
|
console.Printf(format, args...)
|
2015-09-18 02:49:06 -04:00
|
|
|
}
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
2016-08-25 12:39:01 -04:00
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
// Init sets the trimStrings to possible GOPATHs
|
|
|
|
// and GOROOT directories. Also append github.com/minio/minio
|
|
|
|
// This is done to clean up the filename, when stack trace is
|
|
|
|
// displayed when an error happens.
|
|
|
|
func Init(goPath string) {
|
2018-01-17 10:24:46 -05:00
|
|
|
var goPathList []string
|
2018-03-30 22:13:25 -04:00
|
|
|
var defaultgoPathList []string
|
2018-01-17 10:24:46 -05:00
|
|
|
// Add all possible GOPATH paths into trimStrings
|
|
|
|
// Split GOPATH depending on the OS type
|
|
|
|
if runtime.GOOS == "windows" {
|
2018-04-05 18:04:40 -04:00
|
|
|
goPathList = strings.Split(goPath, ";")
|
2018-03-30 22:13:25 -04:00
|
|
|
defaultgoPathList = strings.Split(build.Default.GOPATH, ";")
|
2018-01-17 10:24:46 -05:00
|
|
|
} else {
|
|
|
|
// All other types of OSs
|
2018-04-05 18:04:40 -04:00
|
|
|
goPathList = strings.Split(goPath, ":")
|
2018-03-30 22:13:25 -04:00
|
|
|
defaultgoPathList = strings.Split(build.Default.GOPATH, ":")
|
|
|
|
|
2016-11-23 14:35:04 -05:00
|
|
|
}
|
2017-03-23 19:36:00 -04:00
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// Add trim string "{GOROOT}/src/" into trimStrings
|
|
|
|
trimStrings = []string{filepath.Join(runtime.GOROOT(), "src") + string(filepath.Separator)}
|
2017-03-23 19:36:00 -04:00
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// Add all possible path from GOPATH=path1:path2...:pathN
|
|
|
|
// as "{path#}/src/" into trimStrings
|
|
|
|
for _, goPathString := range goPathList {
|
|
|
|
trimStrings = append(trimStrings, filepath.Join(goPathString, "src")+string(filepath.Separator))
|
|
|
|
}
|
2018-03-30 22:13:25 -04:00
|
|
|
|
|
|
|
for _, defaultgoPathString := range defaultgoPathList {
|
|
|
|
trimStrings = append(trimStrings, filepath.Join(defaultgoPathString, "src")+string(filepath.Separator))
|
|
|
|
}
|
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// Add "github.com/minio/minio" as the last to cover
|
|
|
|
// paths like "{GOROOT}/src/github.com/minio/minio"
|
|
|
|
// and "{GOPATH}/src/github.com/minio/minio"
|
|
|
|
trimStrings = append(trimStrings, filepath.Join("github.com", "minio", "minio")+string(filepath.Separator))
|
2015-09-19 05:36:50 -04:00
|
|
|
}
|
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
func trimTrace(f string) string {
|
|
|
|
for _, trimString := range trimStrings {
|
|
|
|
f = strings.TrimPrefix(filepath.ToSlash(f), filepath.ToSlash(trimString))
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
2018-01-17 10:24:46 -05:00
|
|
|
return filepath.FromSlash(f)
|
|
|
|
}
|
2017-03-23 19:36:00 -04:00
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
// getTrace method - creates and returns stack trace
|
|
|
|
func getTrace(traceLevel int) []string {
|
|
|
|
var trace []string
|
|
|
|
pc, file, lineNumber, ok := runtime.Caller(traceLevel)
|
|
|
|
|
|
|
|
for ok {
|
|
|
|
// Clean up the common prefixes
|
|
|
|
file = trimTrace(file)
|
|
|
|
// Get the function name
|
|
|
|
_, funcName := filepath.Split(runtime.FuncForPC(pc).Name())
|
|
|
|
// Skip duplicate traces that start with file name, "<autogenerated>"
|
|
|
|
// and also skip traces with function name that starts with "runtime."
|
|
|
|
if !strings.HasPrefix(file, "<autogenerated>") &&
|
|
|
|
!strings.HasPrefix(funcName, "runtime.") {
|
|
|
|
// Form and append a line of stack trace into a
|
|
|
|
// collection, 'trace', to build full stack trace
|
|
|
|
trace = append(trace, fmt.Sprintf("%v:%v:%v()", file, lineNumber, funcName))
|
2018-04-05 18:04:40 -04:00
|
|
|
|
|
|
|
// Ignore trace logs beyond the following conditions
|
|
|
|
for _, name := range matchingFuncNames {
|
|
|
|
if funcName == name {
|
|
|
|
return trace
|
|
|
|
}
|
|
|
|
}
|
2018-01-17 10:24:46 -05:00
|
|
|
}
|
|
|
|
traceLevel++
|
|
|
|
// Read stack trace information from PC
|
|
|
|
pc, file, lineNumber, ok = runtime.Caller(traceLevel)
|
|
|
|
}
|
|
|
|
return trace
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
func logIf(level Level, err error, msg string,
|
|
|
|
data ...interface{}) {
|
2018-04-05 18:04:40 -04:00
|
|
|
if err == nil {
|
2015-09-19 05:36:50 -04:00
|
|
|
return
|
|
|
|
}
|
2018-01-31 00:42:15 -05:00
|
|
|
// Get the cause for the Error
|
|
|
|
cause := err.Error()
|
2018-01-17 10:24:46 -05:00
|
|
|
// Get full stack trace
|
|
|
|
trace := getTrace(3)
|
|
|
|
// Get time
|
2018-04-05 18:04:40 -04:00
|
|
|
timeOfError := time.Now().UTC().Format(time.RFC3339Nano)
|
2018-01-17 10:24:46 -05:00
|
|
|
// Output the formatted log message at console
|
|
|
|
var output string
|
|
|
|
message := fmt.Sprintf(msg, data...)
|
2018-04-05 18:04:40 -04:00
|
|
|
if jsonFlag {
|
2018-01-17 10:24:46 -05:00
|
|
|
logJSON, err := json.Marshal(&logEntry{
|
2018-04-05 18:04:40 -04:00
|
|
|
Level: level.String(),
|
|
|
|
Time: timeOfError,
|
|
|
|
Cause: cause,
|
|
|
|
Trace: traceEntry{Source: trace, Message: message},
|
2018-01-17 10:24:46 -05:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
panic("json marshal of logEntry failed: " + err.Error())
|
|
|
|
}
|
|
|
|
output = string(logJSON)
|
|
|
|
} else {
|
|
|
|
// Add a sequence number and formatting for each stack trace
|
|
|
|
// No formatting is required for the first entry
|
|
|
|
trace[0] = "1: " + trace[0]
|
|
|
|
for i, element := range trace[1:] {
|
|
|
|
trace[i+1] = fmt.Sprintf("%8v: %s", i+2, element)
|
|
|
|
}
|
|
|
|
errMsg := fmt.Sprintf("[%s] [%s] %s (%s)",
|
|
|
|
timeOfError, level.String(), message, cause)
|
2017-03-23 19:36:00 -04:00
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
output = fmt.Sprintf("\nTrace: %s\n%s",
|
|
|
|
strings.Join(trace, "\n"),
|
|
|
|
colorRed(colorBold(errMsg)))
|
2015-09-19 05:36:50 -04:00
|
|
|
}
|
2018-01-17 10:24:46 -05:00
|
|
|
fmt.Println(output)
|
2017-02-21 04:32:05 -05:00
|
|
|
|
2018-01-17 10:24:46 -05:00
|
|
|
if level == Fatal {
|
|
|
|
os.Exit(1)
|
2016-11-23 14:35:04 -05:00
|
|
|
}
|
2015-09-18 02:49:06 -04:00
|
|
|
}
|
2016-11-23 19:36:26 -05:00
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
// FatalIf :
|
|
|
|
func FatalIf(err error, msg string, data ...interface{}) {
|
|
|
|
logIf(Fatal, err, msg, data...)
|
2017-03-23 19:36:00 -04:00
|
|
|
}
|
|
|
|
|
2018-04-05 18:04:40 -04:00
|
|
|
// LogIf :
|
|
|
|
func LogIf(ctx context.Context, err error) {
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
req := GetReqInfo(ctx)
|
|
|
|
|
|
|
|
if req == nil {
|
|
|
|
req = &ReqInfo{API: "SYSTEM"}
|
|
|
|
}
|
|
|
|
|
|
|
|
API := "SYSTEM"
|
|
|
|
if req.API != "" {
|
|
|
|
API = req.API
|
|
|
|
}
|
|
|
|
|
|
|
|
tags := make(map[string]string)
|
|
|
|
for _, entry := range req.GetTags() {
|
|
|
|
tags[entry.Key] = entry.Val
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the cause for the Error
|
|
|
|
message := err.Error()
|
|
|
|
// Get full stack trace
|
|
|
|
trace := getTrace(2)
|
|
|
|
// Output the formatted log message at console
|
|
|
|
var output string
|
|
|
|
if jsonFlag {
|
|
|
|
logJSON, err := json.Marshal(&logEntry{
|
|
|
|
Level: Error.String(),
|
|
|
|
RemoteHost: req.RemoteHost,
|
|
|
|
RequestID: req.RequestID,
|
|
|
|
UserAgent: req.UserAgent,
|
|
|
|
Time: time.Now().UTC().Format(time.RFC3339Nano),
|
|
|
|
API: api{Name: API, Args: args{Bucket: req.BucketName, Object: req.ObjectName}},
|
|
|
|
Trace: traceEntry{Message: message, Source: trace, Variables: tags},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
output = string(logJSON)
|
|
|
|
} else {
|
|
|
|
// Add a sequence number and formatting for each stack trace
|
|
|
|
// No formatting is required for the first entry
|
|
|
|
for i, element := range trace {
|
|
|
|
trace[i] = fmt.Sprintf("%8v: %s", i+1, element)
|
|
|
|
}
|
|
|
|
|
|
|
|
tagString := ""
|
|
|
|
for key, value := range tags {
|
|
|
|
if value != "" {
|
|
|
|
if tagString != "" {
|
|
|
|
tagString += ", "
|
|
|
|
}
|
|
|
|
tagString += key + "=" + value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
apiString := "API: " + API + "("
|
|
|
|
if req.BucketName != "" {
|
|
|
|
apiString = apiString + "bucket=" + req.BucketName
|
|
|
|
}
|
|
|
|
if req.ObjectName != "" {
|
|
|
|
apiString = apiString + ", object=" + req.ObjectName
|
|
|
|
}
|
|
|
|
apiString += ")"
|
|
|
|
timeString := "Time: " + time.Now().Format(loggerTimeFormat)
|
|
|
|
|
|
|
|
var requestID string
|
|
|
|
if req.RequestID != "" {
|
|
|
|
requestID = "\nRequestID: " + req.RequestID
|
|
|
|
}
|
|
|
|
|
|
|
|
var remoteHost string
|
|
|
|
if req.RemoteHost != "" {
|
|
|
|
remoteHost = "\nRemoteHost: " + req.RemoteHost
|
|
|
|
}
|
|
|
|
|
|
|
|
var userAgent string
|
|
|
|
if req.UserAgent != "" {
|
|
|
|
userAgent = "\nUserAgent: " + req.UserAgent
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(tags) > 0 {
|
|
|
|
tagString = "\n " + tagString
|
|
|
|
}
|
|
|
|
|
|
|
|
output = fmt.Sprintf("\n%s\n%s%s%s%s\nError: %s%s\n%s",
|
|
|
|
apiString, timeString, requestID, remoteHost, userAgent,
|
|
|
|
colorRed(colorBold(message)), tagString, strings.Join(trace, "\n"))
|
|
|
|
}
|
|
|
|
fmt.Println(output)
|
2016-11-23 19:36:26 -05:00
|
|
|
}
|