forked from slackhq/go-audit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
marshaller.go
182 lines (154 loc) · 4.49 KB
/
marshaller.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
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
package main
import (
"os"
"regexp"
"syscall"
"time"
)
const (
EVENT_EOE = 1320 // End of multi packet event
)
type AuditMarshaller struct {
msgs map[int]*AuditMessageGroup
writer *AuditWriter
lastSeq int
missed map[int]bool
worstLag int
eventMin uint16
eventMax uint16
trackMessages bool
logOutOfOrder bool
maxOutOfOrder int
attempts int
filters map[string]map[uint16][]*regexp.Regexp // { syscall: { mtype: [regexp, ...] } }
}
type AuditFilter struct {
messageType uint16
regex *regexp.Regexp
syscall string
}
// Create a new marshaller
func NewAuditMarshaller(w *AuditWriter, eventMin uint16, eventMax uint16, trackMessages, logOOO bool, maxOOO int, filters []AuditFilter) *AuditMarshaller {
am := AuditMarshaller{
writer: w,
msgs: make(map[int]*AuditMessageGroup, 5), // It is not typical to have more than 2 message groups at any given time
missed: make(map[int]bool, 10),
eventMin: eventMin,
eventMax: eventMax,
trackMessages: trackMessages,
logOutOfOrder: logOOO,
maxOutOfOrder: maxOOO,
filters: make(map[string]map[uint16][]*regexp.Regexp),
}
for _, filter := range filters {
if _, ok := am.filters[filter.syscall]; !ok {
am.filters[filter.syscall] = make(map[uint16][]*regexp.Regexp)
}
if _, ok := am.filters[filter.syscall][filter.messageType]; !ok {
am.filters[filter.syscall][filter.messageType] = []*regexp.Regexp{}
}
am.filters[filter.syscall][filter.messageType] = append(am.filters[filter.syscall][filter.messageType], filter.regex)
}
return &am
}
// Ingests a netlink message and likely prepares it to be logged
func (a *AuditMarshaller) Consume(nlMsg *syscall.NetlinkMessage) {
aMsg := NewAuditMessage(nlMsg)
if aMsg.Seq == 0 {
// We got an invalid audit message, return the current message and reset
a.flushOld()
return
}
if a.trackMessages {
a.detectMissing(aMsg.Seq)
}
if nlMsg.Header.Type < a.eventMin || nlMsg.Header.Type > a.eventMax {
// Drop all audit messages that aren't things we care about or end a multi packet event
a.flushOld()
return
} else if nlMsg.Header.Type == EVENT_EOE {
// This is end of event msg, flush the msg with that sequence and discard this one
a.completeMessage(aMsg.Seq)
return
}
if val, ok := a.msgs[aMsg.Seq]; ok {
// Use the original AuditMessageGroup if we have one
val.AddMessage(aMsg)
} else {
// Create a new AuditMessageGroup
a.msgs[aMsg.Seq] = NewAuditMessageGroup(aMsg)
}
a.flushOld()
}
// Outputs any messages that are old enough
// This is because there is no indication of multi message events coming from kaudit
func (a *AuditMarshaller) flushOld() {
now := time.Now()
for seq, msg := range a.msgs {
if msg.CompleteAfter.Before(now) || now.Equal(msg.CompleteAfter) {
a.completeMessage(seq)
}
}
}
// Write a complete message group to the configured output in json format
func (a *AuditMarshaller) completeMessage(seq int) {
var msg *AuditMessageGroup
var ok bool
if msg, ok = a.msgs[seq]; !ok {
//TODO: attempted to complete a missing message, log?
return
}
if a.dropMessage(msg) {
delete(a.msgs, seq)
return
}
if err := a.writer.Write(msg); err != nil {
el.Println("Failed to write message. Error:", err)
os.Exit(1)
}
delete(a.msgs, seq)
}
func (a *AuditMarshaller) dropMessage(msg *AuditMessageGroup) bool {
filters, ok := a.filters[msg.Syscall]
if !ok {
return false
}
for _, msg := range msg.Msgs {
if fg, ok := filters[msg.Type]; ok {
for _, filter := range fg {
if filter.MatchString(msg.Data) {
return true
}
}
}
}
return false
}
// Track sequence numbers and log if we suspect we missed a message
func (a *AuditMarshaller) detectMissing(seq int) {
if seq > a.lastSeq+1 && a.lastSeq != 0 {
// We likely leap frogged over a msg, wait until the next sequence to make sure
for i := a.lastSeq + 1; i < seq; i++ {
a.missed[i] = true
}
}
for missedSeq := range a.missed {
if missedSeq == seq {
lag := a.lastSeq - missedSeq
if lag > a.worstLag {
a.worstLag = lag
}
if a.logOutOfOrder {
el.Println("Got sequence", missedSeq, "after", lag, "messages. Worst lag so far", a.worstLag, "messages")
}
delete(a.missed, missedSeq)
} else if seq-missedSeq > a.maxOutOfOrder {
el.Printf("Likely missed sequence %d, current %d, worst message delay %d\n", missedSeq, seq, a.worstLag)
delete(a.missed, missedSeq)
}
}
if seq > a.lastSeq {
// Keep track of the largest sequence
a.lastSeq = seq
}
}