From fbfe159cc68fad7a86a1554031d9489561e85996 Mon Sep 17 00:00:00 2001 From: "J. Nick Koston" Date: Tue, 1 Oct 2024 12:02:54 -0500 Subject: [PATCH] Speed up the ConnectionKey (#9365) (cherry picked from commit d684195bb5cb90a85a07b98127a298f3b1ae2ef3) --- CHANGES/9365.breaking.rst | 1 + aiohttp/client_reqrep.py | 24 +++++++++++++++--------- aiohttp/connector.py | 5 ++--- 3 files changed, 18 insertions(+), 12 deletions(-) create mode 100644 CHANGES/9365.breaking.rst diff --git a/CHANGES/9365.breaking.rst b/CHANGES/9365.breaking.rst new file mode 100644 index 00000000000..f0224170f07 --- /dev/null +++ b/CHANGES/9365.breaking.rst @@ -0,0 +1 @@ +Changed ``ClientRequest.connection_key`` to be a `NamedTuple` to improve client performance -- by :user:`bdraco`. diff --git a/aiohttp/client_reqrep.py b/aiohttp/client_reqrep.py index 293c745b349..64795e97ff6 100644 --- a/aiohttp/client_reqrep.py +++ b/aiohttp/client_reqrep.py @@ -18,6 +18,7 @@ Iterable, List, Mapping, + NamedTuple, Optional, Tuple, Type, @@ -208,8 +209,13 @@ def _merge_ssl_params( return ssl -@attr.s(auto_attribs=True, slots=True, frozen=True, cache_hash=True) -class ConnectionKey: +_SSL_SCHEMES = frozenset(("https", "wss")) + + +# ConnectionKey is a NamedTuple because it is used as a key in a dict +# and a set in the connector. Since a NamedTuple is a tuple it uses +# the fast native tuple __hash__ and __eq__ implementation in CPython. +class ConnectionKey(NamedTuple): # the key should contain an information about used proxy / TLS # to prevent reusing wrong connections from a pool host: str @@ -358,7 +364,7 @@ def _writer(self, writer: Optional["asyncio.Task[None]"]) -> None: writer.add_done_callback(self.__reset_writer) def is_ssl(self) -> bool: - return self.url.scheme in ("https", "wss") + return self.url.scheme in _SSL_SCHEMES @property def ssl(self) -> Union["SSLContext", bool, Fingerprint]: @@ -366,16 +372,16 @@ def ssl(self) -> Union["SSLContext", bool, Fingerprint]: @property def connection_key(self) -> ConnectionKey: - proxy_headers = self.proxy_headers - if proxy_headers: + if proxy_headers := self.proxy_headers: h: Optional[int] = hash(tuple(proxy_headers.items())) else: h = None + url = self.url return ConnectionKey( - self.host, - self.port, - self.is_ssl(), - self.ssl, + url.raw_host or "", + url.port, + url.scheme in _SSL_SCHEMES, + self._ssl, self.proxy, self.proxy_auth, h, diff --git a/aiohttp/connector.py b/aiohttp/connector.py index 13c1a0cdc48..31d3c6df083 100644 --- a/aiohttp/connector.py +++ b/aiohttp/connector.py @@ -32,7 +32,6 @@ ) import aiohappyeyeballs -import attr from . import hdrs, helpers from .abc import AbstractResolver, ResolveResult @@ -1401,8 +1400,8 @@ async def _create_proxy_connection( # asyncio handles this perfectly proxy_req.method = hdrs.METH_CONNECT proxy_req.url = req.url - key = attr.evolve( - req.connection_key, proxy=None, proxy_auth=None, proxy_headers_hash=None + key = req.connection_key._replace( + proxy=None, proxy_auth=None, proxy_headers_hash=None ) conn = Connection(self, key, proto, self._loop) proxy_resp = await proxy_req.send(conn)