2017-02-04 02:01:52 +00:00
|
|
|
package logging
|
|
|
|
|
|
|
|
import (
|
2017-02-04 08:59:33 +00:00
|
|
|
"fmt"
|
2017-02-04 02:01:52 +00:00
|
|
|
"net"
|
2017-02-05 00:25:30 +00:00
|
|
|
"sync"
|
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
|
|
|
path = "/tmp/ctop.sock"
|
|
|
|
)
|
|
|
|
|
2017-02-11 03:58:26 +00:00
|
|
|
var (
|
2017-02-12 05:10:40 +00:00
|
|
|
Log *CTopLogger
|
2017-02-11 03:58:26 +00:00
|
|
|
wg sync.WaitGroup
|
2017-02-12 05:10:40 +00:00
|
|
|
exited bool
|
2017-02-11 03:58:26 +00:00
|
|
|
format = logging.MustStringFormatter(
|
|
|
|
`%{color}%{time:15:04:05.000} %{shortfunc} ▶ %{level:.4s} %{id:03x}%{color:reset} %{message}`,
|
|
|
|
)
|
2017-02-04 02:01:52 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type CTopLogger struct {
|
|
|
|
*logging.Logger
|
|
|
|
backend *logging.MemoryBackend
|
2017-02-04 08:59:33 +00:00
|
|
|
}
|
|
|
|
|
2017-02-12 05:14:50 +00:00
|
|
|
func Init() *CTopLogger {
|
2017-02-12 05:10:40 +00:00
|
|
|
if Log == nil {
|
|
|
|
Log = &CTopLogger{
|
|
|
|
logging.MustGetLogger("ctop"),
|
|
|
|
logging.NewMemoryBackend(size),
|
|
|
|
}
|
2017-02-04 08:59:33 +00:00
|
|
|
|
2017-02-12 05:10:40 +00:00
|
|
|
backendFmt := logging.NewBackendFormatter(Log.backend, format)
|
|
|
|
logging.SetBackend(backendFmt)
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
func (log *CTopLogger) Exit() {
|
2017-02-05 00:25:30 +00:00
|
|
|
exited = true
|
|
|
|
wg.Wait()
|
2017-02-04 02:01:52 +00:00
|
|
|
}
|
|
|
|
|
2017-02-12 05:14:50 +00:00
|
|
|
func (log *CTopLogger) StartServer() {
|
2017-02-04 02:01:52 +00:00
|
|
|
ln, err := net.Listen("unix", path)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2017-02-04 08:59:33 +00:00
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
conn, err := ln.Accept()
|
|
|
|
if err != nil {
|
2017-02-05 00:25:30 +00:00
|
|
|
if err, ok := err.(net.Error); ok && err.Temporary() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return
|
2017-02-04 08:59:33 +00:00
|
|
|
}
|
|
|
|
go log.handler(conn)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2017-02-05 00:25:30 +00:00
|
|
|
log.Notice("logging server started")
|
2017-02-04 02:01:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (log *CTopLogger) handler(conn net.Conn) {
|
2017-02-05 00:25:30 +00:00
|
|
|
wg.Add(1)
|
|
|
|
defer wg.Done()
|
2017-02-04 02:01:52 +00:00
|
|
|
defer conn.Close()
|
|
|
|
for msg := range log.tail() {
|
2017-02-04 08:59:33 +00:00
|
|
|
msg = fmt.Sprintf("%s\n", msg)
|
2017-02-04 02:01:52 +00:00
|
|
|
conn.Write([]byte(msg))
|
|
|
|
}
|
2017-02-05 00:25:30 +00:00
|
|
|
conn.Write([]byte("bye\n"))
|
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
|
|
|
|
}
|