Files
deploy-felhom-compose/controller/internal/web/logbuffer.go
T
admin 4edc974404 fix: show actual timestamps in debug log viewer
The Naplóviewer was showing relative times like '3586mp' (seconds ago)
which were also negative due to timezone mismatch — parseLine used
time.Parse (UTC) but log.LstdFlags outputs local time. Now:
- parseLine uses time.ParseInLocation with time.Local
- fmtTime JS shows absolute HH:MM:SS (or MM-DD HH:MM:SS for old entries)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-23 16:02:28 +01:00

188 lines
4.4 KiB
Go

package web
import (
"strings"
"sync"
"time"
)
// LogEntry represents a single parsed log line.
type LogEntry struct {
Timestamp time.Time `json:"timestamp"`
Level string `json:"level"` // "DEBUG", "INFO", "WARN", "ERROR"
Message string `json:"message"`
Source string `json:"source"` // "file.go:123" if Lshortfile enabled
}
// LogBuffer is a thread-safe ring buffer that captures log output.
// It implements io.Writer so it can be used with log.New(io.MultiWriter(...)).
type LogBuffer struct {
mu sync.RWMutex
entries []LogEntry
size int
pos int
full bool
}
// NewLogBuffer creates a ring buffer that keeps the last `size` log entries.
func NewLogBuffer(size int) *LogBuffer {
return &LogBuffer{
entries: make([]LogEntry, size),
size: size,
}
}
// Write implements io.Writer. It parses Go's standard log output format.
// Handles two formats:
// - With Lshortfile: "2026/02/21 18:33:35 file.go:123: [LEVEL] message"
// - Without: "2026/02/21 18:33:35 [LEVEL] message"
func (lb *LogBuffer) Write(p []byte) (n int, err error) {
line := strings.TrimRight(string(p), "\n\r")
if line == "" {
return len(p), nil
}
entry := parseLine(line)
lb.mu.Lock()
lb.entries[lb.pos] = entry
lb.pos = (lb.pos + 1) % lb.size
if lb.pos == 0 && !lb.full {
lb.full = true
}
lb.mu.Unlock()
return len(p), nil
}
// Entries returns log entries filtered by minimum level, limited by count,
// and optionally filtered to entries after a given timestamp.
// Returns the matching entries and the total count in the buffer.
func (lb *LogBuffer) Entries(minLevel string, limit int, after time.Time) ([]LogEntry, int) {
lb.mu.RLock()
defer lb.mu.RUnlock()
// Collect all entries in chronological order
total := lb.size
if !lb.full {
total = lb.pos
}
if limit <= 0 || limit > 1000 {
limit = 200
}
levelOrder := levelPriority(minLevel)
var result []LogEntry
start := 0
if lb.full {
start = lb.pos
}
for i := 0; i < total; i++ {
idx := (start + i) % lb.size
e := lb.entries[idx]
// Filter by level
if levelPriority(e.Level) < levelOrder {
continue
}
// Filter by timestamp
if !after.IsZero() && !e.Timestamp.After(after) {
continue
}
result = append(result, e)
}
// Apply limit (keep the most recent entries)
if len(result) > limit {
result = result[len(result)-limit:]
}
return result, total
}
// parseLine parses a single log line into a LogEntry.
func parseLine(line string) LogEntry {
entry := LogEntry{
Level: "INFO",
Message: line,
}
// Try to parse timestamp: "2006/01/02 15:04:05"
// Use Local timezone because Go's log.LstdFlags outputs in local time.
if len(line) >= 19 {
if t, err := time.ParseInLocation("2006/01/02 15:04:05", line[:19], time.Local); err == nil {
entry.Timestamp = t
rest := line[19:]
if len(rest) > 0 && rest[0] == ' ' {
rest = rest[1:]
}
// Check for source file (Lshortfile): "file.go:123: [LEVEL] ..."
if colonIdx := strings.Index(rest, ": "); colonIdx > 0 && colonIdx < 40 {
candidate := rest[:colonIdx]
// Source file pattern: contains ".go:" or ".go" before the colon
if strings.Contains(candidate, ".go:") || strings.HasSuffix(candidate, ".go") {
entry.Source = candidate
rest = rest[colonIdx+2:]
}
}
// Extract level tag: [DEBUG], [INFO], [WARN], [ERROR], [SYNC], [SCHED], etc.
entry.Level, entry.Message = extractLevel(rest)
}
}
if entry.Timestamp.IsZero() {
entry.Timestamp = time.Now()
}
return entry
}
// extractLevel finds and removes a [LEVEL] tag from the beginning of a string.
func extractLevel(s string) (string, string) {
s = strings.TrimSpace(s)
if len(s) < 3 || s[0] != '[' {
return "INFO", s
}
end := strings.Index(s, "]")
if end < 0 || end > 20 {
return "INFO", s
}
tag := s[1:end]
msg := strings.TrimSpace(s[end+1:])
switch tag {
case "DEBUG":
return "DEBUG", msg
case "INFO":
return "INFO", msg
case "WARN":
return "WARN", msg
case "ERROR":
return "ERROR", msg
case "FATAL":
return "ERROR", msg
default:
// Tags like [SYNC], [SCHED], [STORAGE] etc. — treat as INFO, keep tag in message
return "INFO", s
}
}
// levelPriority returns numeric priority for log levels.
func levelPriority(level string) int {
switch strings.ToUpper(level) {
case "DEBUG":
return 0
case "INFO":
return 1
case "WARN":
return 2
case "ERROR":
return 3
default:
return 0
}
}