longganhua
2019-07-18 2fcec5d0debb4819c651e8f3b1287f18de9efee9
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
package agent
 
import (
    "log"
 
    "github.com/hashicorp/logutils"
)
 
// logStream is used to stream logs to a client over IPC
type logStream struct {
    client streamClient
    filter *logutils.LevelFilter
    logCh  chan string
    logger *log.Logger
    seq    uint64
}
 
func newLogStream(client streamClient, filter *logutils.LevelFilter,
    seq uint64, logger *log.Logger) *logStream {
    ls := &logStream{
        client: client,
        filter: filter,
        logCh:  make(chan string, 512),
        logger: logger,
        seq:    seq,
    }
    go ls.stream()
    return ls
}
 
func (ls *logStream) HandleLog(l string) {
    // Check the log level
    if !ls.filter.Check([]byte(l)) {
        return
    }
 
    // Do a non-blocking send
    select {
    case ls.logCh <- l:
    default:
        // We can't log syncronously, since we are already being invoked
        // from the logWriter, and a log will need to invoke Write() which
        // already holds the lock. We must therefor do the log async, so
        // as to not deadlock
        go ls.logger.Printf("[WARN] agent.ipc: Dropping logs to %v", ls.client)
    }
}
 
func (ls *logStream) Stop() {
    close(ls.logCh)
}
 
func (ls *logStream) stream() {
    header := responseHeader{Seq: ls.seq, Error: ""}
    rec := logRecord{Log: ""}
 
    for line := range ls.logCh {
        rec.Log = line
        if err := ls.client.Send(&header, &rec); err != nil {
            ls.logger.Printf("[ERR] agent.ipc: Failed to stream log to %v: %v",
                ls.client, err)
            return
        }
    }
}