195 lines
5.3 KiB
Go
195 lines
5.3 KiB
Go
|
// Copyright (c) 2012 - Cloud Instruments Co., Ltd.
|
||
|
//
|
||
|
// All rights reserved.
|
||
|
//
|
||
|
// Redistribution and use in source and binary forms, with or without
|
||
|
// modification, are permitted provided that the following conditions are met:
|
||
|
//
|
||
|
// 1. Redistributions of source code must retain the above copyright notice, this
|
||
|
// list of conditions and the following disclaimer.
|
||
|
// 2. Redistributions in binary form must reproduce the above copyright notice,
|
||
|
// this list of conditions and the following disclaimer in the documentation
|
||
|
// and/or other materials provided with the distribution.
|
||
|
//
|
||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||
|
// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||
|
// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
||
|
// ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||
|
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||
|
// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||
|
// ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||
|
// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
|
||
|
package seelog
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"runtime"
|
||
|
"strings"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
var workingDir = "/"
|
||
|
|
||
|
func init() {
|
||
|
wd, err := os.Getwd()
|
||
|
if err == nil {
|
||
|
workingDir = filepath.ToSlash(wd) + "/"
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Represents runtime caller context.
|
||
|
type LogContextInterface interface {
|
||
|
// Caller's function name.
|
||
|
Func() string
|
||
|
// Caller's line number.
|
||
|
Line() int
|
||
|
// Caller's file short path (in slashed form).
|
||
|
ShortPath() string
|
||
|
// Caller's file full path (in slashed form).
|
||
|
FullPath() string
|
||
|
// Caller's file name (without path).
|
||
|
FileName() string
|
||
|
// True if the context is correct and may be used.
|
||
|
// If false, then an error in context evaluation occurred and
|
||
|
// all its other data may be corrupted.
|
||
|
IsValid() bool
|
||
|
// Time when log function was called.
|
||
|
CallTime() time.Time
|
||
|
// Custom context that can be set by calling logger.SetContext
|
||
|
CustomContext() interface{}
|
||
|
}
|
||
|
|
||
|
// Returns context of the caller
|
||
|
func currentContext(custom interface{}) (LogContextInterface, error) {
|
||
|
return specifyContext(1, custom)
|
||
|
}
|
||
|
|
||
|
func extractCallerInfo(skip int) (fullPath string, shortPath string, funcName string, line int, err error) {
|
||
|
pc, fp, ln, ok := runtime.Caller(skip)
|
||
|
if !ok {
|
||
|
err = fmt.Errorf("error during runtime.Caller")
|
||
|
return
|
||
|
}
|
||
|
line = ln
|
||
|
fullPath = fp
|
||
|
if strings.HasPrefix(fp, workingDir) {
|
||
|
shortPath = fp[len(workingDir):]
|
||
|
} else {
|
||
|
shortPath = fp
|
||
|
}
|
||
|
funcName = runtime.FuncForPC(pc).Name()
|
||
|
if strings.HasPrefix(funcName, workingDir) {
|
||
|
funcName = funcName[len(workingDir):]
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Returns context of the function with placed "skip" stack frames of the caller
|
||
|
// If skip == 0 then behaves like currentContext
|
||
|
// Context is returned in any situation, even if error occurs. But, if an error
|
||
|
// occurs, the returned context is an error context, which contains no paths
|
||
|
// or names, but states that they can't be extracted.
|
||
|
func specifyContext(skip int, custom interface{}) (LogContextInterface, error) {
|
||
|
callTime := time.Now()
|
||
|
if skip < 0 {
|
||
|
err := fmt.Errorf("can not skip negative stack frames")
|
||
|
return &errorContext{callTime, err}, err
|
||
|
}
|
||
|
fullPath, shortPath, funcName, line, err := extractCallerInfo(skip + 2)
|
||
|
if err != nil {
|
||
|
return &errorContext{callTime, err}, err
|
||
|
}
|
||
|
_, fileName := filepath.Split(fullPath)
|
||
|
return &logContext{funcName, line, shortPath, fullPath, fileName, callTime, custom}, nil
|
||
|
}
|
||
|
|
||
|
// Represents a normal runtime caller context.
|
||
|
type logContext struct {
|
||
|
funcName string
|
||
|
line int
|
||
|
shortPath string
|
||
|
fullPath string
|
||
|
fileName string
|
||
|
callTime time.Time
|
||
|
custom interface{}
|
||
|
}
|
||
|
|
||
|
func (context *logContext) IsValid() bool {
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
func (context *logContext) Func() string {
|
||
|
return context.funcName
|
||
|
}
|
||
|
|
||
|
func (context *logContext) Line() int {
|
||
|
return context.line
|
||
|
}
|
||
|
|
||
|
func (context *logContext) ShortPath() string {
|
||
|
return context.shortPath
|
||
|
}
|
||
|
|
||
|
func (context *logContext) FullPath() string {
|
||
|
return context.fullPath
|
||
|
}
|
||
|
|
||
|
func (context *logContext) FileName() string {
|
||
|
return context.fileName
|
||
|
}
|
||
|
|
||
|
func (context *logContext) CallTime() time.Time {
|
||
|
return context.callTime
|
||
|
}
|
||
|
|
||
|
func (context *logContext) CustomContext() interface{} {
|
||
|
return context.custom
|
||
|
}
|
||
|
|
||
|
// Represents an error context
|
||
|
type errorContext struct {
|
||
|
errorTime time.Time
|
||
|
err error
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) getErrorText(prefix string) string {
|
||
|
return fmt.Sprintf("%s() error: %s", prefix, errContext.err)
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) IsValid() bool {
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) Line() int {
|
||
|
return -1
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) Func() string {
|
||
|
return errContext.getErrorText("Func")
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) ShortPath() string {
|
||
|
return errContext.getErrorText("ShortPath")
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) FullPath() string {
|
||
|
return errContext.getErrorText("FullPath")
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) FileName() string {
|
||
|
return errContext.getErrorText("FileName")
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) CallTime() time.Time {
|
||
|
return errContext.errorTime
|
||
|
}
|
||
|
|
||
|
func (errContext *errorContext) CustomContext() interface{} {
|
||
|
return nil
|
||
|
}
|