mirror of
https://github.com/Control-D-Inc/ctrld.git
synced 2026-02-03 22:18:39 +00:00
So in case of no logging enabled, useful data could be sent to ControlD server for further troubleshooting.
121 lines
2.6 KiB
Go
121 lines
2.6 KiB
Go
package cli
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"os"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/rs/zerolog"
|
|
|
|
"github.com/Control-D-Inc/ctrld"
|
|
)
|
|
|
|
const (
|
|
logWriterSize = 1024 * 1024 * 5 // 5 MB
|
|
logWriterInitialSize = 32 // 32 B
|
|
logSentInterval = time.Minute
|
|
logTruncatedMarker = "...\n"
|
|
)
|
|
|
|
type logViewResponse struct {
|
|
Data string `json:"data"`
|
|
}
|
|
|
|
// logWriter is an internal buffer to keep track of runtime log when no logging is enabled.
|
|
type logWriter struct {
|
|
mu sync.Mutex
|
|
buf bytes.Buffer
|
|
size int
|
|
}
|
|
|
|
// newLogWriter creates an internal log writer with a fixed buffer size.
|
|
func newLogWriter() *logWriter {
|
|
lw := &logWriter{size: logWriterSize}
|
|
return lw
|
|
}
|
|
|
|
func (lw *logWriter) Write(p []byte) (int, error) {
|
|
lw.mu.Lock()
|
|
defer lw.mu.Unlock()
|
|
|
|
// If writing p causes overflows, discard old data.
|
|
if lw.buf.Len()+len(p) > lw.size {
|
|
buf := lw.buf.Bytes()
|
|
buf = buf[:logWriterInitialSize]
|
|
if idx := bytes.LastIndex(buf, []byte("\n")); idx != -1 {
|
|
buf = buf[:idx]
|
|
}
|
|
lw.buf.Reset()
|
|
lw.buf.Write(buf)
|
|
lw.buf.WriteString(logTruncatedMarker) // indicate that the log was truncated.
|
|
}
|
|
// If p is bigger than buffer size, truncate p by half until its size is smaller.
|
|
for len(p)+lw.buf.Len() > lw.size {
|
|
p = p[len(p)/2:]
|
|
}
|
|
return lw.buf.Write(p)
|
|
}
|
|
|
|
// initInternalLogging performs internal logging if there's no log enabled.
|
|
func (p *prog) initInternalLogging() {
|
|
if !p.needInternalLogging() {
|
|
return
|
|
}
|
|
p.initInternalLogWriterOnce.Do(func() {
|
|
mainLog.Load().Notice().Msg("internal logging enabled")
|
|
lw := newLogWriter()
|
|
p.internalLogWriter = lw
|
|
p.internalLogSent = time.Now().Add(-logSentInterval)
|
|
})
|
|
p.mu.Lock()
|
|
lw := p.internalLogWriter
|
|
p.mu.Unlock()
|
|
multi := zerolog.MultiLevelWriter(lw)
|
|
l := mainLog.Load().Output(multi).With().Logger()
|
|
mainLog.Store(&l)
|
|
ctrld.ProxyLogger.Store(&l)
|
|
if verbose == 0 {
|
|
zerolog.SetGlobalLevel(zerolog.DebugLevel)
|
|
}
|
|
}
|
|
|
|
// needInternalLogging reports whether prog needs to run internal logging.
|
|
func (p *prog) needInternalLogging() bool {
|
|
// Do not run in non-cd mode.
|
|
if cdUID == "" {
|
|
return false
|
|
}
|
|
// Do not run if there's already log file.
|
|
if p.cfg.Service.LogPath != "" {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (p *prog) logContent() ([]byte, error) {
|
|
var data []byte
|
|
if p.needInternalLogging() {
|
|
p.mu.Lock()
|
|
lw := p.internalLogWriter
|
|
p.mu.Unlock()
|
|
if lw == nil {
|
|
return nil, errors.New("nil internal log writer")
|
|
}
|
|
lw.mu.Lock()
|
|
data = lw.buf.Bytes()
|
|
lw.mu.Unlock()
|
|
} else {
|
|
if p.cfg.Service.LogPath == "" {
|
|
return nil, nil
|
|
}
|
|
buf, err := os.ReadFile(p.cfg.Service.LogPath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
data = buf
|
|
}
|
|
return data, nil
|
|
}
|