add ssloop branch
This commit is contained in:
parent
5ad8cd6f3d
commit
4f46400145
4 changed files with 177 additions and 20 deletions
|
@ -25,6 +25,7 @@
|
|||
# https://github.com/clowwindy/ssloop
|
||||
|
||||
|
||||
import os
|
||||
import select
|
||||
from collections import defaultdict
|
||||
|
||||
|
@ -187,3 +188,11 @@ def errno_from_exception(e):
|
|||
return e.args[0]
|
||||
else:
|
||||
return None
|
||||
|
||||
|
||||
# from tornado
|
||||
def get_sock_error(sock):
|
||||
errno = sock.getsockopt(socket.SOL_SOCKET,
|
||||
socket.SO_ERROR)
|
||||
return socket.error(errno, os.strerror(errno))
|
||||
|
||||
|
|
|
@ -29,16 +29,6 @@ else:
|
|||
import json
|
||||
|
||||
|
||||
# TODO remove gevent
|
||||
try:
|
||||
import gevent
|
||||
import gevent.monkey
|
||||
gevent.monkey.patch_all(dns=gevent.version_info[0] >= 1)
|
||||
except ImportError:
|
||||
gevent = None
|
||||
print >>sys.stderr, 'warning: gevent not found, using threading instead'
|
||||
|
||||
|
||||
import socket
|
||||
import select
|
||||
import threading
|
||||
|
|
154
shadowsocks/tcprelay.py
Normal file
154
shadowsocks/tcprelay.py
Normal file
|
@ -0,0 +1,154 @@
|
|||
#!/usr/bin/python
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright (c) 2014 clowwindy
|
||||
#
|
||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
# of this software and associated documentation files (the "Software"), to deal
|
||||
# in the Software without restriction, including without limitation the rights
|
||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
# copies of the Software, and to permit persons to whom the Software is
|
||||
# furnished to do so, subject to the following conditions:
|
||||
#
|
||||
# The above copyright notice and this permission notice shall be included in
|
||||
# all copies or substantial portions of the Software.
|
||||
#
|
||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
# SOFTWARE.
|
||||
|
||||
import time
|
||||
import threading
|
||||
import socket
|
||||
import logging
|
||||
import struct
|
||||
import encrypt
|
||||
import eventloop
|
||||
import errno
|
||||
|
||||
|
||||
class TCPRelayHandler(object):
|
||||
def __init__(self, fd_to_handlers, loop, conn, config, is_local):
|
||||
self._fd_to_handlers = fd_to_handlers
|
||||
self._loop = loop
|
||||
self._local_conn = conn
|
||||
self._remote_conn = None
|
||||
self._remains_data_for_local = None
|
||||
self._remains_data_for_remote = None
|
||||
self._config = config
|
||||
self._is_local = is_local
|
||||
self._stage = 0
|
||||
fd_to_handlers[conn.fileno()] = self
|
||||
conn.setblocking(False)
|
||||
loop.add(conn, eventloop.POLL_IN)
|
||||
|
||||
def on_local_read(self):
|
||||
pass
|
||||
|
||||
def on_remote_read(self):
|
||||
pass
|
||||
|
||||
def on_local_write(self):
|
||||
pass
|
||||
|
||||
def on_remote_write(self):
|
||||
pass
|
||||
|
||||
def on_local_error(self):
|
||||
self.destroy()
|
||||
|
||||
def on_remote_error(self):
|
||||
self.destroy()
|
||||
|
||||
def handle_event(self, sock, event):
|
||||
# order is important
|
||||
if sock == self._local_conn:
|
||||
if event & eventloop.POLL_IN:
|
||||
self.on_local_read()
|
||||
if event & eventloop.POLL_OUT:
|
||||
self.on_local_write()
|
||||
if event & eventloop.POLL_ERR:
|
||||
self.on_local_error()
|
||||
elif sock == self._remote_conn:
|
||||
if event & eventloop.POLL_IN:
|
||||
self.on_remote_read()
|
||||
if event & eventloop.POLL_OUT:
|
||||
self.on_remote_write()
|
||||
if event & eventloop.POLL_ERR:
|
||||
self.on_remote_error()
|
||||
else:
|
||||
logging.warn('unknown socket')
|
||||
|
||||
def destroy(self):
|
||||
if self._local_conn:
|
||||
self._local_conn.close()
|
||||
eventloop.remove(self._local_conn)
|
||||
# TODO maybe better to delete the key
|
||||
self._fd_to_handlers[self._local_conn.fileno()] = None
|
||||
if self._remote_conn:
|
||||
self._remote_conn.close()
|
||||
eventloop.remove(self._remote_conn)
|
||||
self._fd_to_handlers[self._local_conn.fileno()] = None
|
||||
|
||||
|
||||
class TCPRelay(object):
|
||||
def __init__(self, config, is_local):
|
||||
self._config = config
|
||||
self._is_local = is_local
|
||||
self._closed = False
|
||||
self._fd_to_handlers = {}
|
||||
|
||||
addrs = socket.getaddrinfo(self._listen_addr, self._listen_port, 0,
|
||||
socket.SOCK_STREAM, socket.SOL_TCP)
|
||||
if len(addrs) == 0:
|
||||
raise Exception("can't get addrinfo for %s:%d" %
|
||||
(self._listen_addr, self._listen_port))
|
||||
af, socktype, proto, canonname, sa = addrs[0]
|
||||
server_socket = socket.socket(af, socktype, proto)
|
||||
server_socket.bind((self._listen_addr, self._listen_port))
|
||||
server_socket.setblocking(False)
|
||||
self._server_socket = server_socket
|
||||
|
||||
def _run(self):
|
||||
server_socket = self._server_socket
|
||||
self._eventloop = eventloop.EventLoop()
|
||||
self._eventloop.add(server_socket, eventloop.POLL_IN)
|
||||
last_time = time.time()
|
||||
while not self._closed:
|
||||
try:
|
||||
events = self._eventloop.poll(1)
|
||||
except (OSError, IOError) as e:
|
||||
if eventloop.errno_from_exception(e) == errno.EPIPE:
|
||||
# Happens when the client closes the connection
|
||||
continue
|
||||
else:
|
||||
logging.error(e)
|
||||
continue
|
||||
for sock, event in events:
|
||||
if sock == self._server_socket:
|
||||
try:
|
||||
conn = self._server_socket.accept()
|
||||
TCPRelayHandler(loop, conn, remote_addr, remote_port,
|
||||
password, method, timeout, is_local)
|
||||
except (OSError, IOError) as e:
|
||||
error_no = eventloop.errno_from_exception(e)
|
||||
if error_no in [errno.EAGAIN, errno.EINPROGRESS]:
|
||||
continue
|
||||
else:
|
||||
handler = self._fd_to_handlers.get(sock.fileno(), None)
|
||||
if handler:
|
||||
handler.handle_event(sock, event)
|
||||
else:
|
||||
logging.warn('can not find handler for fd %d',
|
||||
sock.fileno())
|
||||
now = time.time()
|
||||
if now - last_time > 5:
|
||||
# TODO sweep timeouts
|
||||
last_time = now
|
||||
|
||||
|
||||
|
|
@ -123,16 +123,20 @@ def client_key(a, b, c, d):
|
|||
|
||||
|
||||
class UDPRelay(object):
|
||||
def __init__(self, listen_addr='127.0.0.1', listen_port=1080,
|
||||
remote_addr='127.0.0.1', remote_port=8387, password=None,
|
||||
method='table', timeout=300, is_local=True):
|
||||
self._listen_addr = listen_addr
|
||||
self._listen_port = listen_port
|
||||
self._remote_addr = remote_addr
|
||||
self._remote_port = remote_port
|
||||
self._password = password
|
||||
self._method = method
|
||||
self._timeout = timeout
|
||||
def __init__(self, config, is_local=True):
|
||||
if is_local:
|
||||
self._listen_addr = config['local_address']
|
||||
self._listen_port = config['local_port']
|
||||
self._remote_addr = config['server']
|
||||
self._remote_port = config['server_port']
|
||||
else:
|
||||
self._listen_addr = config['server']
|
||||
self._listen_port = config['server_port']
|
||||
self._remote_addr = None
|
||||
self._remote_port = None
|
||||
self._password = config['password']
|
||||
self._method = config['method']
|
||||
self._timeout = config['timeout']
|
||||
self._is_local = is_local
|
||||
self._cache = lru_cache.LRUCache(timeout=timeout,
|
||||
close_callback=self._close_client)
|
||||
|
|
Loading…
Reference in a new issue