forked from knownsec/rtcp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrtcp.py
executable file
·166 lines (137 loc) · 4.37 KB
/
rtcp.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
#!/usr/bin/env python
# coding=utf-8
"""
filename: rtcp.py
@desc:
利用 Python 的 socket 端口转发,用于远程维护
如果连接不到远程,会 sleep 36s,最多尝试 200(即两小时)
@usage:
./rtcp.py stream1 stream2
stream 为:l:port 或 c:host:port
l:port 表示监听指定的本地端口
c:host:port 表示监听远程指定的端口
@author: watercloud, zd, knownsec team
@web: www.knownsec.com, blog.knownsec.com
@date: 2009-7
"""
import socket
import sys
import threading
import time
streams = [None, None] # 存放需要进行数据转发的两个数据流(都是 SocketObj 对象)
debug = 1 # 调试状态 0 or 1
def _usage():
print('Usage: ./rtcp.py stream1 stream2\nstream: l:port or c:host:port')
def _get_another_stream(num):
"""
从streams获取另外一个流对象,如果当前为空,则等待
"""
if num == 0:
num = 1
elif num == 1:
num = 0
else:
raise 'ERROR'
while True:
if streams[num] == 'quit':
print('can not connect to the target, quit now!')
sys.exit(1)
if streams[num] is not None:
return streams[num]
else:
time.sleep(1)
def _xstream(num, s1, s2):
"""
交换两个流的数据
num为当前流编号,主要用于调试目的,区分两个回路状态用。
"""
try:
while True:
# 注意,recv 函数会阻塞,直到对端完全关闭(close 后还需要一定时间才能关闭,最快关闭方法是 shutdow)
buff = s1.recv(1024)
if debug > 0:
print('%d recv' % num)
if len(buff) == 0: # 对端关闭连接,读不到数据
print('%d one closed' % num)
break
s2.sendall(buff)
if debug > 0:
print('%d sendall' % num)
except:
print('%d one connect closed.' % num)
try:
s1.shutdown(socket.SHUT_RDWR)
s1.close()
except:
pass
try:
s2.shutdown(socket.SHUT_RDWR)
s2.close()
except:
pass
streams[0] = None
streams[1] = None
print('%d CLOSED' % num)
def _server(port, num):
"""
处理服务情况,num 为流编号(第 0 号还是第 1 号)
"""
srv = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
srv.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
srv.bind(('0.0.0.0', port))
srv.listen(1)
while True:
conn, addr = srv.accept()
print('connected from: %s' % str(addr))
streams[num] = conn # 放入本端流对象
s2 = _get_another_stream(num) # 获取另一端流对象
_xstream(num, conn, s2)
def _connect(host, port, num):
"""处理连接,num 为流编号(第 0 号还是第 1 号)
@note: 如果连接不到远程,会 sleep 36s,最多尝试 200(即两小时)
"""
not_connet_time = 0
wait_time = 36
try_cnt = 199
while True:
if not_connet_time > try_cnt:
streams[num] = 'quit'
print('not connected')
return None
conn = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
try:
conn.connect((host, port))
except Exception:
print('can not connect %s:%s!' % (host, port))
not_connet_time += 1
time.sleep(wait_time)
continue
print('connected to %s:%i' % (host, port))
streams[num] = conn # 放入本端流对象
s2 = _get_another_stream(num) # 获取另一端流对象
_xstream(num, conn, s2)
def main():
if len(sys.argv) != 3:
_usage()
sys.exit(1)
tlist = [] # 线程列表,最终存放两个线程对象
targv = [sys.argv[1], sys.argv[2]]
for i in [0, 1]:
s = targv[i] # stream 描述 c:ip:port 或 l:port
sl = s.split(':')
if len(sl) == 2 and (sl[0] == 'l' or sl[0] == 'L'): # l:port
t = threading.Thread(target=_server, args=(int(sl[1]), i))
tlist.append(t)
elif len(sl) == 3 and (sl[0] == 'c' or sl[0] == 'C'): # c:host:port
t = threading.Thread(target=_connect, args=(sl[1], int(sl[2]), i))
tlist.append(t)
else:
_usage()
sys.exit(1)
for t in tlist:
t.start()
for t in tlist:
t.join()
sys.exit(0)
if __name__ == '__main__':
main()