mirror of
https://github.com/bitechdev/ResolveSpec.git
synced 2025-12-29 07:44:25 +00:00
203 lines
4.7 KiB
Go
203 lines
4.7 KiB
Go
package logger
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"runtime/debug"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
errortracking "github.com/bitechdev/ResolveSpec/pkg/errortracking"
|
|
)
|
|
|
|
var Logger *zap.SugaredLogger
|
|
var errorTracker errortracking.Provider
|
|
|
|
func Init(dev bool) {
|
|
|
|
if dev {
|
|
cfg := zap.NewDevelopmentConfig()
|
|
UpdateLogger(&cfg)
|
|
} else {
|
|
cfg := zap.NewProductionConfig()
|
|
UpdateLogger(&cfg)
|
|
}
|
|
|
|
}
|
|
|
|
func UpdateLoggerPath(path string, dev bool) {
|
|
defaultConfig := zap.NewProductionConfig()
|
|
if dev {
|
|
defaultConfig = zap.NewDevelopmentConfig()
|
|
}
|
|
defaultConfig.OutputPaths = []string{path}
|
|
UpdateLogger(&defaultConfig)
|
|
}
|
|
|
|
func UpdateLogger(config *zap.Config) {
|
|
defaultConfig := zap.NewProductionConfig()
|
|
defaultConfig.OutputPaths = []string{"resolvespec.log"}
|
|
if config == nil {
|
|
config = &defaultConfig
|
|
}
|
|
|
|
logger, err := config.Build()
|
|
if err != nil {
|
|
log.Print(err)
|
|
return
|
|
}
|
|
|
|
Logger = logger.Sugar()
|
|
Info("ResolveSpec Logger initialized")
|
|
}
|
|
|
|
// InitErrorTracking initializes the error tracking provider
|
|
func InitErrorTracking(provider errortracking.Provider) {
|
|
errorTracker = provider
|
|
if errorTracker != nil {
|
|
Info("Error tracking initialized")
|
|
}
|
|
}
|
|
|
|
// GetErrorTracker returns the current error tracking provider
|
|
func GetErrorTracker() errortracking.Provider {
|
|
return errorTracker
|
|
}
|
|
|
|
// CloseErrorTracking flushes and closes the error tracking provider
|
|
func CloseErrorTracking() error {
|
|
if errorTracker != nil {
|
|
errorTracker.Flush(5)
|
|
return errorTracker.Close()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// extractContext attempts to find a context.Context in the given arguments.
|
|
// It returns the found context (or context.Background() if not found) and
|
|
// the remaining arguments without the context.
|
|
func extractContext(args ...interface{}) (context.Context, []interface{}) {
|
|
ctx := context.Background()
|
|
var newArgs []interface{}
|
|
found := false
|
|
|
|
for _, arg := range args {
|
|
if c, ok := arg.(context.Context); ok && !found {
|
|
ctx = c
|
|
found = true
|
|
} else {
|
|
newArgs = append(newArgs, arg)
|
|
}
|
|
}
|
|
return ctx, newArgs
|
|
}
|
|
|
|
func Info(template string, args ...interface{}) {
|
|
if Logger == nil {
|
|
log.Printf(template, args...)
|
|
return
|
|
}
|
|
Logger.Infow(fmt.Sprintf(template, args...), "process_id", os.Getpid())
|
|
}
|
|
|
|
func Warn(template string, args ...interface{}) {
|
|
ctx, remainingArgs := extractContext(args...)
|
|
message := fmt.Sprintf(template, remainingArgs...)
|
|
if Logger == nil {
|
|
log.Printf("%s", message)
|
|
} else {
|
|
Logger.Warnw(message, "process_id", os.Getpid())
|
|
}
|
|
|
|
// Send to error tracker
|
|
if errorTracker != nil {
|
|
errorTracker.CaptureMessage(ctx, message, errortracking.SeverityWarning, map[string]interface{}{
|
|
"process_id": os.Getpid(),
|
|
})
|
|
}
|
|
}
|
|
|
|
func Error(template string, args ...interface{}) {
|
|
ctx, remainingArgs := extractContext(args...)
|
|
message := fmt.Sprintf(template, remainingArgs...)
|
|
if Logger == nil {
|
|
log.Printf("%s", message)
|
|
} else {
|
|
Logger.Errorw(message, "process_id", os.Getpid())
|
|
}
|
|
|
|
// Send to error tracker
|
|
if errorTracker != nil {
|
|
errorTracker.CaptureMessage(ctx, message, errortracking.SeverityError, map[string]interface{}{
|
|
"process_id": os.Getpid(),
|
|
})
|
|
}
|
|
}
|
|
|
|
func Debug(template string, args ...interface{}) {
|
|
if Logger == nil {
|
|
log.Printf(template, args...)
|
|
return
|
|
}
|
|
Logger.Debugw(fmt.Sprintf(template, args...), "process_id", os.Getpid())
|
|
}
|
|
|
|
// CatchPanic - Handle panic
|
|
func CatchPanicCallback(location string, cb func(err any), args ...interface{}) {
|
|
ctx, _ := extractContext(args...)
|
|
if err := recover(); err != nil {
|
|
callstack := debug.Stack()
|
|
|
|
if Logger != nil {
|
|
Error("Panic in %s : %v", location, err, ctx) // Pass context implicitly
|
|
} else {
|
|
fmt.Printf("%s:PANIC->%+v", location, err)
|
|
debug.PrintStack()
|
|
}
|
|
|
|
// Send to error tracker
|
|
if errorTracker != nil {
|
|
errorTracker.CapturePanic(ctx, err, callstack, map[string]interface{}{
|
|
"location": location,
|
|
"process_id": os.Getpid(),
|
|
})
|
|
}
|
|
|
|
if cb != nil {
|
|
cb(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
// CatchPanic - Handle panic
|
|
func CatchPanic(location string, args ...interface{}) {
|
|
CatchPanicCallback(location, nil, args...)
|
|
}
|
|
|
|
// HandlePanic logs a panic and returns it as an error
|
|
// This should be called with the result of recover() from a deferred function
|
|
// Example usage:
|
|
//
|
|
// defer func() {
|
|
// if r := recover(); r != nil {
|
|
// err = logger.HandlePanic("MethodName", r)
|
|
// }
|
|
// }()
|
|
func HandlePanic(methodName string, r any, args ...interface{}) error {
|
|
ctx, _ := extractContext(args...)
|
|
stack := debug.Stack()
|
|
Error("Panic in %s: %v\nStack trace:\n%s", methodName, r, string(stack), ctx) // Pass context implicitly
|
|
|
|
// Send to error tracker
|
|
if errorTracker != nil {
|
|
errorTracker.CapturePanic(ctx, r, stack, map[string]interface{}{
|
|
"method": methodName,
|
|
"process_id": os.Getpid(),
|
|
})
|
|
}
|
|
|
|
return fmt.Errorf("panic in %s: %v", methodName, r)
|
|
}
|