forked from cloudflare/bpftools
-
Notifications
You must be signed in to change notification settings - Fork 1
/
pcap2hex
executable file
·115 lines (92 loc) · 2.91 KB
/
pcap2hex
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
#!/usr/bin/env python
import getopt
import os
import sys
import string
import struct
import pcappy
import bpftools.utils
def usage():
print """
pcap2hex.py [ OPTIONS ] [ pcap file... ]
Read pcap from stdin or given pcap files and print hex-encoded
packets. Additionally pcap2hex can normalize DLT_LINUX_SLL l2 header
(used by "tcpdump -i any") to something that looks more like
DLT_EN10MB (by "tcpdump -i eth0"). Pcap2hex can also scrub (anonymize)
packets by overwriting l2 MAC addresses and l3 IPv4 and IPv6 addresses
with zeros.
Options are:
-h, --help print this message
-s, --scrub scrub/anonymize MAC and IP addresses
-n, --no-normalize don't normalize L2 headers from 16 to
14 bytes (from DLT_LINUX_SLL to DLT_EN10MB)
-a, --ascii print printable asci characters
""".lstrip()
sys.exit(2)
def main():
scrub = ascii = False
normalize = True
try:
opts, args = getopt.getopt(sys.argv[1:], "hsna",
["help", "scrub", "no-normalize", "ascii"])
except getopt.GetoptError as err:
print str(err)
usage()
for o, a in opts:
if o in ("-h", "--help"):
usage()
elif o in ("-s", "--scrub"):
scrub = True
elif o in ("-n", "--no-normalize"):
normalize = False
elif o in ("-a", "--ascii"):
ascii = True
else:
assert False, "unhandled option"
if not args:
readfds = [sys.stdin]
else:
readfds = [open(fname, 'rb') for fname in args]
l3_off = None
l3_off_bad = 0
for fd in readfds:
p = pcappy.open_offline(fd)
while True:
try:
r = p.next_ex()
except pcappy.PcapPyException:
break
if r is None:
break
hdr, data = r
if l3_off is None:
l3_off = bpftools.utils.find_ip_offset(data)
if l3_off is None:
l3_off_bad += 1
if l3_off_bad > 5:
raise Exception("Can't find IP offset")
continue
if scrub:
data = bpftools.utils.do_scrub(data, l3_off)
if normalize and l3_off in (16,):
data = data[2:]
h = data.encode('hex')
if not ascii:
print h
else:
s = ''.join([c if (c in string.printable and
c not in string.whitespace) else '.'
for c in data])
print "%s\t%s" % (h, s)
sys.stdout.flush()
# normal exit crashes due to a double free error in pcappy
os._exit(0)
if __name__ == "__main__":
try:
main()
except IOError, e:
if e.errno == 32:
os._exit(-1)
raise
except KeyboardInterrupt:
os._exit(-1)