logger.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package common
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "github.com/gin-gonic/gin"
  7. "io"
  8. "log"
  9. "os"
  10. "path/filepath"
  11. "sync"
  12. "time"
  13. )
  14. const (
  15. loggerINFO = "INFO"
  16. loggerWarn = "WARN"
  17. loggerError = "ERR"
  18. )
  19. const maxLogCount = 1000000
  20. var logCount int
  21. var setupLogLock sync.Mutex
  22. var setupLogWorking bool
  23. func SetupLogger() {
  24. if *LogDir != "" {
  25. ok := setupLogLock.TryLock()
  26. if !ok {
  27. log.Println("setup log is already working")
  28. return
  29. }
  30. defer func() {
  31. setupLogLock.Unlock()
  32. setupLogWorking = false
  33. }()
  34. logPath := filepath.Join(*LogDir, fmt.Sprintf("oneapi-%s.log", time.Now().Format("20060102")))
  35. fd, err := os.OpenFile(logPath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
  36. if err != nil {
  37. log.Fatal("failed to open log file")
  38. }
  39. gin.DefaultWriter = io.MultiWriter(os.Stdout, fd)
  40. gin.DefaultErrorWriter = io.MultiWriter(os.Stderr, fd)
  41. }
  42. }
  43. func SysLog(s string) {
  44. t := time.Now()
  45. _, _ = fmt.Fprintf(gin.DefaultWriter, "[SYS] %v | %s \n", t.Format("2006/01/02 - 15:04:05"), s)
  46. }
  47. func SysError(s string) {
  48. t := time.Now()
  49. _, _ = fmt.Fprintf(gin.DefaultErrorWriter, "[SYS] %v | %s \n", t.Format("2006/01/02 - 15:04:05"), s)
  50. }
  51. func LogInfo(ctx context.Context, msg string) {
  52. logHelper(ctx, loggerINFO, msg)
  53. }
  54. func LogWarn(ctx context.Context, msg string) {
  55. logHelper(ctx, loggerWarn, msg)
  56. }
  57. func LogError(ctx context.Context, msg string) {
  58. logHelper(ctx, loggerError, msg)
  59. }
  60. func logHelper(ctx context.Context, level string, msg string) {
  61. writer := gin.DefaultErrorWriter
  62. if level == loggerINFO {
  63. writer = gin.DefaultWriter
  64. }
  65. id := ctx.Value(RequestIdKey)
  66. now := time.Now()
  67. _, _ = fmt.Fprintf(writer, "[%s] %v | %s | %s \n", level, now.Format("2006/01/02 - 15:04:05"), id, msg)
  68. logCount++ // we don't need accurate count, so no lock here
  69. if logCount > maxLogCount && !setupLogWorking {
  70. logCount = 0
  71. setupLogWorking = true
  72. go func() {
  73. SetupLogger()
  74. }()
  75. }
  76. }
  77. func FatalLog(v ...any) {
  78. t := time.Now()
  79. _, _ = fmt.Fprintf(gin.DefaultErrorWriter, "[FATAL] %v | %v \n", t.Format("2006/01/02 - 15:04:05"), v)
  80. os.Exit(1)
  81. }
  82. func LogQuota(quota int) string {
  83. if DisplayInCurrencyEnabled {
  84. return fmt.Sprintf("$%.6f 额度", float64(quota)/QuotaPerUnit)
  85. } else {
  86. return fmt.Sprintf("%d 点额度", quota)
  87. }
  88. }
  89. // LogJson 仅供测试使用 only for test
  90. func LogJson(ctx context.Context, msg string, obj any) {
  91. jsonStr, err := json.Marshal(obj)
  92. if err != nil {
  93. LogError(ctx, fmt.Sprintf("json marshal failed: %s", err.Error()))
  94. return
  95. }
  96. LogInfo(ctx, fmt.Sprintf("%s | %s", msg, string(jsonStr)))
  97. }