-
Notifications
You must be signed in to change notification settings - Fork 114
/
grottsniffer.py
118 lines (99 loc) · 5.34 KB
/
grottsniffer.py
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
import socket
#import select
#import time
import sys
import struct
#import textwrap
#from itertools import cycle # to support "cycling" the iterator
#import time, json, datetime, codecs
from grottdata import procdata
class Sniff:
def __init__(self,conf):
self.conn = socket.socket(socket.AF_PACKET, socket.SOCK_RAW, socket.ntohs(3))
# if conf.verbose: print("\nGrott monitoring started\n")
if conf.verbose:
print("")
print("\nGrott sniff mode started\n")
def main(self,conf):
while True:
self.raw_data, self.addr = self.conn.recvfrom(65535)
self.eth = Ethernet(self.raw_data)
if conf.trace:
print("\n" + "\t - " + 'Ethernet Frame:')
print("\t - " + 'Destination: {}, Source: {}, Protocol: {}'.format(self.eth.dest_mac, self.eth.src_mac, self.eth.proto))
# IPv4
if self.eth.proto == 8:
self.ipv4 = IPv4(self.eth.data)
if conf.trace:
print("\t - " + 'IPv4 Packet protocol 8 :')
print("\t\t - " + 'Version: {}, Header Length: {}, TTL: {},'.format(self.ipv4.version, self.ipv4.header_length, self.ipv4.ttl))
print("\t\t - " + 'Protocol: {}, Source: {}, Target: {}'.format(self.ipv4.proto, self.ipv4.src, self.ipv4.target))
# TCP
#elif self.ipv4.proto == 6:
if self.ipv4.proto == 6:
self.tcp = TCP(self.ipv4.data)
if conf.trace:
print("\t - " + 'TCP Segment protocol 6 found')
print("\t\t - " + 'Source Port: {}, Destination Port: {}'.format(self.tcp.src_port, self.tcp.dest_port))
print("\t\t - " + 'Source IP: {}, Destination IP: {}'.format(self.ipv4.src, self.ipv4.target))
if self.tcp.dest_port == conf.growattport and self.ipv4.target == conf.growattip:
if conf.verbose:
print("\t - "+ 'TCP Segment Growatt:')
print("\t\t - " + 'Source Port: {}, Destination Port: {}'.format(self.tcp.src_port, self.tcp.dest_port))
print("\t\t - " + 'Source IP: {}, Destination IP: {}'.format(self.ipv4.src, self.ipv4.target))
print("\t\t - " + 'Sequence: {}, Acknowledgment: {}'.format(self.tcp.sequence, self.tcp.acknowledgment))
print("\t\t - " + 'Flags:')
print("\t\t\t - " + 'URG: {}, ACK: {}, PSH: {}'.format(self.tcp.flag_urg, self.tcp.flag_ack, self.tcp.flag_psh))
print("\t\t\t - " + 'RST: {}, SYN: {}, FIN:{}'.format(self.tcp.flag_rst, self.tcp.flag_syn, self.tcp.flag_fin))
if len(self.tcp.data) > conf.minrecl :
procdata(conf,self.tcp.data)
else:
if conf.verbose: print("\t - " + 'Data less then minimum record length, data not processed')
# Other IPv4 Not used
else:
if conf.trace:
print("\t - " + 'Other IPv4 Data')
#print(format_multi_line(DATA_TAB_2, self.ipv4.data))
else:
if conf.trace:
print("\t - " + 'No IPV4 Ethernet Data')
#print(TAB_1 + format_multi_line(DATA_TAB_1, self.eth.data))
# Returns MAC as string from bytes (ie AA:BB:CC:DD:EE:FF)
def get_mac_addr(mac_raw):
byte_str = map('{:02x}'.format, mac_raw)
mac_addr = ':'.join(byte_str).upper()
return mac_addr
#Unpack ethernet packet
class Ethernet:
def __init__(self, raw_data):
dest, src, prototype = struct.unpack('! 6s 6s H', raw_data[:14])
self.dest_mac = get_mac_addr(dest)
self.src_mac = get_mac_addr(src)
self.proto = socket.htons(prototype)
self.data = raw_data[14:]
#Unpacks IPV4 packet
class IPv4:
def __init__(self, raw_data):
version_header_length = raw_data[0]
self.version = version_header_length >> 4
self.header_length = (version_header_length & 15) * 4
self.ttl, self.proto, src, target = struct.unpack('! 8x B B 2x 4s 4s', raw_data[:20])
self.src = self.ipv4addr(src)
self.target = self.ipv4addr(target)
self.data = raw_data[self.header_length:]
# Returns properly formatted IPv4 address
def ipv4addr(self, addr):
return '.'.join(map(str, addr))
# Unpack TCP Segment
class TCP:
def __init__(self, raw_data):
(self.src_port, self.dest_port, self.sequence, self.acknowledgment, offset_reserved_flags) = struct.unpack(
'! H H L L H', raw_data[:14])
offset = (offset_reserved_flags >> 12) * 4
self.flag_urg = (offset_reserved_flags & 32) >> 5
self.flag_ack = (offset_reserved_flags & 16) >> 4
self.flag_psh = (offset_reserved_flags & 8) >> 3
self.flag_rst = (offset_reserved_flags & 4) >> 2
self.flag_syn = (offset_reserved_flags & 2) >> 1
self.flag_fin = offset_reserved_flags & 1
self.data = raw_data[offset:]