Skip to content

Commit

Permalink
[PR #10049/006fbc37 backport][3.12] Improve client performance when t…
Browse files Browse the repository at this point in the history
…here are no auto headers to skip (#10052)

Co-authored-by: J. Nick Koston <nick@koston.org>
  • Loading branch information
patchback[bot] and bdraco authored Nov 26, 2024
1 parent d8e9a6b commit c9eb8e7
Show file tree
Hide file tree
Showing 3 changed files with 24 additions and 15 deletions.
1 change: 1 addition & 0 deletions CHANGES/10049.misc.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Improved performance of making requests when there are no auto headers to skip -- by :user:`bdraco`.
37 changes: 22 additions & 15 deletions aiohttp/client_reqrep.py
Original file line number Diff line number Diff line change
Expand Up @@ -271,6 +271,8 @@ class ClientRequest:
__writer = None # async task for streaming data
_continue = None # waiter future for '100 Continue' response

_skip_auto_headers: Optional["CIMultiDict[None]"] = None

# N.B.
# Adding __del__ method with self._writer closing doesn't make sense
# because _writer is instance method, thus it keeps a reference to self.
Expand Down Expand Up @@ -358,6 +360,10 @@ def __init__(
def __reset_writer(self, _: object = None) -> None:
self.__writer = None

@property
def skip_auto_headers(self) -> CIMultiDict[None]:
return self._skip_auto_headers or CIMultiDict()

@property
def _writer(self) -> Optional["asyncio.Task[None]"]:
return self.__writer
Expand Down Expand Up @@ -469,20 +475,19 @@ def update_headers(self, headers: Optional[LooseHeaders]) -> None:

def update_auto_headers(self, skip_auto_headers: Optional[Iterable[str]]) -> None:
if skip_auto_headers is not None:
self.skip_auto_headers = CIMultiDict(
self._skip_auto_headers = CIMultiDict(
(hdr, None) for hdr in sorted(skip_auto_headers)
)
used_headers = self.headers.copy()
used_headers.extend(self.skip_auto_headers) # type: ignore[arg-type]
used_headers.extend(self._skip_auto_headers) # type: ignore[arg-type]
else:
# Fast path when there are no headers to skip
# which is the most common case.
self.skip_auto_headers = CIMultiDict()
used_headers = self.headers

for hdr, val in self.DEFAULT_HEADERS.items():
if hdr not in used_headers:
self.headers.add(hdr, val)
self.headers[hdr] = val

if hdrs.USER_AGENT not in used_headers:
self.headers[hdrs.USER_AGENT] = SERVER_SOFTWARE
Expand Down Expand Up @@ -584,21 +589,20 @@ def update_body_from_data(self, body: Any) -> None:
self.body = body

# enable chunked encoding if needed
if not self.chunked:
if hdrs.CONTENT_LENGTH not in self.headers:
size = body.size
if size is None:
self.chunked = True
else:
if hdrs.CONTENT_LENGTH not in self.headers:
self.headers[hdrs.CONTENT_LENGTH] = str(size)
if not self.chunked and hdrs.CONTENT_LENGTH not in self.headers:
if (size := body.size) is not None:
self.headers[hdrs.CONTENT_LENGTH] = str(size)
else:
self.chunked = True

# copy payload headers
assert body.headers
headers = self.headers
skip_headers = self._skip_auto_headers
for key, value in body.headers.items():
if key in self.headers or key in self.skip_auto_headers:
if key in headers or (skip_headers is not None and key in skip_headers):
continue
self.headers[key] = value
headers[key] = value

def update_expect_continue(self, expect: bool = False) -> None:
if expect:
Expand Down Expand Up @@ -723,7 +727,10 @@ async def send(self, conn: "Connection") -> "ClientResponse":
# set default content-type
if (
self.method in self.POST_METHODS
and hdrs.CONTENT_TYPE not in self.skip_auto_headers
and (
self._skip_auto_headers is None
or hdrs.CONTENT_TYPE not in self._skip_auto_headers
)
and hdrs.CONTENT_TYPE not in self.headers
):
self.headers[hdrs.CONTENT_TYPE] = "application/octet-stream"
Expand Down
1 change: 1 addition & 0 deletions tests/test_client_request.py
Original file line number Diff line number Diff line change
Expand Up @@ -687,6 +687,7 @@ async def test_content_type_skip_auto_header_bytes(loop, conn) -> None:
skip_auto_headers={"Content-Type"},
loop=loop,
)
assert req.skip_auto_headers == CIMultiDict({"CONTENT-TYPE": None})
resp = await req.send(conn)
assert "CONTENT-TYPE" not in req.headers
resp.close()
Expand Down

0 comments on commit c9eb8e7

Please sign in to comment.