forked from softwaresec-labs/MADONNA
-
Notifications
You must be signed in to change notification settings - Fork 0
/
feature_extractions.py
251 lines (210 loc) · 7.29 KB
/
feature_extractions.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
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
import re
import math
import dns.exception
import dns.resolver
import dns.reversename
from itertools import combinations as combs
from difflib import SequenceMatcher
from geoip2.database import Reader
import os.path
import sys
import whois
import tldextract as tld
import urllib.error
import urllib.request
import http.client
import socket
from bs4 import BeautifulSoup
headers = {'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:80.0) Gecko/20100101 Firefox/80.0'}
servers = ['1.1.1.1', '8.8.8.8', '208.67.222.222']
resolvers = []
for server in servers:
resolver = dns.resolver.Resolver(False)
resolver.nameservers = [server]
resolvers.append(resolver)
TTLs = list()
def extract_features(domain):
domain = re.sub(r'^www\d*', '', domain.lower()).lstrip('.')
# domain_name_features
length = len(domain)
# n_constants = sum([c in domain for c in 'bcdfghjklmnpqrstvwxyz'])
n_vowels = sum([c in domain for c in 'aeiou'])
n_vowel_chars = sum([domain.count(c) for c in 'aeiou'])
n_constant_chars = sum([domain.count(c) for c in 'bcdfghjklmnpqrstvwxyz'])
n_nums = sum([domain.count(c) for c in '0123456789'])
n_other_chars = sum([c not in 'abcdefghijklmnopqrstuvwxyz0123456789.' for c in domain])
probas = {i: domain.count(i) / len(domain) for i in set(domain)}
entropy = -sum((p * math.log2(p)) for p in probas.values())
# dns_features
ns_names = __get_rr(domain,'NS')
mx_names = __get_rr(domain, 'MX')
ips = __get_rr(domain, 'A', ttl=True)
n_ns = len(ns_names)
n_mx = len(mx_names)
# n_ptr = get_n_ptr(ips)
ns_similarity = get_ns_similarity(ns_names, ips)
n_countries = get_n_countries(ips)
# n_countries = ""
domain_2 = domain.rstrip('\n')
ext = tld.extract(domain_2)
compact_domain = '.'.join(filter(None, [ext.domain, ext.suffix]))
#web_features
life_time = get_life_time(compact_domain)
# active_time = get_active_time(compact_domain)
n_labels = get_n_labels(domain_2,compact_domain)
# features = {
# "domain": domain,
# "length": length,
# "n_ns": n_ns,
# "n_vowels": n_vowels,
# "life_time": life_time,
# "n_vowel_chars": n_vowel_chars,
# "n_constant_chars": n_constant_chars,
# "n_nums": n_nums,
# "n_other_chars": n_other_chars,
# "entropy": entropy,
# "n_mx": n_mx,
# "ns_similarity": ns_similarity,
# "n_countries": n_countries,
# "n_labels": n_labels
# }
features_array = [length,n_ns,n_vowels,life_time,n_vowel_chars,n_constant_chars,n_nums,n_other_chars,entropy,n_mx,ns_similarity,n_countries,n_labels]
return features_array
def __get_rr(domain,_type, ttl=False):
names = set()
for resolver in resolvers:
try:
records = resolver.resolve(domain, _type)
for record in records:
names.add(str(record))
if ttl:
TTLs.append(records.rrset.ttl)
except (dns.resolver.NXDOMAIN, dns.resolver.NoAnswer,
dns.exception.Timeout, dns.resolver.NoNameservers, dns.name.LabelTooLong):
pass
return names
def get_n_ptr(ips):
cloudflare = dns.resolver.Resolver()
cloudflare.nameservers = ['1.1.1.1']
ptr_names = set()
for ip in ips:
rev_name = dns.reversename.from_address(ip)
try:
ptr_records = cloudflare.resolve(rev_name, 'PTR')
except (dns.resolver.NXDOMAIN, dns.resolver.NoAnswer,
dns.exception.Timeout, dns.resolver.NoNameservers):
continue
for record in ptr_records:
ptr_names.add(str(record))
return len(ptr_names)
def similarity(a, b):
return SequenceMatcher(None, a, b).ratio()
def get_ns_similarity(ns_names,ips):
if len(ns_names) > 2:
similarities = list()
all_combs = combs(ns_names, 2)
for comb in all_combs:
similarities.append(similarity(*comb))
return sum(similarities) / len(similarities)
elif len(ips) == 0 or len(ns_names) == 0:
return 0.0
else:
return 1.0
def get_n_countries(ips):
ip_countries = set()
packagedir = os.path.dirname(__file__)
# # dbpath = os.path.join(packagedir, '../../thirdparty/geoip/GeoLite2-City.mmdb')
# # city_reader = Reader(dbpath)
city_reader = Reader("GeoLite2-City.mmdb")
for ip in ips:
try:
city_resp = city_reader.city(ip)
ip_countries.add(city_resp.country.iso_code)
except:
pass
return len(ip_countries)
def __get_whois(compact_domain):
expire=""
create=""
update=""
try:
sys.stdout = open(os.devnull, 'w')
w = whois.whois(compact_domain)
sys.stdout = sys.__stdout__
if w.creation_date:
if isinstance(w.creation_date, list):
create = w.creation_date[0]
elif isinstance(w.creation_date, str):
pass
else:
create = w.creation_date
if isinstance(create, str):
create = None
if w.updated_date:
if isinstance(w.updated_date, list):
update = w.updated_date[0]
elif isinstance(w.updated_date, str):
pass
else:
update = w.updated_date
if isinstance(update, str):
update = None
if w.expiration_date:
if isinstance(w.expiration_date, list):
expire = w.expiration_date[0]
elif isinstance(w.expiration_date, str):
pass
else:
expire = w.expiration_date
if isinstance(expire, str):
expire = None
except whois.parser.PywhoisError:
pass
__whois = True
return __whois,expire,create,update
def get_life_time(compact_domain):
__whois,expire,create,update = __get_whois(compact_domain)
if expire and create:
td = expire - create
return td.days
else:
return 0
def get_active_time(compact_domain):
__whois,expire,create,update = __get_whois(compact_domain)
if update and create:
td = update - create
return td.days
else:
return get_life_time(compact_domain)
def get_html(url):
request = urllib.request.Request('http://'+url, headers=headers)
try:
resp = urllib.request.urlopen(request, timeout=5)
return resp.read()
except (http.client.BadStatusLine, http.client.IncompleteRead, http.client.HTTPException,
UnicodeError, UnicodeEncodeError): # possibly plaintext or HTTP/1.0
print("get_html:error")
return None
except:
raise
def get_n_labels(domain,compact_domain):
html = None
try:
html = get_html(domain)
except (urllib.error.HTTPError, urllib.error.URLError,
ConnectionResetError, socket.timeout):
try:
html = get_html(compact_domain)
except (urllib.error.HTTPError, urllib.error.URLError,
ConnectionResetError, socket.timeout):
print("get_n_labels:error")
pass
if html:
try:
soup = BeautifulSoup(html, features='html.parser')
return len(soup.find_all())
except UnboundLocalError:
print("get_n_labels_if_html:error")
return 0
else:
return 0