-
Notifications
You must be signed in to change notification settings - Fork 77
/
Copy pathparselog.go
118 lines (106 loc) · 2.68 KB
/
parselog.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
package ovpm
import (
"bufio"
"io"
"strconv"
"strings"
"time"
"github.com/sirupsen/logrus"
)
// clEntry reprsents a parsed entry that is present on OpenVPN
// log section CLIENT LIST.
type clEntry struct {
CommonName string `json:"common_name"`
RealAddress string `json:"real_address"`
BytesReceived uint64 `json:"bytes_received"`
BytesSent uint64 `json:"bytes_sent"`
ConnectedSince time.Time `json:"connected_since"`
}
// rtEntry reprsents a parsed entry that is present on OpenVPN
// log section ROUTING TABLE.
type rtEntry struct {
VirtualAddress string `json:"virtual_address"`
CommonName string `json:"common_name"`
RealAddress string `json:"real_address"`
LastRef time.Time `json:"last_ref"`
}
// parseStatusLog parses the received OpenVPN status log file.
// And then returns the parsed client information.
func parseStatusLog(f io.Reader) ([]clEntry, []rtEntry) {
// Recover any panics
defer func() {
if r := recover(); r != nil {
logrus.WithField("panic", r).Error("OpenVPN log file is corrupt")
}
}()
// Parsing stages.
const stageCL int = 0
const stageRT int = 1
const stageFin int = 2
// Parsing variables.
var currStage int
var skipFor int
var cl []clEntry
var rt []rtEntry
// Scan and parse the file by dividing it into chunks.
scanner, skipFor := bufio.NewScanner(f), 3
for lc := 0; scanner.Scan(); lc++ {
if skipFor > 0 {
skipFor--
continue
}
txt := scanner.Text()
switch currStage {
case stageCL:
if strings.Contains(txt, "ROUTING TABLE") {
currStage = stageRT
skipFor = 1
continue
}
dat := strings.Split(txt, ",")
cl = append(cl, clEntry{
CommonName: trim(dat[0]),
RealAddress: trim(dat[1]),
BytesReceived: stoui64(trim(dat[2])),
BytesSent: stoui64(trim(dat[3])),
ConnectedSince: stodt(trim(dat[4])),
})
case stageRT:
if strings.Contains(txt, "GLOBAL STATS") {
currStage = stageFin
break
}
dat := strings.Split(txt, ",")
rt = append(rt, rtEntry{
VirtualAddress: trim(dat[0]),
CommonName: trim(dat[1]),
RealAddress: trim(dat[2]),
LastRef: stodt(trim(dat[3])),
})
}
}
if err := scanner.Err(); err != nil {
panic(err)
}
return cl, rt
}
// stoi64 converts string to uint64.
func stoui64(s string) uint64 {
i, err := strconv.ParseInt(s, 0, 64)
if err != nil {
panic(err)
}
return uint64(i)
}
// stodt converts string to date time.
func stodt(s string) time.Time {
t, err := time.ParseInLocation(time.ANSIC, s, time.Local)
if err != nil {
panic(err)
}
return t
}
// trim will trim all leading and trailing whitespace from the s.
func trim(s string) string {
return strings.TrimSpace(s)
}