2017-02-04 02:01:52 +00:00
|
|
|
package logging
|
|
|
|
|
|
|
|
import (
|
2018-01-11 18:19:01 +00:00
|
|
|
"fmt"
|
2017-03-19 04:59:01 +00:00
|
|
|
"os"
|
2017-02-04 02:01:52 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/op/go-logging"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2017-02-04 08:59:33 +00:00
|
|
|
size = 1024
|
2017-02-04 02:01:52 +00:00
|
|
|
)
|
|
|
|
|
2017-02-11 03:58:26 +00:00
|
|
|
var (
|
2017-02-12 05:10:40 +00:00
|
|
|
Log *CTopLogger
|
|
|
|
exited bool
|
2017-03-19 04:59:01 +00:00
|
|
|
level = logging.INFO // default level
|
2017-02-11 03:58:26 +00:00
|
|
|
format = logging.MustStringFormatter(
|
2017-02-16 23:32:16 +00:00
|
|
|
`%{color}%{time:15:04:05.000} ▶ %{level:.4s} %{id:03x}%{color:reset} %{message}`,
|
2017-02-11 03:58:26 +00:00
|
|
|
)
|
2017-02-04 02:01:52 +00:00
|
|
|
)
|
|
|
|
|
2018-01-11 18:19:01 +00:00
|
|
|
type statusMsg struct {
|
|
|
|
Text string
|
|
|
|
IsError bool
|
|
|
|
}
|
|
|
|
|
2017-02-04 02:01:52 +00:00
|
|
|
type CTopLogger struct {
|
|
|
|
*logging.Logger
|
|
|
|
backend *logging.MemoryBackend
|
2018-01-11 18:19:01 +00:00
|
|
|
sLog []statusMsg
|
2017-02-04 08:59:33 +00:00
|
|
|
}
|
|
|
|
|
2018-01-11 18:19:01 +00:00
|
|
|
func (c *CTopLogger) FlushStatus() chan statusMsg {
|
|
|
|
ch := make(chan statusMsg)
|
|
|
|
go func() {
|
|
|
|
for _, sm := range c.sLog {
|
|
|
|
ch <- sm
|
|
|
|
}
|
|
|
|
close(ch)
|
|
|
|
c.sLog = []statusMsg{}
|
|
|
|
}()
|
|
|
|
return ch
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *CTopLogger) StatusQueued() bool { return len(c.sLog) > 0 }
|
|
|
|
func (c *CTopLogger) Status(s string) { c.addStatus(statusMsg{s, false}) }
|
|
|
|
func (c *CTopLogger) StatusErr(err error) { c.addStatus(statusMsg{err.Error(), true}) }
|
|
|
|
func (c *CTopLogger) addStatus(sm statusMsg) { c.sLog = append(c.sLog, sm) }
|
|
|
|
|
|
|
|
func (c *CTopLogger) Statusf(s string, a ...interface{}) { c.Status(fmt.Sprintf(s, a...)) }
|
|
|
|
|
2017-02-12 05:14:50 +00:00
|
|
|
func Init() *CTopLogger {
|
2017-02-12 05:10:40 +00:00
|
|
|
if Log == nil {
|
2017-02-16 23:32:16 +00:00
|
|
|
logging.SetFormatter(format) // setup default formatter
|
|
|
|
|
2017-02-12 05:10:40 +00:00
|
|
|
Log = &CTopLogger{
|
|
|
|
logging.MustGetLogger("ctop"),
|
|
|
|
logging.NewMemoryBackend(size),
|
2018-01-11 18:19:01 +00:00
|
|
|
[]statusMsg{},
|
2017-02-12 05:10:40 +00:00
|
|
|
}
|
2017-02-04 08:59:33 +00:00
|
|
|
|
2020-11-19 09:21:27 +00:00
|
|
|
debugMode := debugMode()
|
|
|
|
if debugMode {
|
2017-03-19 04:59:01 +00:00
|
|
|
level = logging.DEBUG
|
|
|
|
}
|
2017-02-16 23:32:16 +00:00
|
|
|
backendLvl := logging.AddModuleLevel(Log.backend)
|
|
|
|
backendLvl.SetLevel(level, "")
|
|
|
|
|
|
|
|
logging.SetBackend(backendLvl)
|
2020-11-19 09:21:27 +00:00
|
|
|
|
|
|
|
if debugMode {
|
|
|
|
StartServer()
|
|
|
|
}
|
2017-02-12 05:10:40 +00:00
|
|
|
Log.Notice("logger initialized")
|
2017-02-04 08:59:33 +00:00
|
|
|
}
|
2017-02-12 05:10:40 +00:00
|
|
|
return Log
|
2017-02-04 08:59:33 +00:00
|
|
|
}
|
|
|
|
|
2017-02-04 02:01:52 +00:00
|
|
|
func (log *CTopLogger) tail() chan string {
|
|
|
|
stream := make(chan string)
|
|
|
|
|
|
|
|
node := log.backend.Head()
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
stream <- node.Record.Formatted(0)
|
|
|
|
for {
|
|
|
|
nnode := node.Next()
|
|
|
|
if nnode != nil {
|
|
|
|
node = nnode
|
|
|
|
break
|
|
|
|
}
|
2017-02-05 00:25:30 +00:00
|
|
|
if exited {
|
|
|
|
close(stream)
|
|
|
|
return
|
|
|
|
}
|
2017-02-04 02:01:52 +00:00
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return stream
|
|
|
|
}
|
2017-02-17 06:48:38 +00:00
|
|
|
|
|
|
|
func (log *CTopLogger) Exit() {
|
|
|
|
exited = true
|
|
|
|
StopServer()
|
|
|
|
}
|
2017-03-19 04:59:01 +00:00
|
|
|
|
2017-05-31 14:45:48 +00:00
|
|
|
func debugMode() bool { return os.Getenv("CTOP_DEBUG") == "1" }
|
|
|
|
func debugModeTCP() bool { return os.Getenv("CTOP_DEBUG_TCP") == "1" }
|