18

Python“请求”库目前风靡一时,因为它为发出 HTTP 请求提供了漂亮的接口——但在它之下似乎有许多间接层——会话、HTTP 适配器,最后是 urllib3 的机制。

如果我已经持有一个打开的套接字,并且想要使用“请求”沿着该套接字发送 HTTP 响应并接收回复,那么在这个抽象堆栈中的哪个位置是正确的干预位置?

如果没有某种干预(或定制?),堆栈将尝试为我创建一个新的 TCP/IP 套接字,但在我的特定应用程序中,我的代码在代表我建立连接之前不会被调用,所以我会如果我希望能够使用 Requests 的功能,需要说服 Requests 谈论现有的套接字。

请求库:

http://pypi.python.org/pypi/requests

https://github.com/kennethreitz/requests

4

2 回答 2

12

以下代码需要来自 git 的请求(尤其是requests.packages.urllib3.poolmanager.PoolManager._new_pool()

我用ncat -v -l 127.0.0.1 8000

问题在于,连接不是由 urllib3 而是由标准库中的 httplib 打开的。

import socket
import requests
from requests.adapters import HTTPAdapter
from requests.packages.urllib3 import PoolManager, HTTPConnectionPool

try:
    from http.client import HTTPConnection
except ImportError:
    from httplib import HTTPConnection


class MyAdapter(HTTPAdapter):
    def init_poolmanager(self, connections, maxsize):
        self.poolmanager = MyPoolManager(num_pools=connections,
                                         maxsize=maxsize)


class MyPoolManager(PoolManager):
    def _new_pool(self, scheme, host, port):
        # Important!
        if scheme == 'http' and host == my_host and port == my_port:
            return MyHTTPConnectionPool(host, port, **self.connection_pool_kw)
        return super(PoolManager, self)._new_pool(self, scheme, host, port)


class MyHTTPConnectionPool(HTTPConnectionPool):
    def _new_conn(self):
        self.num_connections += 1
        return MyHTTPConnection(host=self.host,
                            port=self.port,
                            strict=self.strict)


class MyHTTPConnection(HTTPConnection):
    def connect(self):
        """Connect to the host and port specified in __init__."""
        # Original
        # self.sock = socket.create_connection((self.host, self.port),
        #                                    self.timeout, self.source_address)
        # Important!
        self.sock = my_socket
        if self._tunnel_host:
            self._tunnel()


if __name__ == '__main__':
    import time

    my_host = '127.0.0.1'
    my_port = 8000

    my_socket = socket.create_connection((my_host, my_port))
    time.sleep(4)
    s = requests.Session()
    s.mount('http://', MyAdapter())
    s.get('http://127.0.0.1:8000/foo')

编辑:

或者直接对连接池进行monkeypatching:

class MyHTTPConnection(HTTPConnection):
    def connect(self):
        self.sock = my_socket
        if self._tunnel_host:
            self._tunnel()

requests.packages.urllib3.connectionpool.HTTPConnection = MyHTTPConnection

if __name__ == '__main__':
    my_host = '127.0.0.1'
    my_port = 8000

    my_socket = socket.create_connection((my_host, my_port))
    requests.get('http://127.0.0.1:8000/foo')
于 2013-02-03T09:33:11.557 回答
0

直接去urllib3图书馆;urllib3.connectionpool它在模块中拥有一个连接池。

您可以更换池或通过破解poolmanager模块来调整它。

于 2013-02-02T18:32:25.263 回答