logger.go 2.5 KB

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