clearml-session/clearml_session/single_thread_proxy.py
2020-12-22 21:32:02 +02:00

127 lines
4.3 KiB
Python

import threading
import socket
import time
import select
import sys
class SingleThreadProxy(object):
max_timeout_for_remote_connection = 60
class Forward(object):
def __init__(self):
self.forward = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
def start(self, host, port):
try:
self.forward.connect((host, port))
return self.forward
except Exception as e:
return False
def __init__(self, port, tgtport, host="127.0.0.1", tgthost="127.0.0.1",
buffer_size=4096, delay=0.0001, state=None):
self.input_list = []
self.channel = {}
self.sidmap = {}
self.state = state or {}
# set max number of open files
# noinspection PyBroadException
try:
if sys.platform == 'win32':
import ctypes
ctypes.windll.msvcrt._setmaxstdio(max(2048, ctypes.windll.msvcrt._getmaxstdio())) # noqa
else:
import resource
soft, hard = resource.getrlimit(resource.RLIMIT_NOFILE)
resource.setrlimit(resource.RLIMIT_NOFILE, (max(4096, soft), hard))
except Exception:
pass
self.server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
self.server.bind((host, port))
self.server.listen(100)
self.tgthost, self.tgtport = tgthost, tgtport
self.buffer_size, self.delay = buffer_size, delay
self._proxy_daemon_thread = threading.Thread(target=self.main_loop)
self._proxy_daemon_thread.setDaemon(True)
self._proxy_daemon_thread.start()
def main_loop(self):
self.input_list.append(self.server)
ss = select.select
while 1:
time.sleep(self.delay)
try:
inputready, outputready, exceptready = ss(self.input_list, [], [])
except:
continue
for self.s in inputready:
if self.s == self.server:
try:
self.on_accept()
except:
pass
break
try:
self.data = self.s.recv(self.buffer_size)
except:
continue
if len(self.data) == 0:
try:
self.on_close()
except:
pass
break
else:
try:
self.on_recv()
except:
pass
def on_accept(self):
clientsock, clientaddr = self.server.accept()
for i in range(self.max_timeout_for_remote_connection):
forward = self.Forward().start(self.tgthost, self.tgtport)
if forward:
break
# print('waiting for remote...')
time.sleep(1)
if forward:
# logger.info("{0} has connected".format(clientaddr))
self.input_list.append(clientsock)
self.input_list.append(forward)
self.channel[clientsock] = forward
self.channel[forward] = clientsock
_sidbase = "{0}_{1}_{2}_{3}".format(self.tgthost, self.tgtport, clientaddr[0], clientaddr[1])
self.sidmap[clientsock] = (_sidbase, 1)
self.sidmap[forward] = (_sidbase, -1)
else:
# logger.warn("Can't establish connection with remote server.\n"
# "Closing connection with client side{0}".format(clientaddr))
clientsock.close()
def on_close(self):
# logger.info("{0} has disconnected".format(self.s.getpeername()))
self.input_list.remove(self.s)
self.input_list.remove(self.channel[self.s])
out = self.channel[self.s]
self.channel[out].close()
self.channel[self.s].close()
del self.channel[out]
del self.channel[self.s]
del self.sidmap[out]
del self.sidmap[self.s]
def on_recv(self):
_sidbase = self.sidmap[self.s][0]
_c_or_s = self.sidmap[self.s][1]
data = self.data
# logger.debug(ctrl_less(data.strip()))
self.channel[self.s].send(data)