Python中真正无阻塞的HTTPS服务器

dis*_*rdo 4 python ssl http

我正在尝试用Python构建一个真正无阻塞的HTTPS服务器.如果每个人都玩得很好,下面的最小代码就可以了:

import BaseHTTPServer
import SimpleHTTPServer
import SocketServer
import ssl 

class ThreadedHTTPServer(SocketServer.ThreadingMixIn, BaseHTTPServer.HTTPServer):
    pass

httpd = ThreadedHTTPServer(('localhost', 4443), SimpleHTTPServer.SimpleHTTPRequestHandler)
httpd.socket = ssl.wrap_socket(httpd.socket, keyfile="localhost.key", certfile="localhost.pem", server_side=True)
httpd.serve_forever()
Run Code Online (Sandbox Code Playgroud)

但是,问题是该服务器至少在TLS握手期间阻塞.

测试:

$ nc localhost 4443  # leave this open
Run Code Online (Sandbox Code Playgroud)

然后(在另一个终端):

$ wget --no-check-certificate https://localhost:4443/
--2014-10-23 16:55:54--  https://localhost:4443/
Resolving localhost (localhost)... 127.0.0.1
Connecting to localhost (localhost)|127.0.0.1|:4443... connected.
Run Code Online (Sandbox Code Playgroud)

wget进程阻塞,表示服务器中的某些内容被阻止.一旦我关闭了nc进程,wget就会继续.这显然不实用.

如何在Python中获得真正无阻塞的HTTPS服务器,最好不需要额外的第三方软件?

我应该提一下,在没有TLS的情况下,相同的代码可以正常工作(即没有wrap_socket行).


Steffen Ullrich指出如何做到这一点:传递do_handshake_on_connect=Falsewrap_socket自己,然后自己做握手.在这种情况下,子类BaseHTTPServer.HTTPServer,覆盖handle,然后执行握手,如Python文档(self.request调用套接字)中所示,然后调用super方法.

Ste*_*ich 6

你必须做一个非阻塞SSL通过调用接受ssl.wrap_socketdo_handshake_on_connect=False后来调用do_handshake自己,直到成功为止.请参阅https://docs.python.org/3/library/ssl.html#notes-on-non-blocking-sockets.

您也可以简单地使用Tornado,这是一个用python编写的Web服务器,它也可以完全无阻塞地处理SSL.即使您不想自己使用它,也可以查看源代码以了解如何完成此操作(搜索do_handshake).