-
Notifications
You must be signed in to change notification settings - Fork 125
/
client.py
499 lines (432 loc) · 21.5 KB
/
client.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
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
#!/usr/bin/env python
import logging
import logging.handlers
import socket
import sys
import time
from struct import pack, unpack
import select
import optparse
import errno
import relay
import threading
from ntlm_auth.ntlm import Ntlm
import re
logger = None
def key_by_value(my_dict, value):
for k, v in my_dict.iteritems():
if v == value:
return k
return None
class SocksRelay:
STATUS_SUCCESS = 0
STATUS_REFUSED = 1
STATUS_TIMEOUT = 2
def __init__(self, bc_sock):
self.channel = {}
self.id_by_socket = {}
self.bc_sock = bc_sock
self.input_list = [self.bc_sock]
self.establishing_dict = {}
self.forward_socket = None
self.data = None
self.last_ping_time = time.time()
logger.debug('Starting ping thread')
self.ping_thread = threading.Thread(target=self.ping_worker)
self.ping_thread.start()
self.remote_side_down = False
def ping_worker(self):
while True:
time.sleep(10)
current_time = time.time()
logger.debug('In ping worker')
if self.remote_side_down:
logger.debug('Remote side down. Exiting ping worker')
return
if current_time - self.last_ping_time > relay.relay_timeout:
logger.info('No response from remote side for {0} seconds. Restarting relay'.format(relay.relay_timeout))
self.bc_sock.close()
return
def shutdown(self):
self.remote_side_down = True
relay.close_sockets(self.input_list)
sys.exit(1)
def run(self):
inputready = None
outputready = None
exceptready = None
while True:
try:
time.sleep(relay.delay)
logger.debug("Active channels: {0}. Pending Channels {1}".format(self.channel.keys(), self.establishing_dict.values()))
inputready, outputready, exceptready = select.select(self.input_list, self.establishing_dict.keys(), [], 15)
except KeyboardInterrupt:
logger.info('SIGINT received. Closing relay and exiting')
self.send_remote_cmd(self.bc_sock, relay.CLOSE_RELAY)
self.shutdown()
except select.error as (code, msg):
logger.debug('Select error on select. Errno: {0} Msg: {1}'.format(errno.errorcode[code], msg))
self.shutdown()
except socket.error as (code, msg):
logger.debug('Socket error on select. Errno: {0} Msg: {1}'.format(errno.errorcode[code], msg))
self.shutdown()
for sock in outputready:
channel_id = self.establishing_dict[sock]
logger.debug('Establishing connection with channel id {0}'.format(channel_id))
try:
sock.recv(0)
except socket.error as (code, err_msg):
if code == errno.ECONNREFUSED or code == errno.ETIMEDOUT:
logger.debug('Connection {0}'.format(errno.errorcode[code]))
if sock in inputready:
inputready.remove(sock)
del self.establishing_dict[sock]
self.send_remote_cmd(self.bc_sock, relay.FORWARD_CONNECTION_FAILURE, channel_id)
sock.close()
continue
elif code == errno.EAGAIN:
logger.debug('Recv(0) return errno.EAGAIN for socket {0} on channel {1}. Connection established.'.format(sock, channel_id))
elif code == 10035:
logger.debug('Recv(0) raised windows-specific exception 10035. Connection established.')
else:
raise
logger.debug('Connection established on channel {0}'.format(channel_id))
sock.setblocking(1)
self.send_remote_cmd(self.bc_sock, relay.FORWARD_CONNECTION_SUCCESS, self.establishing_dict[sock])
del self.establishing_dict[sock]
self.input_list.append(sock)
self.set_channel(sock, channel_id)
for self.selected_input_socket in inputready:
if self.selected_input_socket == self.bc_sock:
try:
self.manage_remote_socket(self.bc_sock)
except relay.RelayError:
logger.debug('Remote side closed socket')
relay.close_sockets(self.input_list)
return
else:
self.manage_forward_socket(self.selected_input_socket)
def handle_remote_cmd(self, data):
cmd = data[0]
logger.debug('Received cmd data from remote side. Cmd: {0}'.format(relay.cmd_names[cmd]))
if cmd == relay.CHANNEL_CLOSE_CMD:
channel_id = unpack('<H', data[1:3])[0]
logger.debug('Channel close request with id: {0}'.format(channel_id))
establishing_sock = key_by_value(self.establishing_dict, channel_id)
if establishing_sock is not None:
logger.debug('Closing establishing socket with id: {0}'.format(channel_id))
del self.establishing_dict[establishing_sock]
elif channel_id not in self.channel:
logger.debug('Channel {0} non existent'.format(channel_id))
return
else:
sock_to_close = self.channel[channel_id]
self.unset_channel(channel_id)
logger.debug('Closing socket with id: {0}'.format(channel_id))
sock_to_close.close()
self.input_list.remove(sock_to_close)
elif cmd == relay.CHANNEL_OPEN_CMD:
channel_id, packed_ip, port = unpack('<HIH', data[1:9])
ip = socket.inet_ntoa(data[3:7])
logger.debug('Got new channel request with id {0} . Opening new forward connection to host {1} port {2}'.format(channel_id, ip, port))
self.establish_forward_socket(channel_id, ip, port)
elif cmd == relay.CLOSE_RELAY:
logger.info('Got command to close relay. Closing socket and exiting.')
self.shutdown()
elif cmd == relay.PING_CMD:
self.last_ping_time = time.time()
self.send_remote_cmd(self.bc_sock, relay.PING_CMD)
else:
logger.debug('Received unknown cmd: {0}'.format(cmd))
def get_channel_data(self, sock):
try:
tlv_header = relay.recvall(sock, 4)
channel_id, tlv_data_len = unpack('<HH', tlv_header)
data = relay.recvall(sock, tlv_data_len)
except socket.error as (code, msg):
logger.debug('Exception on receiving tlv message from remote side. Exiting')
logger.debug('Errno: {0} Msg: {1}'.format(errno.errorcode[code], msg))
raise relay.RelayError
return channel_id, data
def manage_remote_socket(self, sock):
try:
(channel_id, data) = self.get_channel_data(sock)
except relay.RelayError:
logger.debug('Exiting!')
self.close_remote_connection(sock)
raise relay.RelayError
if channel_id == relay.COMMAND_CHANNEL:
self.handle_remote_cmd(data)
elif channel_id in self.channel:
relay_to_sock = self.channel[channel_id]
logger.debug('Got data to relay from remote side. Channel id {0}. Data length: {1}'.format(channel_id, len(data)))
logger.debug('Data contents: {0}'.format(data.encode('hex')))
self.relay(data, relay_to_sock)
else:
logger.debug('Relay from socket {0} with channel {1} not possible. Channel does not exist'.format(sock, channel_id))
return
def close_remote_connection(self, sock):
sock.close()
self.input_list.remove(sock)
def manage_forward_socket(self, sock):
if sock not in self.id_by_socket:
logger.debug('Channel corresponding to remote socket {0} already closed. Closing forward socket'.format(sock))
return
channel_id = self.id_by_socket[sock]
#logger.debug('Readable socket {0} with channel id {1}'.format(sock, channel_id))
try:
data = sock.recv(relay.buffer_size)
except socket.error as (code, msg):
logger.debug('Exception on receiving data from socket {0} with channel id {1}'.format(sock, channel_id))
logger.debug('Errno: {0} Msg: {1}'.format(errno.errorcode[code], msg))
logger.debug('Closing socket {0} with channel id {1}'.format(sock, channel_id))
self.close_forward_connection(sock)
return
data_len = len(data)
if data_len == 0:
self.close_forward_connection(sock)
return
else:
channel_id = self.id_by_socket[sock]
tlv_header = pack('<HH', channel_id, len(data))
logger.debug('Got data to relay from app side. Channel id {0}. Data length: {1}'.format(channel_id, len(data)))
logger.debug('Preparing tlv header: {0}'.format(tlv_header.encode('hex')))
logger.debug('Data contents: {0}'.format(data.encode('hex')))
self.relay(tlv_header + data, self.bc_sock)
def close_forward_connection(self, sock):
channel_id = self.id_by_socket[sock]
logger.debug('Closing forward socket {0} with id {1}'.format(sock, channel_id))
logger.debug('Current remote side socket: {0}'.format(self.bc_sock))
logger.debug('Notifying remote side')
self.unset_channel(channel_id)
self.input_list.remove(sock)
sock.close()
self.send_remote_cmd(self.bc_sock, relay.CHANNEL_CLOSE_CMD, channel_id)
def send_remote_cmd(self, sock, cmd, *args):
logger.debug('Sending cmd to remote side. Cmd: {0}'.format(relay.cmd_names[cmd]))
if cmd == relay.CHANNEL_CLOSE_CMD:
cmd_buffer = cmd + pack('<H', args[0])
tlv_header = pack('<HH', relay.COMMAND_CHANNEL, len(cmd_buffer))
elif cmd == relay.FORWARD_CONNECTION_SUCCESS:
cmd_buffer = cmd + pack('<H', args[0])
tlv_header = pack('<HH', relay.COMMAND_CHANNEL, len(cmd_buffer))
elif cmd == relay.FORWARD_CONNECTION_FAILURE:
cmd_buffer = cmd + pack('<H', args[0])
tlv_header = pack('<HH', relay.COMMAND_CHANNEL, len(cmd_buffer))
else:
cmd_buffer = cmd
tlv_header = pack('<HH', relay.COMMAND_CHANNEL, len(cmd_buffer))
try:
sock.send(tlv_header + cmd_buffer)
except socket.error as (code, cmd):
logger.error('Socket error on sending command to remote side. Code {0}. Msg {1}'.format(code, cmd))
def set_channel(self, sock, channel_id):
self.channel[channel_id] = sock
self.id_by_socket[sock] = channel_id
def unset_channel(self, channel_id):
sock = self.channel[channel_id]
del self.id_by_socket[sock]
del self.channel[channel_id]
def establish_forward_socket(self, channel_id, host, port):
try:
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setblocking(0)
sock.connect_ex((host, port))
except socket.error as (code, msg):
logger.debug("Caught exception socket.error during establishing forward connection. Code {0}. Msg {1}".format(code, msg))
self.send_remote_cmd(self.bc_sock, relay.FORWARD_CONNECTION_FAILURE, channel_id)
return
logger.debug('Adding new pending forward connection with channel id {0} and socket {1}'.format(channel_id, sock))
self.establishing_dict[sock] = channel_id
def relay(self, data, to_socket):
if to_socket is None:
return
try:
to_socket.send(data)
except socket.error as (code, msg):
logger.debug('Exception on relaying data to socket {0}'.format(to_socket))
logger.debug('Errno: {0} Msg: {1}'.format(errno.errorcode[code], msg))
if to_socket == self.bc_sock:
raise relay.RelayError
else:
logger.debug('Closing socket')
to_socket.close()
self.input_list.remove(to_socket)
channel_id = self.id_by_socket[to_socket]
self.unset_channel(channel_id)
self.send_remote_cmd(self.socket_with_server, relay.CHANNEL_CLOSE_CMD, channel_id)
class NtlmProxyContext(object):
negotiate_request = '''CONNECT {0}:{1} HTTP/1.1
User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:45.0) Gecko/20100101 Firefox/45.0
Proxy-Connection: keep-alive
Connection: keep-alive
Proxy-Authorization: NTLM {2}
'''
authenticate_request = '''CONNECT {0}:{1} HTTP/1.1
User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:45.0) Gecko/20100101 Firefox/45.0
Proxy-Connection: keep-alive
Connection: keep-alive
Proxy-Authorization: NTLM {2}
'''
def __init__(self, sock, proxy_ip, proxy_port, username, domain=None, password=None, nthash=None, lmhash=None):
self._sock = sock
self._proxy_ip = proxy_ip
self._proxy_port = proxy_port
self._username = username
self._password = password
self._nthash = nthash
self._lmhash = lmhash
self._domain = domain
self._workstation = socket.gethostname().upper()
def connect(self, host_port):
(host, port) = host_port
ntlm_context = Ntlm(ntlm_compatibility=5)
negotiate_message = ntlm_context.create_negotiate_message(self._domain, self._workstation).decode()
resp = None
try:
self._sock.connect((self._proxy_ip, self._proxy_port))
self._sock.send(NtlmProxyContext.negotiate_request.format(host, str(port), negotiate_message))
resp = self._sock.recv(4096)
except socket.error as (code, msg):
logger.error("Caught socket error trying to establish connection to proxy. Code {0}. Msg {1}".format(code, msg))
raise
try:
chal_msg = NtlmProxyContext.get_challenge(resp)
ntlm_context.parse_challenge_message(chal_msg)
except TypeError:
logger.error("Couldn't parse proxy challenge. Code {0}. Msg {1}".format(code, msg))
if resp is not None:
logger.error("Challenge contents: {0}".format(resp))
else:
logger.error("Challenge contents is 'None'")
self._sock.close()
authenticate_message = ntlm_context.create_authenticate_message(user_name=self._username,
domain_name=self._domain,
password=self._password,
nthash=self._nthash,
lmhash=self._lmhash).decode()
resp = None
try:
self._sock.send(NtlmProxyContext.authenticate_request.format(host, str(port), authenticate_message))
resp = self._sock.recv(4096)
except socket.error as (code, msg):
logger.error("Caught socket error trying to send challenge response connection to proxy. Code {0}. Msg {1}".format(code, msg))
self._sock.close()
raise
if resp is None:
logger.error("Received an empty response to the challenge response")
self._sock.close()
if 'HTTP/1.1 200 Connection established' in resp:
logger.info('Ntlm proxy established connection')
logger.debug(resp)
elif 'HTTP/1.1 503 Service Unavailable' in resp:
logger.error('Ntlm proxy response: Service Unavailable')
logger.debug(resp)
self._sock.close()
elif 'HTTP/1.1 407 Proxy Authentication Required' in resp:
logger.error('Ntlm proxy authentication failed')
logger.debug(resp)
self._sock.close()
sys.exit(1)
else:
logger.error('Ntlm proxy unknown error')
logger.debug(resp)
self._sock.close()
def __getattr__(self, attribute_name):
"""Defer unknown behaviour to the socket"""
return getattr(self._sock, attribute_name)
@staticmethod
def get_challenge(raw_msg):
if raw_msg is None:
return None
re_res = re.search(r'^Proxy-Authenticate: NTLM (.*)$', raw_msg, re.MULTILINE)
if re_res is None:
return None
else:
return re_res.group(1)
def main():
global logger
parser = optparse.OptionParser(description='Reverse socks client')
parser.add_option('--server-ip', action="store", dest='server_ip')
parser.add_option('--server-port', action="store", dest='server_port', default='9999')
parser.add_option('--verbose', action="store_true", dest="verbose", default=False)
parser.add_option('--logfile', action="store", dest="logfile", default=None)
proxy_group = optparse.OptionGroup(parser, 'Ntlm Proxy authentication')
proxy_group.add_option('--ntlm-proxy-ip', dest='ntlm_proxy_ip', default=None, action='store', help='IP address of NTLM proxy')
proxy_group.add_option('--ntlm-proxy-port', dest='ntlm_proxy_port', default=None, action='store', help='Port of NTLM proxy')
proxy_group.add_option('--username', dest='username', default='', action='store', help='Username to authenticate with NTLM proxy')
proxy_group.add_option('--domain', dest='domain', default='', action='store', help='Domain to authenticate with NTLM proxy')
proxy_group.add_option('--password', dest='password', default='', action='store', help='Password to authenticate with NTLM proxy')
proxy_group.add_option('--hashes', dest='hashes', default=None, action='store', help='Hashes to authenticate with instead of password. Format - LMHASH:NTHASH')
parser.add_option_group(proxy_group)
cmd_options = parser.parse_args()[0]
if cmd_options.server_ip is None:
print 'Server IP required'
sys.exit()
logger = logging.getLogger('root')
logger.setLevel(logging.DEBUG)
if cmd_options.logfile is None:
ch = logging.StreamHandler()
else:
ch = logging.FileHandler(cmd_options.logfile)
if cmd_options.verbose:
ch.setLevel(logging.DEBUG)
else:
ch.setLevel(logging.INFO)
logger.addHandler(ch)
logger.addHandler(ch)
while True:
logger.info('Backconnecting to server {0} port {1}'.format(cmd_options.server_ip, cmd_options.server_port))
backconnect_host = cmd_options.server_ip
backconnect_port = int(cmd_options.server_port)
bc_sock = None
while True:
try:
bc_sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
if cmd_options.ntlm_proxy_ip is not None:
if cmd_options.ntlm_proxy_port is None:
logger.error('Error. Must specify ntlm proxy port')
sys.exit(1)
if cmd_options.hashes is not None:
if re.match('[a-zA-Z0-9]{32}:[a-zA-Z0-9]{32}', cmd_options.hashes) is None:
logger.error('Hash format error. Valid hash format - LMHASH:NTHASH')
sys.exit(1)
logger.info('Connecting via NTLM proxy at {0}:{1}'.format(cmd_options.ntlm_proxy_ip, cmd_options.ntlm_proxy_port))
ntlm_con = NtlmProxyContext(bc_sock, proxy_ip=cmd_options.ntlm_proxy_ip,
proxy_port=int(cmd_options.ntlm_proxy_port),
username=cmd_options.username,
domain=cmd_options.domain,
password=cmd_options.password,
nthash=None if cmd_options.hashes is None else cmd_options.hashes.split(':')[1],
lmhash=None if cmd_options.hashes is None else cmd_options.hashes.split(':')[0])
bc_sock = ntlm_con
bc_sock.connect((backconnect_host, backconnect_port))
break
except socket.error as (code, msg):
logger.info('Unable to connect to {0} port: {1}. Caught socket error trying to establish connection with RPIVOT server. Code {2}. Msg {3}'.format(cmd_options.server_ip, cmd_options.server_port, code, msg))
logger.info('Retrying')
time.sleep(5)
try:
bc_sock.send(relay.banner)
banner_reponse_rcv = bc_sock.recv(4096)
if banner_reponse_rcv != relay.banner_response:
logger.error("Wrong banner response {0} from server. Retrying".format(repr(banner_reponse_rcv)))
bc_sock.close()
time.sleep(5)
continue
except socket.error as (code, msg):
logger.error("Caught socket error trying to establish connection with RPIVOT server. Code {0}. Msg {1}".format(code, msg))
bc_sock.close()
time.sleep(5)
continue
socks_relayer = SocksRelay(bc_sock)
try:
socks_relayer.run()
except socket.error as (code, msg):
logger.debug('Exception in socks_relayer.run(). Restarting relay')
logger.debug('Errno: {0} Msg: {1}'.format(errno.errorcode[code], msg))
bc_sock.close()
continue
time.sleep(10)
if __name__ == '__main__':
main()