Python HTTP 服务器保持连接活动

问题描述 投票:0回答:2

我正在尝试测试用 C 编写的 HTTP 客户端,该客户端将 HTTP POST 请求发送到我计算机上的本地服务器。我在我的 POST 请求中添加了标头

keep-alive
,在我的计算机上运行的 python3 HTTP 服务器上看起来像这样:

<ip-address-1> - - [29/Apr/2018 18:27:49] "POST /html HTTP/1.1" 200 -
Host: <ip-address-2>
Content-Type: application/json
Content-Length: 168
Connection: Keep-Alive
Keep-Alive: timeout=5, max=100


INFO:root:POST request,
Body:
{
"field": "abc",
"time": "2018-04-29T01:27:50.322000Z" 
}

HTTP 服务器 POST 处理程序如下所示:

class S(BaseHTTPRequestHandler):
    def _set_response(self):
        self.send_response(200)
        self.send_header('Content-type', 'text/html')
        self.send_header("Connection", "keep-alive")
        self.send_header("keep-alive", "timeout=5, max=30")
        self.end_headers()

    def do_POST(self):
        content_length = int(self.headers['Content-Length']) # <--- Gets the size of data
        post_data = self.rfile.read(content_length) # <--- Gets the data itself
        print(self.headers)
        logging.info("POST request,\nBody:\n%s\n", post_data.decode('utf-8'))

        self._set_response()
        self.wfile.write("POST request for {}".format(self.path).encode('utf-8'))

def run(server_class=HTTPServer, handler_class=S, port=8080):
    logging.basicConfig(level=logging.INFO)
    server_address = ('', port)
    httpd = server_class(server_address, handler_class)
    logging.info('Starting httpd...\n')
    try:
        httpd.serve_forever()
    except KeyboardInterrupt:
        pass
    httpd.server_close()
    logging.info('Stopping httpd...\n')

我在客户端看到的标头响应是:

HTTP/1.0 200 OK
Server: BaseHTTP/0.6 Python/3.5.2
Date: Tue, 29 April 2018 16:07:42 GMT
Content-type: text/html
Connection: keep-alive
keep-alive: timeout=5, max=30

我最终仍然收到断开连接回调,所以我的问题是如何从服务器端设置保持活动连接参数?

python-3.x http http-headers httpserver
2个回答
11
投票

默认情况下,

BaseHTTPRequestHandler
会发出 HTTP/1.0 响应,如您在
HTTP/1.0 200 OK
中所见。
HTTP/1.1
是保持活动响应所必需的,如 文档(或 对于 v3)所示:

协议版本

这指定响应中使用的 HTTP 协议版本。如果设置为“HTTP/1.1”,服务器将允许 HTTP 持久连接; 但是,您的服务器必须包含准确的内容长度 对客户端的所有响应中的标头(使用 send_header())。为了 向后兼容,设置默认为“HTTP/1.0”。

然后,正如您在引用中所看到的,您还必须为您的回复设置正确的内容长度。

请注意,当前您发送的响应没有正文,您应该使用 204(无内容)代码并添加

Content-length: 0
标头,或添加一个小正文(在 Content-Length 中具有正确的字节数,警告,这不是字符计数器,即字节计数器,在 ascii7 中几乎相同,但与其他编码不同)。


0
投票

我将主要展示如何实施所提出的已接受的答案。接受的答案中缺少让它工作的基础步骤2。

1/ 让 HTTPServer 服务于 HTTP/1.1 而不是 HTTP/1.0

class HTTP11RequestHandler(BaseHTTPRequestHandler):
    def __init__(self, request, client_address, server):
         self.protocol_version = 'HTTP/1.1'
         super().__init__(request, client_address, server)
         
    #...as you would do in a regular RequestHandler, except that you must specify the Content-Length header
    def do_GET(self):
        message = 'the backend answer'
        self.send_response(200)
        self.send_header('Content-Type','text/plain')
        self.send_header('Content-Length', len(message))
        self.end_headers()

2/ HTTPServer 无法同时处理多个连接。 连接保持活动状态,这将不起作用。

=> 使用 https://docs.python.org/3.8/library/http.server.html#http.server.ThreadingHTTPServer

ThreadingHTTPServer
而不是
HTTPServer

def start_backend_server(ip,port,requestHandler):
    backend_server = ThreadingHTTPServer((ip,port), requestHandler)
    f = lambda : backend_server.serve_forever()
    backend_thread = threading.Thread(target=f)
    backend_thread.daemon=True
    backend_thread.start()
    return backend_thread

就这样称呼吧

backend_thread = start_backend_thread('127.0.0.1', '80', HTTP11RequestHandler)
© www.soinside.com 2019 - 2024. All rights reserved.