forked from hellxoooo/SScan
-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathSScan.py
322 lines (280 loc) · 12.6 KB
/
SScan.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
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
#!/usr/bin/python3
# -*- coding:utf-8 -*-
# @Author : yhy
import fire
import math
import os
from datetime import datetime
from config.log import logger
import glob
import re
import time
from lib.common.TestProxy import testProxy
from config.banner import SScan_banner
from lib.common.report import save_report
from lib.common.common import prepare_targets, scan_process
from config import setting
from lib.common.utils import clear_queue, check_fofa, ctrl_quit
import multiprocessing
import signal
import warnings
warnings.filterwarnings('ignore')
class SScan(object):
"""
InfoScan help summary page\n
InfoScan is a Sensitive information detection and vulnerability scanning program
Example:
python3 SScan.py version
python3 SScan.py --host example.com run
python3 SScan.py --file domains.txt run
:param str host: HOST1 HOST2 ... Scan several hosts from command line
:param str file: Load new line delimited targets from TargetFile
:param str dire: Load all *.txt files from TargetDirectory
:param int network: Scan all Target/MASK neighbour hosts, should be an int between 8 and 31
:param int t: Num of scan threads for each scan process, 10 by default
:param tuple rule: RuleFileName1,RuleFileName2,... Import specified rules files only.
:param bool crawl: crawling, crawl <a href='...'> (default True)
:param bool checkcdn: Check the CDN and skip the IP where the CDN exists (default True)
:param bool full: Process all sub directories /x/y/z/,/x/ /x/y/ (default True)
:param str script: ScriptName1,ScriptName2,...
:param bool script_only: Scan with user scripts only
:param bool noscripts: Disable all scripts (default False)
:param bool browser: Do not open web browser to view report (default True)
"""
def __init__(self, host=None, file=None, dire="", network=32, t=10, rule=None,
full=True, script=None, noscripts=False, crawl=True,
browser=True, script_only=False, checkcdn=True):
self.host = host
self.file = file
self.rule_files = []
self.script_files = []
self.dire = dire
self.network = network
self.t = t
self.rule = rule
self.crawl = crawl
self.checkcdn = checkcdn
self.fileull = full
self.scripts_only = script_only
self.script = script
self.no_scripts = noscripts
self.browser = browser
if self.file:
self.input_files = [self.file]
elif self.dire:
self.input_files = glob.glob(self.dire + '/*.txt')
elif self.host:
self.input_files = [self.host]
self.require_no_http = True # 所有插件都不依赖 HTTP 连接池
self.require_index_doc = False # 插件需要请求首页
self.require_ports = set() # 插件扫描所需端口
# 加载相关配置
def config_param(self):
"""
Config parameter
"""
if self.dire:
self.dire = glob.glob(self.dire + '/*.txt')
if self.rule is None:
self.rule_files = glob.glob('rules/*.txt')
else:
if isinstance(self.rule, str):
rule = self.rule.split()
else:
rule = self.rule
for rule_name in rule:
if not rule_name.endswith('.txt'):
rule_name += '.txt'
if not os.path.exists('rules/%s' % rule_name):
logger.log('FATAL', 'Rule file not found: %s' % rule_name)
exit(-1)
self.rule_files.append(f'rules/{rule_name}')
if not self.no_scripts:
if self.script is None:
self.script_files = glob.glob('scripts/*.py')
else:
if isinstance(self.script, str):
script = self.script.split()
else:
script = self.script
for script_name in script:
if not script_name.lower().endswith('.py'):
script_name += '.py'
if not os.path.exists('scripts/%s' % script_name):
logger.log('FATAL', 'Rule file not found: %s' % script_name)
exit(-1)
self.script_files.append('scripts/%s' % script_name)
pattern = re.compile(r'ports_to_check.*?\\=(.*)')
for _script in self.script_files:
with open(_script, encoding='UTF-8') as f:
content = f.read()
if content.find('self.http_request') > 0 or content.find('self.session') > 0:
self.require_no_http = False # 插件依赖HTTP连接池
if content.find('self.index_') > 0:
self.require_no_http = False
self.require_index_doc = True
# 获取插件需要的端口
m = pattern.search(content)
if m:
m_str = m.group(1).strip()
if m_str.find('#') > 0: # 去掉注释
m_str = m_str[:m_str.find('#')]
if m_str.find('[') < 0:
if int(m_str) not in self.require_ports:
self.require_ports.add(int(m_str))
else:
for port in eval(m_str):
if port not in self.require_ports:
self.require_ports.add(int(port))
# 检查命令行输入
def check_param(self):
"""
Check parameter
"""
if not (self.file or self.dire or self.host):
msg = '\nself missing! One of following self should be specified \n' \
' \t--f TargetFile \n' \
' \t--d TargetDirectory \n' \
' \t--host www.host1.com www.host2.com 8.8.8.8'
logger.log('FATAL', msg)
exit(-1)
if self.file and not os.path.isfile(self.file):
logger.log('FATAL', 'TargetFile not found: %s' % self.file)
exit(-1)
if self.dire and not os.path.isdir(self.dire):
logger.log('FATAL', 'TargetDirectory not found: %s' % self.dire)
exit(-1)
self.network = int(self.network)
if not (8 <= self.network <= 32):
logger.log('FATAL', 'Network should be an integer between 24 and 31')
exit(-1)
def main(self):
q_targets = multiprocessing.Manager().Queue() # targets Queue
q_targets_list = []
q_results = multiprocessing.Manager().Queue() # log Queue
for input_file in self.input_files:
# 读取目标
if self.host:
target_list = self.host.replace(',', ' ').strip().split()
elif self.file or self.dire:
with open(input_file, encoding='UTF-8') as inFile:
target_list = inFile.readlines()
try:
import threading
# 实时生成报告
target_count = len(target_list) # 目标数
# 生成报告,管理标准输出
threading.Thread(target=save_report, args=(self, q_results, input_file, target_count)).start()
clear_queue(q_results)
clear_queue(q_targets)
start_time = time.time()
# 根据电脑 CPU 的内核数量, 创建相应的进程池
# count = multiprocessing.cpu_count()
count = 16
# 少量目标,至多创建2倍扫描进程
if len(target_list) * 2 < count:
count = len(target_list) * 2
pool = multiprocessing.Pool(count)
num = math.ceil(len(target_list)/count)
domain_start_time = time.time()
logger.log('INFOR', f'Domain name resolution, subnet mask processing, CDN detection, 80、443 port detection, FOFA search started.')
for i in range(0, len(target_list), count):
target = target_list[i:i + count]
pool.apply_async(prepare_targets, args=(target, q_targets, self))
pool.close()
pool.join()
logger.log("INFOR", f'Domain name resolution, subnet mask processing, CDN detection, 80、443 port detection, FOFA search is over in %.1f seconds!' % (
time.time() - domain_start_time))
time.sleep(1.0)
while True:
if not q_targets.empty():
q_targets_list.append(q_targets.get())
else:
break
logger.log("INFOR", f'Targets process all done in %.1f seconds!' % (time.time() - start_time))
# q_targets.get() {'scheme': 'https', 'host': '127.0.0.1', 'port': 443, 'path': '', 'ports_open': [80, 443], 'is_neighbor': 0}
if len(target_list) * 2 < count:
count = len(target_list) * 2
# pool = multiprocessing.Pool(count)
# logger.log('INFOR', f'{count} scan process created.')
#
# for target in q_targets_list:
# pool.apply_async(scan_process, args=(target, q_results, self))
#
# pool.close()
# pool.join()
# 进度条设置
from rich.progress import (
BarColumn,
TimeRemainingColumn,
Progress,
)
progress = Progress(
"[progress.description]{task.description}",
BarColumn(),
"[progress.percentage]{task.percentage:>3.0f}%",
TimeRemainingColumn(),
"[bold red]{task.completed}/{task.total}",
transient=True, # 100%后隐藏进度条
)
with progress:
targets = []
for target in q_targets_list:
tmp = [target, q_results, self]
targets.append(tmp)
task_id = progress.add_task("[cyan]Leak detection...", total=len(targets), start=False)
with multiprocessing.Pool(processes=16) as pool:
results = pool.imap_unordered(scan_process, targets)
for result in results:
# progress.print(result)
progress.advance(task_id)
time.sleep(1.0)
cost_time = time.time() - start_time
cost_min = int(cost_time / 60)
cost_min = '%s min ' % cost_min if cost_min > 0 else ''
cost_seconds = '%.1f' % (cost_time % 60)
logger.log('INFOR', f'Scanned {len(q_targets_list)} targets in {cost_min}{cost_seconds} seconds.')
except KeyboardInterrupt as e:
setting.stop_me = True
logger.log('INFOR', 'Scan aborted.')
exit(-1)
except FileNotFoundError as e:
logger.log('INFOR', 'Scan aborted.')
exit(-1)
pass
except Exception as e:
logger.log('ERROR', '[__main__.exception] %s %s' % (type(e), str(e)))
setting.stop_me = True
def print(self):
"""
InfoScan running entrance
:return: All subdomain log
:rtype: list
"""
print(SScan_banner)
dt = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
print(f'[*] Starting InfoScan @ {dt}\n')
self.check_param()
self.config_param()
check_fofa()
if self.no_scripts:
logger.log('INFOR', '* Scripts scan was disabled.')
if self.require_ports:
logger.log('INFOR', '* Scripts scan port check: %s' % ','.join([str(x) for x in self.require_ports]))
def run(self):
self.print()
# 网络连通性检查
# testProxy(cmd, 1)
# if testProxy(1):
self.main()
@staticmethod
def version():
"""
Print version information and exit
"""
print(SScan_banner)
exit(0)
if __name__ == '__main__':
# 优雅的使用 ctrl c 退出
signal.signal(signal.SIGINT, ctrl_quit)
fire.Fire(SScan)