logger.go 2.8 KB

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