-
Notifications
You must be signed in to change notification settings - Fork 86
/
Copy pathSWProxy.py
executable file
·281 lines (233 loc) · 9.64 KB
/
SWProxy.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
#!/usr/bin/env python
from SWParser.smon_decryptor import decrypt_request, decrypt_response
import json
import logging
import os
import proxy
from SWPlugin import *
import socket
import sys
import argparse
import struct
import dpkt
VERSION = "0.100"
GITHUB = 'https://github.com/kakaroto/SWProxy'
logger = logging.getLogger("SWProxy")
class HTTP(proxy.TCP):
"""
HTTP proxy server implementation.
Spawns new process to proxy accepted client connection.
"""
def handle(self, client):
callback = SWProxyCallback()
proc = proxy.Proxy(client, callback)
proc.daemon = True
proc.start()
logger.debug('Started process {} to handle connection {}'.format(proc, client.conn))
class SWProxyCallback(object):
def __init__(self):
self.request = None
def onRequest(self, proxy, host, port, request):
try:
if request.url.path.startswith('/api/gateway'):
self.request = request # if we care about this api call, store request for decryption later
except AttributeError:
pass
def onResponse(self, proxy, response):
if self.request is None:
# we have not obtained a valid request yet
return
try:
req_plain, req_json = self._parse_request(self.request)
resp_plain, resp_json = self._parse_response(response)
if 'command' not in resp_json:
# we only want apis that are commands
self.request = None
return
try:
SWPlugin.call_plugins('process_request', (req_json, resp_json))
except Exception as e:
logger.exception('Exception while executing plugin : {}'.format(e))
except Exception as e:
logger.debug('unknown exception: {}'.format(e))
def onDone(self, proxy):
pass
def _parse_request(self, request):
""" takes a request, returns the decrypted plain and json """
plain = decrypt_request(request.body, 2 if '_c2.php' in self.request.url.path else 1)
return plain, json.loads(plain)
def _parse_response(self, response):
""" takes a response body, returns the decrypted plain and json """
plain = decrypt_response(response.body, 2 if '_c2.php' in self.request.url.path else 1)
return plain, json.loads(plain)
def get_external_ip():
my_ip = [[(s.connect(('8.8.8.8', 80)), s.getsockname()[0], s.close()) for s in [socket.socket(socket.AF_INET, socket.SOCK_DGRAM)]][0][1]][0]
return my_ip
def read_file_lines(fpath):
try:
fpath = resource_path(fpath)
with open(fpath, 'r') as fh:
return map(lambda x:x.strip(), fh.readlines())
except Exception:
logger.debug('Failed to read file at {}'.format(fpath))
return ''
def get_usage_text():
authors_text = read_file_lines('AUTHORS')
lines = []
lines.append("#"*40)
lines.append("# SWParser v{} - Summoners War Proxy # ".format(VERSION))
lines.append("#"*40)
lines.append("\tWritten by:\n\t\tKaKaRoTo\n")
lines.append("\tAuthors:")
for author in authors_text:
lines.append("\t\t{}".format(author))
lines.append("\n\tPlugins:")
for plugin in SWPlugin.plugins:
lines.append("\t\t{}".format(plugin.name))
lines.append("\nLicensed under LGPLv3 and available at: \n\t{}\n".format(GITHUB))
return "\n".join(lines)
def resource_path(relative_path):
# function to locate data files for pyinstaller single file executable
# ref: http://stackoverflow.com/a/32048136
if hasattr(sys, '_MEIPASS'):
return os.path.join(sys._MEIPASS, relative_path)
return os.path.join(os.path.abspath("."), relative_path)
def start_proxy_server(options):
my_ip = get_external_ip()
try:
print "Running Proxy server at {} on port {}".format(my_ip, options.port)
p = HTTP(my_ip, options.port)
p.run()
except KeyboardInterrupt:
pass
def parse_pcap(filename):
streams = dict() # Connections with current buffer
with open(filename, "rb") as f:
pcap = dpkt.pcap.Reader(f)
for ts, buf in pcap:
eth = dpkt.ethernet.Ethernet(buf)
if eth.type != dpkt.ethernet.ETH_TYPE_IP:
continue
ip = eth.data
if not isinstance(ip, dpkt.ip.IP):
try:
ip = dpkt.ip.IP(ip)
except:
continue
if ip.p != dpkt.ip.IP_PROTO_TCP:
continue
tcp = ip.data
if not isinstance(tcp, dpkt.tcp.TCP):
try:
tcp = dpkt.tcp.TCP(tcp)
except:
continue
tupl = (ip.src, ip.dst, tcp.sport, tcp.dport)
if tupl in streams:
streams[tupl] = streams[tupl] + tcp.data
else:
streams[tupl] = tcp.data
if (tcp.flags & dpkt.tcp.TH_FIN) != 0 and \
(tcp.dport == 80 or tcp.sport == 80) and \
len(streams[tupl]) > 0:
other_tupl = (ip.dst, ip.src, tcp.dport, tcp.sport)
stream1 = streams[tupl]
del streams[tupl]
try:
stream2 = streams[other_tupl]
del streams[other_tupl]
except:
stream2 = ""
if tcp.dport == 80:
requests = stream1
responses = stream2
else:
requests = stream2
responses = stream1
while len(requests):
try:
request = dpkt.http.Request(requests)
#print request.method, request.uri
except:
request = ''
requests = ''
try:
response = dpkt.http.Response(responses)
#print response.status
except:
response = ''
responses = ''
requests = requests[len(request):]
responses = requests[len(responses):]
if len(request) > 0 and len(response) > 0 and \
request.method == 'POST' and \
request.uri == '/api/gateway_c2.php' and \
response.status == '200':
try:
req_plain = decrypt_request(request.body, 2)
resp_plain = decrypt_response(response.body, 2)
req_json = json.loads(req_plain)
resp_json = json.loads(resp_plain)
if resp_json.get('command') in ['HubUserLogin', 'VisitFriend']:
try:
SWPlugin.call_plugins('process_request', (req_json, resp_json))
except Exception as e:
logger.exception('Exception while executing plugin : {}'.format(e))
except:
import traceback
e = sys.exc_info()[0]
traceback.print_exc()
elif (tcp.flags & dpkt.tcp.TH_FIN) != 0:
del streams[tupl]
if __name__ == "__main__":
parser = argparse.ArgumentParser(description='SWParser')
parser.add_argument('-d', '--debug', action="store_true", default=False)
parser.add_argument('-g', '--no-gui', action="store_true", default=False)
parser.add_argument('-p', '--port', type=int, default=8080)
options, unknown_args = parser.parse_known_args()
# Set up logger
level = "DEBUG" if options.debug else "INFO"
logging.basicConfig(level=level, filename="proxy.log", format='%(asctime)s: %(name)s - %(levelname)s - %(message)s')
logger.setLevel(logging.INFO)
print get_usage_text()
# Check if a PCAP file was passed in
pcap_filename = None
for arg in unknown_args:
if arg.endswith('.pcap'):
pcap_filename = arg
break
if pcap_filename:
# Parse a PCAP file
print "Parsing PCAP file..."
parse_pcap(sys.argv[1])
raw_input("Press Enter to exit...")
else:
# Run the proxy
# attempt to load gui; fallback if import error
if not options.no_gui:
try:
# Import here to avoid importing QT in CLI mode
from SWParser.gui import gui
from PyQt4.QtGui import QApplication, QIcon
from PyQt4.QtCore import QSize
except ImportError:
print "Failed to load GUI dependencies. Switching to CLI mode"
options.no_gui = True
if options.no_gui:
logger.addHandler(logging.StreamHandler())
start_proxy_server(options)
else:
app = QApplication(sys.argv)
# set the icon
icons_path = os.path.join(os.getcwd(), resource_path("icons/"))
app_icon = QIcon()
app_icon.addFile(icons_path +'16x16.png', QSize(16,16))
app_icon.addFile(icons_path + '24x24.png', QSize(24,24))
app_icon.addFile(icons_path + '32x32.png', QSize(32,32))
app_icon.addFile(icons_path + '48x48.png', QSize(48,48))
app_icon.addFile(icons_path + '256x256.png', QSize(256,256))
app.setWindowIcon(app_icon)
win = gui.MainWindow(get_external_ip(), options.port)
logger.addHandler(gui.GuiLogHandler(win))
win.show()
sys.exit(app.exec_())