Files
ez-api/internal/service/logger.go
zenfun 5891722526 feat(log): replace standard logger with zerolog
- Integrate `rs/zerolog` for structured and leveled logging
- Add `EZ_LOG_LEVEL` environment variable support (default: info)
- Configure console output with timestamps and service fields
- Migrate `main.go` and `LogWriter` to use the new logger instance
- Update README with logging configuration and design details
2025-12-12 22:39:10 +08:00

80 lines
1.6 KiB
Go

package service
import (
"context"
"time"
"github.com/ez-api/ez-api/internal/model"
"github.com/rs/zerolog/log"
"gorm.io/gorm"
)
// LogWriter batches log records to reduce IO overhead.
type LogWriter struct {
ch chan model.LogRecord
batchSize int
flushInterval time.Duration
db *gorm.DB
}
func NewLogWriter(db *gorm.DB, queueCapacity, batchSize int, flushInterval time.Duration) *LogWriter {
if batchSize <= 0 {
batchSize = 10
}
if queueCapacity <= 0 {
queueCapacity = 1000
}
if flushInterval <= 0 {
flushInterval = time.Second
}
return &LogWriter{
ch: make(chan model.LogRecord, queueCapacity),
batchSize: batchSize,
flushInterval: flushInterval,
db: db,
}
}
// Start begins a background writer. Should be called once at startup.
func (w *LogWriter) Start(ctx context.Context) {
go func() {
ticker := time.NewTicker(w.flushInterval)
defer ticker.Stop()
buf := make([]model.LogRecord, 0, w.batchSize)
flush := func() {
if len(buf) == 0 {
return
}
if err := w.db.Create(&buf).Error; err != nil {
log.Error().Err(err).Msg("log batch insert failed")
}
buf = buf[:0]
}
for {
select {
case <-ctx.Done():
flush()
return
case rec := <-w.ch:
buf = append(buf, rec)
if len(buf) >= w.batchSize {
flush()
}
case <-ticker.C:
flush()
}
}
}()
}
// Write queues a log record; drops silently if buffer is full to protect performance.
func (w *LogWriter) Write(rec model.LogRecord) {
select {
case w.ch <- rec:
default:
// drop to avoid blocking hot path
}
}