-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmain.go
143 lines (130 loc) · 3.23 KB
/
main.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
package main
import (
"bytes"
"flag"
"fmt"
"github.com/couchbase/gomemcached"
"github.com/google/gopacket"
"github.com/google/gopacket/layers"
"github.com/google/gopacket/pcap"
"log"
"os"
"runtime/debug"
"time"
)
var options struct {
localIP string
remoteIP string
snapshotLen int
printInterval int
printAll bool
timeout int
mode string
}
func argParse() {
flag.StringVar(&options.localIP, "localIP", "",
"the ip of the machine which dcpdump is running")
flag.StringVar(&options.remoteIP, "remoteIP", "",
"the ip which is interacting with this machine")
flag.IntVar(&options.snapshotLen, "snapshotLen", 1024,
"package will be cut if more than snapshotLen")
flag.IntVar(&options.printInterval, "printInterval", 60,
"the interval to pop the metrics")
flag.BoolVar(&options.printAll, "printAll", true,
"whether to print all the info")
flag.IntVar(&options.timeout, "timeout", 0,
"timeout setting, in milliseconds")
flag.StringVar(&options.mode, "mode", "client",
"run at server or client")
flag.Usage = usage
flag.Parse()
}
func usage() {
fmt.Fprintf(os.Stderr, "Usage : %s [OPTIONS] \n", os.Args[0])
flag.PrintDefaults()
}
func init() {
argParse()
}
var (
promiscuous bool = false
timeout time.Duration = 30 * time.Second
reqChan = make(chan MCReqAndTime)
respChan = make(chan MCRespAndTime)
)
func main() {
// Find device
network,err := FindInterface(options.localIP)
if err != nil {
panic(err)
}
// Open device
handle, err := pcap.OpenLive(network, int32(options.snapshotLen), promiscuous, timeout)
if err != nil {
log.Fatal(err)
}
defer handle.Close()
// Set filter
var filter string
if options.remoteIP != "" {
filter = fmt.Sprintf("port 11210 and host %s", options.remoteIP)
} else {
filter = fmt.Sprintf("port 11210")
}
err = handle.SetBPFFilter(filter)
if err != nil {
log.Fatal(err)
}
// analyse the couchbase dcp packet
go analyse()
packetSource := gopacket.NewPacketSource(handle, handle.LinkType())
for packet := range packetSource.Packets() {
dispatch(packet)
}
}
func dispatch(packet gopacket.Packet) {
defer func() {
if err := recover(); err != nil {
debug.PrintStack()
}
}()
// ip
ipLayer := packet.Layer(layers.LayerTypeIPv4)
if ipLayer == nil {
return
}
ip, _ := ipLayer.(*layers.IPv4)
// tcp
tcpLayer := packet.Layer(layers.LayerTypeTCP)
if tcpLayer == nil {
return
}
tcp, _ := tcpLayer.(*layers.TCP)
// application
applicationLayer := packet.ApplicationLayer()
if applicationLayer == nil {
return
}
payload := applicationLayer.Payload()
r := bytes.NewReader(payload)
switch payload[0] {
case 128:
rv := gomemcached.MCRequest{}
_, err := rv.Receive(r, nil)
if err == nil {
reqChan <- MCReqAndTime{rv, packet.Metadata().CaptureInfo.Timestamp, ip.SrcIP, tcp.SrcPort, ip.DstIP, tcp.DstPort}
} else {
//fmt.Println("Error decoding some part of the packet:", err)
}
case 129:
rv := gomemcached.MCResponse{}
_, err := rv.Receive(r, nil)
if err == nil {
respChan <- MCRespAndTime{rv, packet.Metadata().CaptureInfo.Timestamp, ip.SrcIP, tcp.SrcPort, ip.DstIP, tcp.DstPort}
} else {
//fmt.Println("Error decoding some part of the packet:", err)
}
default:
/* fmt.Printf("%s\n", payload) */
}
}