-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver.py
176 lines (153 loc) · 6.24 KB
/
server.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
import random
import socket
import threading
from flask import Flask
from flask_sock import Sock
config = {
"servers": [
{
"name": "my-device",
"listen_port": 3033, # Websocket server port (client.py use this port) Ex: client connect to wss://example.com:3033/ws-tunnel
"password": "MySecurePassword",
"ports": {
"4000": { # Local port 4000 redirects to port 22 on the device running client.py.
"redirect_ip": "127.0.0.1",
"redirect_port": 22
},
"4001": { # Local port 4001 redirects to port 8080 on the device running client.py.
"redirect_ip": "127.0.0.1",
"redirect_port": 8080
},
"4002": { # Local port 4003 redirects to port 80 on the "192.168.1.1" host in the network of device running client.py.
"redirect_ip": "192.168.1.1",
"redirect_port": 80
}
}
}
]
}
def recvall(sock):
builded = ""
while not builded.endswith("|"):
builded += sock.receive()
liste = builded.split("|")[:-1]
return [x for x in liste]
def log(type, message, instance=None):
types = {
"INFO": bcolors.OKGREEN,
"WARNING": bcolors.WARNING,
"ERROR": bcolors.FAIL
}
print("[{}] [{}] {}".format(instance, types[type] + type + bcolors.ENDC, message) + bcolors.ENDC)
class bcolors:
HEADER = '\033[95m'
OKBLUE = '\033[94m'
OKCYAN = '\033[96m'
OKGREEN = '\033[92m'
WARNING = '\033[93m'
FAIL = '\033[91m'
ENDC = '\033[0m'
BOLD = '\033[1m'
UNDERLINE = '\033[4m'
class TunnelInstance:
def __init__(self, config: dict):
self.config = config
self.port = config['listen_port']
self.name = config['name']
self.tokens = {}
log("INFO", "Starting encapsulator server on port {}...".format(self.port), self.name)
self.app = Flask(__name__)
self.sock = Sock(self.app)
self.host_connected = False
self.deviceconn = None
@self.sock.route('/ws-tunnel')
def echo(sock):
log("INFO", "Host connected ! Waiting login...", self.name)
login = sock.receive()
if login != self.config['password']:
log("WARNING", "Client fails authentification", self.name)
return
log("INFO", "Client authentificated !", self.name)
self.host_connected = True
self.deviceconn = sock
while True:
try:
list = recvall(sock)
except (ConnectionResetError, BrokenPipeError):
break
for data in list:
if data == "":
continue
data = data
token = data.split(";")[0]
if token == "CLOSE":
if data.split(";")[1] in self.tokens:
self.tokens[data.split(";")[1]].client.close()
del self.tokens[data.split(";")[1]]
elif token in self.tokens:
self.tokens[token].client.send(bytes.fromhex(data.split(";")[1]))
else:
print("Token not found ( {} )".format(token))
log("ERROR", "Host disconnected ! ({})".format("EMPTY"), self.name)
log("WARNING", "Disconnecting all clients...", self.name)
[x.stop() for x in self.tokens.values()]
self.servers = []
log("INFO", "Opening ports...", self.name)
for source, data in config['ports'].items():
try:
self.servers.append(SocketServer(data['redirect_ip'], int(data['redirect_port']), int(source), self))
log("INFO", "Local port {} opened.".format(source), self.name)
except OSError as e:
log("ERROR", "Port {} is already in use.".format(source), self.name)
self.app.run(host='0.0.0.0', port=self.port, threaded=True)
class ClientSocketConnection:
def __init__(self, token, client, instance):
self.client = client
self.token = token
self.closed = False
self.instance = instance
threading.Thread(target=self.recv_thread).start()
def stop(self):
self.closed = True
self.client.close()
def recv_thread(self):
while not self.closed:
try:
data = self.client.recv(1024)
if not data:
self.instance.deviceconn.send(str("CLOSE" + ";" + self.token + "|"))
if self.token in self.instance.tokens:
del self.instance.tokens[self.token]
self.client.close()
break
self.instance.deviceconn.send(str(self.token + ";" + str(data.hex()) + "|"))
except (ConnectionAbortedError, ConnectionResetError, BrokenPipeError):
break
class SocketServer:
def __init__(self, redirect_ip, redirect_port, listen_port, instance):
self.redirect_ip = redirect_ip
self.redirect_port = redirect_port
self.listen_port = listen_port
self.instance = instance
self.server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
self.server.bind(('0.0.0.0', self.listen_port))
self.server.listen(5)
self.stopped = False
threading.Thread(target=self.start).start()
def start(self):
while not self.stopped:
client = self.server.accept()[0]
if not self.instance.host_connected:
client.close()
continue
if self.stopped:
return
i = 0
while i == 0 or str(i) in self.instance.tokens:
i = random.randint(1, 999999999)
self.instance.deviceconn.send("AUTH;{};{};{}|".format(i, self.redirect_ip, self.redirect_port))
cli = ClientSocketConnection(str(i), client, self.instance)
self.instance.tokens[str(i)] = cli
for server in config['servers']:
TunnelInstance(server)