forked from mattn/goreman
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlog.go
133 lines (120 loc) · 2.62 KB
/
log.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
package main
import (
"bytes"
"fmt"
"io"
"sync"
"time"
"github.com/mattn/go-colorable"
)
type clogger struct {
idx int
proc string
writes chan []byte
done chan struct{}
timeout time.Duration // how long to wait before printing partial lines
buffers buffers // partial lines awaiting printing
}
var colors = []int{
32, // green
36, // cyan
35, // magenta
33, // yellow
34, // blue
31, // red
}
var mutex = new(sync.Mutex)
var out = colorable.NewColorableStdout()
type buffers [][]byte
func (v *buffers) consume(n int64) {
for len(*v) > 0 {
ln0 := int64(len((*v)[0]))
if ln0 > n {
(*v)[0] = (*v)[0][n:]
return
}
n -= ln0
*v = (*v)[1:]
}
}
func (v *buffers) WriteTo(w io.Writer) (n int64, err error) {
for _, b := range *v {
nb, err := w.Write(b)
n += int64(nb)
if err != nil {
v.consume(n)
return n, err
}
}
v.consume(n)
return n, nil
}
// write any stored buffers, plus the given line, then empty out
// the buffers.
func (l *clogger) writeBuffers(line []byte) {
now := time.Now().Format("15:04:05")
mutex.Lock()
fmt.Fprintf(out, "\x1b[%dm", colors[l.idx])
fmt.Fprintf(out, "%s %*s | ", now, maxProcNameLength, l.proc)
fmt.Fprintf(out, "\x1b[m")
l.buffers = append(l.buffers, line)
l.buffers.WriteTo(out)
l.buffers = l.buffers[0:0]
mutex.Unlock()
}
// bundle writes into lines, waiting briefly for completion of lines
func (l *clogger) writeLines() {
var tick <-chan time.Time
for {
select {
case w, ok := <-l.writes:
if !ok {
if len(l.buffers) > 0 {
l.writeBuffers([]byte("\n"))
}
return
}
buf := bytes.NewBuffer(w)
for {
line, err := buf.ReadBytes('\n')
if len(line) > 0 {
if line[len(line)-1] == '\n' {
// any text followed by a newline should flush
// existing buffers. a bare newline should flush
// existing buffers, but only if there are any.
if len(line) != 1 || len(l.buffers) > 0 {
l.writeBuffers(line)
}
tick = nil
} else {
l.buffers = append(l.buffers, line)
tick = time.After(l.timeout)
}
}
if err != nil {
break
}
}
l.done <- struct{}{}
case <-tick:
if len(l.buffers) > 0 {
l.writeBuffers([]byte("\n"))
}
tick = nil
}
}
}
// write handler of logger.
func (l *clogger) Write(p []byte) (int, error) {
l.writes <- p
<-l.done
return len(p), nil
}
// create logger instance.
func createLogger(proc string, colorIndex int) *clogger {
mutex.Lock()
defer mutex.Unlock()
l := &clogger{idx: colorIndex, proc: proc, writes: make(chan []byte), done: make(chan struct{}), timeout: 2 * time.Millisecond}
go l.writeLines()
return l
}