logger.go 2.6 KB

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