ss的local端和server端的工作流程相似,因此复用了TCPRelay类和TCPRelayHandler类。

两端均是使用TCPRelay类监听连接,并使用TCPRelayHandler类处理请求。

以server端为例:

  1. # server.py
  2. ...
  3. tcp_servers = []
  4. tcp_servers.append(tcprelay.TCPRelay(a_config, dns_resolver, False))
  5. ...
  6. class
  7. def run_server():
  8. ...
  9. try:
  10. loop = eventloop.EventLoop()
  11. dns_resolver.add_to_loop(loop)
  12. list(map(lambda s: s.add_to_loop(loop), tcp_servers + udp_servers))
  13. daemon.set_user(config.get('user', None))
  14. loop.run()
  15. except Exception as e:
  16. shell.print_exception(e)
  17. sys.exit(1)

这里创建了一个TCPRelay对象以及EventLoop,并将所有tcp_server加入eventloop中。

在TCPRelay的构造函数中,打开了一个监听套接字,监听1024端口。

  1. # tcpreply.py
  2. class TCPRelay:
  3. def __init__(self, config, dns_resolver, is_local, stat_callback=None):
  4. self._config = config
  5. # 该标志用来区分该对象是客户端还是服务端
  6. self._is_local = is_local
  7. self._dns_resolver = dns_resolver
  8. self._closed = False
  9. self._eventloop = None
  10. # 字典存放的为(fd,handle) 键值对,用来保存每个fd对应的handle函数
  11. self._fd_to_handlers = {}
  12. self._timeout = config['timeout']
  13. self._timeouts = [] # a list for all the handlers
  14. # we trim the timeouts once a while
  15. self._timeout_offset = 0 # last checked position for timeout
  16. self._handler_to_timeouts = {} # key: handler value: index in timeouts
  17. if is_local:
  18. listen_addr = config['local_address']
  19. listen_port = config['local_port']
  20. else:
  21. listen_addr = config['server']
  22. listen_port = config['server_port']
  23. self._listen_port = listen_port
  24. addrs = socket.getaddrinfo(listen_addr, listen_port, 0,
  25. socket.SOCK_STREAM, socket.SOL_TCP)
  26. if len(addrs) == 0:
  27. raise Exception("can't get addrinfo for %s:%d" %
  28. (listen_addr, listen_port))
  29. af, socktype, proto, canonname, sa = addrs[0]
  30. server_socket = socket.socket(af, socktype, proto)
  31. server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
  32. server_socket.bind(sa)
  33. server_socket.setblocking(False)
  34. if config['fast_open']:
  35. try:
  36. server_socket.setsockopt(socket.SOL_TCP, 23, 5)
  37. except socket.error:
  38. logging.error('warning: fast open is not available')
  39. self._config['fast_open'] = False
  40. server_socket.listen(1024)
  41. self._server_socket = server_socket
  42. self._stat_callback = stat_callback

然后执行add_to_loop()将自己加入到eventloop中。

  1. # tcprelay.py
  2. class TCPRelay:
  3. def add_to_loop(self, loop):
  4. if self._eventloop:
  5. raise Exception('already add to loop')
  6. if self._closed:
  7. raise Exception('already closed')
  8. self._eventloop = loop
  9. # 将事件加入到loop,监听IN和ERR
  10. self._eventloop.add(self._server_socket,
  11. eventloop.POLL_IN | eventloop.POLL_ERR, self)
  12. self._eventloop.add_periodic(self.handle_periodic)

这里实质上是调用了eventloop的add

  1. # eventloop.py
  2. class EventLoop:
  3. def add(self, f, mode, handler):
  4. fd = f.fileno()
  5. self._fdmap[fd] = (f, handler)
  6. self._impl.register(fd, mode)

注意这里是将TCPRelay这个类自身作为handler放入_fdmap字典中。这个字典的作用主要是存储每个描述符对应的handler,当事件发生时,从中取出对应的handler对事件进行处理。

接着执行loop.run()开始事件循环。

  1. # eventloop.py
  2. class EventLoop:
  3. def run(self):
  4. # 事件集合
  5. events = []
  6. while not self._stopping:
  7. asap = False
  8. try:
  9. # 返回所有事件
  10. events = self.poll(TIMEOUT_PRECISION)
  11. except (OSError, IOError) as e:
  12. if errno_from_exception(e) in (errno.EPIPE, errno.EINTR):
  13. # EPIPE: Happens when the client closes the connection
  14. # EINTR: Happens when received a signal
  15. # handles them as soon as possible
  16. asap = True
  17. logging.debug('poll:%s', e)
  18. else:
  19. logging.error('poll:%s', e)
  20. import traceback
  21. traceback.print_exc()
  22. continue
  23. for sock, fd, event in events:
  24. # 对于监听到的事件,获取其对应的handle
  25. handler = self._fdmap.get(fd, None)
  26. if handler is not None:
  27. # handler[0]为socket
  28. handler = handler[1]
  29. try:
  30. # 使用对应的handle处理这个事件
  31. handler.handle_event(sock, fd, event)
  32. except (OSError, IOError) as e:
  33. shell.print_exception(e)
  34. now = time.time()
  35. if asap or now - self._last_time >= TIMEOUT_PRECISION:
  36. for callback in self._periodic_callbacks:
  37. callback()
  38. self._last_time = now

事件发生时,首先根据fd获取其对应的handler,并调用handler.handle_event()

  1. # tcprelay.py
  2. class TCPRelay:
  3. def handle_event(self, sock, fd, event):
  4. # handle events and dispatch to handlers
  5. if sock:
  6. logging.log(shell.VERBOSE_LEVEL, 'fd %d %s', fd,
  7. eventloop.EVENT_NAMES.get(event, event))
  8. # 如果为监听套接字
  9. if sock == self._server_socket:
  10. if event & eventloop.POLL_ERR:
  11. # TODO
  12. raise Exception('server_socket error')
  13. try:
  14. logging.debug('accept')
  15. # 尝试接受客户端的连接
  16. conn = self._server_socket.accept()
  17. TCPRelayHandler(self, self._fd_to_handlers,
  18. self._eventloop, conn[0], self._config,
  19. self._dns_resolver, self._is_local)
  20. except (OSError, IOError) as e:
  21. error_no = eventloop.errno_from_exception(e)
  22. if error_no in (errno.EAGAIN, errno.EINPROGRESS,
  23. errno.EWOULDBLOCK):
  24. return
  25. else:
  26. shell.print_exception(e)
  27. if self._config['verbose']:
  28. traceback.print_exc()
  29. # 不是监听套接字,说明是已连接的客户端有可读操作
  30. else:
  31. if sock:
  32. handler = self._fd_to_handlers.get(fd, None)
  33. if handler:
  34. handler.handle_event(sock, event)
  35. else:
  36. logging.warn('poll removed fd')

如果是监听套接字的事件,则说明有新的客户端连接。先通过_fdmap.get()来获取handler(这里为TCPRelay对象) 接着accept,然后创建了一个TCPRelayHandler对象。看一下构造函数:

  1. # tcprelay.py
  2. class TCPRelayHandler(object):
  3. def __init__(self, server, fd_to_handlers, loop, local_sock, config,
  4. dns_resolver, is_local):
  5. self._server = server
  6. self._fd_to_handlers = fd_to_handlers
  7. self._loop = loop
  8. self._local_sock = local_sock
  9. self._remote_sock = None
  10. self._config = config
  11. self._dns_resolver = dns_resolver
  12. # TCP Relay works as either sslocal or ssserver
  13. # if is_local, this is sslocal
  14. self._is_local = is_local
  15. self._stage = STAGE_INIT
  16. self._encryptor = encrypt.Encryptor(config['password'],
  17. config['method'])
  18. self._fastopen_connected = False
  19. self._data_to_write_to_local = []
  20. self._data_to_write_to_remote = []
  21. self._upstream_status = WAIT_STATUS_READING
  22. self._downstream_status = WAIT_STATUS_INIT
  23. self._client_address = local_sock.getpeername()[:2]
  24. self._remote_address = None
  25. if 'forbidden_ip' in config:
  26. self._forbidden_iplist = config['forbidden_ip']
  27. else:
  28. self._forbidden_iplist = None
  29. if is_local:
  30. self._chosen_server = self._get_a_server()
  31. # 将自身作为handler加入fd_to_handlers
  32. fd_to_handlers[local_sock.fileno()] = self
  33. local_sock.setblocking(False)
  34. local_sock.setsockopt(socket.SOL_TCP, socket.TCP_NODELAY, 1)
  35. # 加入loop的handle仍然是TCPRelay
  36. loop.add(local_sock, eventloop.POLL_IN | eventloop.POLL_ERR,
  37. self._server)
  38. self.last_activity = 0
  39. self._update_activity()

这里将与客户端连接的套接字加入了loop,handler为self._server即那个监听到该连接的TCPRelay对象。

接着上面事件循环,如果是已连接的套接字产生事件,则通过_fdmap.get()获取到那个监听到它的TCPRelay对象,然后在handle_event的过程中,通过_fd_to_handlers.get()获取对应的TCPRelayHandler对象,再使用TCPRelayHandler.handle_event()来处理具体的读写事件。

ps:描述的好乱,抽时间再整理下吧。

参考:

ss2.8 源码

ss源码学习--工作流程的更多相关文章

  1. Netty 源码学习——客户端流程分析

    Netty 源码学习--客户端流程分析 友情提醒: 需要观看者具备一些 NIO 的知识,否则看起来有的地方可能会不明白. 使用版本依赖 <dependency> <groupId&g ...

  2. ss源码学习--从协议建立到完成一次代理请求

    上一次介绍了ss源码中各个事件处理函数完成的工作,这次具体分析一下协议的建立以及请求数据的传输过程. 因为ss的local和server共用一个类以及一系列的事件处理函数,所以看起来稍显复杂.下面来将 ...

  3. live555源码学习1---Socket流程架构图

    怎么说呢,换了工作环境,好多软件公司禁止使用了,有道笔记也无法使用了.发现博客园还可以上传图片,以后只能在这里记录了. 越发的感觉需要尽快把live555的代码拿下.因为工作环境问题,webrtc的源 ...

  4. async源码学习 - 控制流程waterfall函数

    waterfall函数会连续执行数组中的函数,每次通过数组下一个函数的结果.然而,数组任务中的任意一个函数结果传递失败,那么该函数的下一个函数将不会执行,并且主回调函数立马把错误作为参数执行. 以上是 ...

  5. ss源码学习--事件处理

    为了方便区分,以下分别使用local,server,remote代表ss客户端,ss服务端,以及ss客户端请求访问的远程主机. 在shadowsocks中,无论对于local还是server,都需要建 ...

  6. Netty 源码学习——EventLoop

    Netty 源码学习--EventLoop 在前面 Netty 源码学习--客户端流程分析中我们已经知道了一个 EventLoop 大概的流程,这一章我们来详细的看一看. NioEventLoopGr ...

  7. Mybatis源码学习第六天(核心流程分析)之Executor分析

    今Executor这个类,Mybatis虽然表面是SqlSession做的增删改查,其实底层统一调用的是Executor这个接口 在这里贴一下Mybatis查询体系结构图 Executor组件分析 E ...

  8. SpringMVC源码学习之request处理流程

    目的:为看源码提供调用地图,最长调用逻辑深度为8层,反正我是springMVC源码学习地址看了两周才理出来的. 建议看完后还比较晕的,参照这个简单的模型深入底层,仿SpringMVC自己写框架,再来理 ...

  9. mybatis源码学习:插件定义+执行流程责任链

    目录 一.自定义插件流程 二.测试插件 三.源码分析 1.inteceptor在Configuration中的注册 2.基于责任链的设计模式 3.基于动态代理的plugin 4.拦截方法的interc ...

随机推荐

  1. tkinter 布局

  2. Java 日期与数字转换

    package com.test; import org.apache.commons.lang.StringUtils; import org.junit.Test; import java.tex ...

  3. <转载> MySQL 性能优化的最佳20多条经验分享 http://www.jb51.net/article/24392.htm

    当我们去设计数据库表结构,对操作数据库时(尤其是查表时的SQL语句),我们都需要注意数据操作的性能.这里,我们不会讲过多的SQL语句的优化,而只是针对MySQL这一Web应用最多的数据库.希望下面的这 ...

  4. leetcode540

    这道题目的要求,Note: Your solution should run in O(log n) time and O(1) space. 因此应该用二分查找的方式,代码如下: class Sol ...

  5. 利用STM32CubeMX生成HID双向通讯工程

    使用开发板为正点原子ministm32 现在我们先使用HID descriptor Tool来生成我们需要的hid的 保存使用选择.H // D:\usb资料\HID\MSDEV\Projects\t ...

  6. Dubbo 消费者

    1. pom <dependency> <groupId>com.alibaba</groupId> <artifactId>dubbo</art ...

  7. span标签 宽度无效解决方案

    完美的解决方案 下 面代码的CSS定义完美解决了span的宽度设置问题. 由于浏览器通常对不支持的CSS属性采取忽略处理的态度, 所以最好将display:inline -block行写在后面,这样在 ...

  8. ACM__容器之vector

    今天做题碰到了深搜的题,有一种存图方式需要用到vector,对vector不是很熟悉,回顾了一下 vector都知道是一个容器,但并不准确,它是一个多功能的能够操作多种数据结构和算法的模板类和函数库. ...

  9. js 获取北京时间

    <SCRIPT LANGUAGE = "JavaScript"> var xmlhttp = new ActiveXObject("MSXML2.XMLHTT ...

  10. Java 配置环境变量教程

    [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/3 ...