Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: aio-libs/aiohttp
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: v3.11.13
Choose a base ref
...
head repository: aio-libs/aiohttp
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: v3.11.14
Choose a head ref
  • 12 commits
  • 17 files changed
  • 3 contributors

Commits on Feb 24, 2025

  1. Increment version to 3.11.14.dev0

    bdraco committed Feb 24, 2025
    Copy the full SHA
    613a3f0 View commit details

Commits on Feb 26, 2025

  1. [PR #10493/8e8fa959 backport][3.11] Document adjustments to the relea…

    …se process (#10495)
    
    **This is a backport of PR #10493 as merged into master
    (8e8fa95).**
    
    
    - Adding some notes here so I do not forget the social media posting
    - Add note to check RTD to verify the changelog looks good
    - Sign the tags
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Feb 26, 2025
    Copy the full SHA
    40fe535 View commit details

Commits on Mar 15, 2025

  1. [PR #10553/55c5f1fc backport][3.11] Add benchmark for JSON post reque…

    …sts that check the content type (#10554)
    
    **This is a backport of PR #10553 as merged into master
    (55c5f1f).**
    
    <!-- Thank you for your contribution! -->
    
    ## What do these changes do?
    
    Add benchmark for JSON post requests that check the content type
    
    ## Are there changes in behavior for the user?
    
    no
    
    ## Is it a substantial burden for the maintainers to support this?
    no
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Mar 15, 2025
    Copy the full SHA
    7205661 View commit details
  2. [PR #10552/44e669be backport][3.11] Cache parsing of the content-type (

    …#10557)
    
    **This is a backport of PR #10552 as merged into master
    (44e669b).**
    
    
    
    <!-- Thank you for your contribution! -->
    
    ## What do these changes do?
    
    When profiling some frequent POST requests, I found the bulk of the time
    was spent parsing the content-type string. Use the same strategy as we
    do for `parse_mimetype` to cache the parsing.
    
    ## Are there changes in behavior for the user?
    
    performance improvement
    
    ## Is it a substantial burden for the maintainers to support this?
    
    no
    
    ## Related issue number
    
    <!-- Are there any issues opened that will be resolved by merging this
    change? -->
    <!-- Remember to prefix with 'Fixes' if it should close the issue (e.g.
    'Fixes #123'). -->
    
    ## Checklist
    
    - [x] I think the code is well written
    - [ ] Unit tests for the changes exist
    - [ ] Documentation reflects the changes
    - [ ] If you provide code modification, please add yourself to
    `CONTRIBUTORS.txt`
      * The format is &lt;Name&gt; &lt;Surname&gt;.
      * Please keep alphabetical order, the file is sorted by names.
    - [ ] Add a new news fragment into the `CHANGES/` folder
      * name it `<issue_or_pr_num>.<type>.rst` (e.g. `588.bugfix.rst`)
      * if you don't have an issue number, change it to the pull request
        number after creating the PR
        * `.bugfix`: A bug fix for something the maintainers deemed an
          improper undesired behavior that got corrected to match
          pre-agreed expectations.
        * `.feature`: A new behavior, public APIs. That sort of stuff.
        * `.deprecation`: A declaration of future API removals and breaking
          changes in behavior.
        * `.breaking`: When something public is removed in a breaking way.
          Could be deprecated in an earlier release.
        * `.doc`: Notable updates to the documentation structure or build
          process.
        * `.packaging`: Notes for downstreams about unobvious side effects
          and tooling. Changes in the test invocation considerations and
          runtime assumptions.
        * `.contrib`: Stuff that affects the contributor experience. e.g.
          Running tests, building the docs, setting up the development
          environment.
        * `.misc`: Changes that are hard to assign to any of the above
          categories.
      * Make sure to use full sentences with correct case and punctuation,
        for example:
        ```rst
        Fixed issue with non-ascii contents in doctest text files
        -- by :user:`contributor-gh-handle`.
        ```
    
        Use the past tense or the present tense a non-imperative mood,
        referring to what's changed compared to the last released version
        of this project.
    
    <img width="570" alt="Screenshot 2025-03-15 at 11 25 10 AM"
    src="https://github.com/user-attachments/assets/cabaaa7c-3a39-4f90-b450-a6a0559d22d6"
    />
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Mar 15, 2025
    Copy the full SHA
    928e6d7 View commit details

Commits on Mar 16, 2025

  1. [PR #10529/492f63dc backport][3.11] Fixed bug that lead to infinite w…

    …ait for dns futures (#10559)
    
    **This is a backport of PR #10529 as merged into master
    (492f63d).**
    
    <!-- Thank you for your contribution! -->
    
    ## What do these changes do?
    
    Fixed bug that lead to infinite wait for dns futures when exception
    occured in trace.send_dns_cache_miss call.
    
    ## Are there changes in behavior for the user?
    
    No
    
    ## Is it a substantial burden for the maintainers to support this?
    
    No
    
    ## Related issue number
    
    No issue. 
    
    ## Checklist
    
    - [x] I think the code is well written
    - [x] Unit tests for the changes exist
    - [x] Documentation reflects the changes
    - [x] If you provide code modification, please add yourself to
    `CONTRIBUTORS.txt`
      * The format is &lt;Name&gt; &lt;Surname&gt;.
      * Please keep alphabetical order, the file is sorted by names.
    - [x] Add a new news fragment into the `CHANGES/` folder
      * name it `<issue_or_pr_num>.<type>.rst` (e.g. `588.bugfix.rst`)
      * if you don't have an issue number, change it to the pull request
        number after creating the PR
        * `.bugfix`: A bug fix for something the maintainers deemed an
          improper undesired behavior that got corrected to match
          pre-agreed expectations.
        * `.feature`: A new behavior, public APIs. That sort of stuff.
        * `.deprecation`: A declaration of future API removals and breaking
          changes in behavior.
        * `.breaking`: When something public is removed in a breaking way.
          Could be deprecated in an earlier release.
        * `.doc`: Notable updates to the documentation structure or build
          process.
        * `.packaging`: Notes for downstreams about unobvious side effects
          and tooling. Changes in the test invocation considerations and
          runtime assumptions.
        * `.contrib`: Stuff that affects the contributor experience. e.g.
          Running tests, building the docs, setting up the development
          environment.
        * `.misc`: Changes that are hard to assign to any of the above
          categories.
      * Make sure to use full sentences with correct case and punctuation,
        for example:
        ```rst
        Fixed issue with non-ascii contents in doctest text files
        -- by :user:`contributor-gh-handle`.
        ```
    
    Co-authored-by: Alexey Stavrov <logioniz@ya.ru>
    patchback[bot] and Logioniz authored Mar 16, 2025
    Copy the full SHA
    e9f3f03 View commit details
  2. [PR #10551/d067260d backport][3.11] Re-raise OSError as ClientConnect…

    …ionError when failing to explicitly close connector socket (#10561)
    
    **This is a backport of PR #10551 as merged into master
    (d067260).**
    
    
    <!-- Thank you for your contribution! -->
    
    ## What do these changes do?
    
    This is a followup to #10464 to handle the case where `socket.close()`
    can also raise. This matches the logic we have in aiohappyeyeballs:
    
    
    https://github.com/aio-libs/aiohappyeyeballs/blob/e3bd5bdf44f5d187802de6dcb08d27e1ca6da048/src/aiohappyeyeballs/impl.py#L227
    
    We shouldn't raising `OSError` externally from this method as callers
    expect a `ClientError`
    
    
    ## Are there changes in behavior for the user?
    
    bugfix
    
    ## Is it a substantial burden for the maintainers to support this?
    
    no
    
    ## Related issue number
    
    fixes #10506
    
    ## Checklist
    
    - [x] I think the code is well written
    - [x] Unit tests for the changes exist
    - [x] Documentation reflects the changes
    - [x] If you provide code modification, please add yourself to
    `CONTRIBUTORS.txt`
      * The format is &lt;Name&gt; &lt;Surname&gt;.
      * Please keep alphabetical order, the file is sorted by names.
    - [x] Add a new news fragment into the `CHANGES/` folder
      * name it `<issue_or_pr_num>.<type>.rst` (e.g. `588.bugfix.rst`)
      * if you don't have an issue number, change it to the pull request
        number after creating the PR
        * `.bugfix`: A bug fix for something the maintainers deemed an
          improper undesired behavior that got corrected to match
          pre-agreed expectations.
        * `.feature`: A new behavior, public APIs. That sort of stuff.
        * `.deprecation`: A declaration of future API removals and breaking
          changes in behavior.
        * `.breaking`: When something public is removed in a breaking way.
          Could be deprecated in an earlier release.
        * `.doc`: Notable updates to the documentation structure or build
          process.
        * `.packaging`: Notes for downstreams about unobvious side effects
          and tooling. Changes in the test invocation considerations and
          runtime assumptions.
        * `.contrib`: Stuff that affects the contributor experience. e.g.
          Running tests, building the docs, setting up the development
          environment.
        * `.misc`: Changes that are hard to assign to any of the above
          categories.
      * Make sure to use full sentences with correct case and punctuation,
        for example:
        ```rst
        Fixed issue with non-ascii contents in doctest text files
        -- by :user:`contributor-gh-handle`.
        ```
    
        Use the past tense or the present tense a non-imperative mood,
        referring to what's changed compared to the last released version
        of this project.
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Mar 16, 2025
    Copy the full SHA
    6357c05 View commit details
  3. [PR #10556/9d4e1161 backport][3.11] Break cyclic references at connec…

    …tion close when there was a traceback (#10566)
    
    **This is a backport of PR #10556 as merged into master
    (9d4e116).**
    
    
    <!-- Thank you for your contribution! -->
    
    ## What do these changes do?
    
    Clears the exception on the `DataQueue` and `WebSocketDataQueue` when
    the connection is closed to break cyclic references.
    
    ## Are there changes in behavior for the user?
    
    bugfix
    
    ## Is it a substantial burden for the maintainers to support this?
    
    no
    
    ## Related issue number
    
    fixes #10535
    
    ## Checklist
    
    - [ ] I think the code is well written
    - [ ] Unit tests for the changes exist
    - [ ] Documentation reflects the changes
    - [ ] If you provide code modification, please add yourself to
    `CONTRIBUTORS.txt`
      * The format is &lt;Name&gt; &lt;Surname&gt;.
      * Please keep alphabetical order, the file is sorted by names.
    - [ ] Add a new news fragment into the `CHANGES/` folder
      * name it `<issue_or_pr_num>.<type>.rst` (e.g. `588.bugfix.rst`)
      * if you don't have an issue number, change it to the pull request
        number after creating the PR
        * `.bugfix`: A bug fix for something the maintainers deemed an
          improper undesired behavior that got corrected to match
          pre-agreed expectations.
        * `.feature`: A new behavior, public APIs. That sort of stuff.
        * `.deprecation`: A declaration of future API removals and breaking
          changes in behavior.
        * `.breaking`: When something public is removed in a breaking way.
          Could be deprecated in an earlier release.
        * `.doc`: Notable updates to the documentation structure or build
          process.
        * `.packaging`: Notes for downstreams about unobvious side effects
          and tooling. Changes in the test invocation considerations and
          runtime assumptions.
        * `.contrib`: Stuff that affects the contributor experience. e.g.
          Running tests, building the docs, setting up the development
          environment.
        * `.misc`: Changes that are hard to assign to any of the above
          categories.
      * Make sure to use full sentences with correct case and punctuation,
        for example:
        ```rst
        Fixed issue with non-ascii contents in doctest text files
        -- by :user:`contributor-gh-handle`.
        ```
    
        Use the past tense or the present tense a non-imperative mood,
        referring to what's changed compared to the last released version
        of this project.
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Mar 16, 2025
    Copy the full SHA
    771d203 View commit details
  4. [PR #10569/dfbf782b backport][3.11] Break cyclic references when ther…

    …e is an exception handling a request (#10571)
    
    **This is a backport of PR #10569 as merged into master
    (dfbf782).**
    
    <!-- Thank you for your contribution! -->
    
    ## What do these changes do?
    
    This is a partial fix for #10548
    
    - There is still another case for `SystemRoute`s that needs to be
    addressed. No reproducer available yet.
    - There is also another case on the client side on connection refused
    that still needs to be addressed
    #10548 (comment)
    
    ## Are there changes in behavior for the user?
    
    fixes memory leak
    
    ## Is it a substantial burden for the maintainers to support this?
    no
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Mar 16, 2025
    Copy the full SHA
    6ae2570 View commit details
  5. [PR #10564/a59e74b7 backport][3.11] Log offending websocket client ad…

    …dress when no protocols overlap (#10575)
    
    **This is a backport of PR #10564 as merged into master
    (a59e74b).**
    
    
    
    <!-- Thank you for your contribution! -->
    
    ## What do these changes do?
    
    Logs the remote address of a WebSocket client that has no overlapping
    protocols
    
    ## Are there changes in behavior for the user?
    
    Which client has the problem should be a bit more discoverable
    
    ## Is it a substantial burden for the maintainers to support this?
    
    no
    ## Related issue number
    closes #10563
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Mar 16, 2025
    Copy the full SHA
    4005080 View commit details
  6. [PR #10542/e1d2d77c backport][3.11] only use AI_ADDRCONFIG when sup…

    …ported by getaddrinfo (#10578)
    patchback[bot] authored Mar 16, 2025
    Copy the full SHA
    9396ef1 View commit details

Commits on Mar 17, 2025

  1. [PR #10577/3c60cd22 backport][3.11] Parametrize leak tests (#10580)

    **This is a backport of PR #10577 as merged into master
    (3c60cd2).**
    
    
    
    Small cleanup to the leak tests
    #10569 (comment)
    
    Co-authored-by: J. Nick Koston <nick@koston.org>
    patchback[bot] and bdraco authored Mar 17, 2025
    Copy the full SHA
    d40e227 View commit details
  2. Release 3.11.14 (#10582)

    <img width="466" alt="Screenshot 2025-03-16 at 2 32 10 PM"
    src="https://github.com/user-attachments/assets/635511fd-6b63-49c7-bb1a-cf514545b604"
    />
    bdraco authored Mar 17, 2025
    Copy the full SHA
    1a48a62 View commit details
77 changes: 77 additions & 0 deletions CHANGES.rst
Original file line number Diff line number Diff line change
@@ -10,6 +10,83 @@

.. towncrier release notes start
3.11.14 (2025-03-16)
====================

Bug fixes
---------

- Fixed an issue where dns queries were delayed indefinitely when an exception occurred in a ``trace.send_dns_cache_miss``
-- by :user:`logioniz`.


*Related issues and pull requests on GitHub:*
:issue:`10529`.



- Fixed DNS resolution on platforms that don't support ``socket.AI_ADDRCONFIG`` -- by :user:`maxbachmann`.


*Related issues and pull requests on GitHub:*
:issue:`10542`.



- The connector now raises :exc:`aiohttp.ClientConnectionError` instead of :exc:`OSError` when failing to explicitly close the socket after :py:meth:`asyncio.loop.create_connection` fails -- by :user:`bdraco`.


*Related issues and pull requests on GitHub:*
:issue:`10551`.



- Break cyclic references at connection close when there was a traceback -- by :user:`bdraco`.

Special thanks to :user:`availov` for reporting the issue.


*Related issues and pull requests on GitHub:*
:issue:`10556`.



- Break cyclic references when there is an exception handling a request -- by :user:`bdraco`.


*Related issues and pull requests on GitHub:*
:issue:`10569`.




Features
--------

- Improved logging on non-overlapping WebSocket client protocols to include the remote address -- by :user:`bdraco`.


*Related issues and pull requests on GitHub:*
:issue:`10564`.




Miscellaneous internal changes
------------------------------

- Improved performance of parsing content types by adding a cache in the same manner currently done with mime types -- by :user:`bdraco`.


*Related issues and pull requests on GitHub:*
:issue:`10552`.




----


3.11.13 (2025-02-24)
====================

1 change: 1 addition & 0 deletions CONTRIBUTORS.txt
Original file line number Diff line number Diff line change
@@ -31,6 +31,7 @@ Alexandru Mihai
Alexey Firsov
Alexey Nikitin
Alexey Popravka
Alexey Stavrov
Alexey Stepanov
Amin Etesamian
Amit Tulshyan
2 changes: 1 addition & 1 deletion aiohttp/__init__.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
__version__ = "3.11.13"
__version__ = "3.11.14"

from typing import TYPE_CHECKING, Tuple

1 change: 1 addition & 0 deletions aiohttp/_websocket/reader_py.py
Original file line number Diff line number Diff line change
@@ -93,6 +93,7 @@ def _release_waiter(self) -> None:
def feed_eof(self) -> None:
self._eof = True
self._release_waiter()
self._exception = None # Break cyclic references

def feed_data(self, data: "WSMessage", size: "int_") -> None:
self._size += size
1 change: 1 addition & 0 deletions aiohttp/client_proto.py
Original file line number Diff line number Diff line change
@@ -64,6 +64,7 @@ def force_close(self) -> None:
self._should_close = True

def close(self) -> None:
self._exception = None # Break cyclic references
transport = self.transport
if transport is not None:
transport.close()
11 changes: 7 additions & 4 deletions aiohttp/connector.py
Original file line number Diff line number Diff line change
@@ -1015,11 +1015,11 @@ async def _resolve_host_with_throttle(
This method must be run in a task and shielded from cancellation
to avoid cancelling the underlying lookup.
"""
if traces:
for trace in traces:
await trace.send_dns_cache_miss(host)
try:
if traces:
for trace in traces:
await trace.send_dns_cache_miss(host)

for trace in traces:
await trace.send_dns_resolvehost_start(host)

@@ -1138,7 +1138,10 @@ async def _wrap_create_connection(
# Will be hit if an exception is thrown before the event loop takes the socket.
# In that case, proactively close the socket to guard against event loop leaks.
# For example, see https://github.com/MagicStack/uvloop/issues/653.
sock.close()
try:
sock.close()
except OSError as exc:
raise client_error(req.connection_key, exc) from exc

async def _wrap_existing_connection(
self,
24 changes: 19 additions & 5 deletions aiohttp/helpers.py
Original file line number Diff line number Diff line change
@@ -21,7 +21,7 @@
from email.utils import parsedate
from math import ceil
from pathlib import Path
from types import TracebackType
from types import MappingProxyType, TracebackType
from typing import (
Any,
Callable,
@@ -357,6 +357,20 @@ def parse_mimetype(mimetype: str) -> MimeType:
)


@functools.lru_cache(maxsize=56)
def parse_content_type(raw: str) -> Tuple[str, MappingProxyType[str, str]]:
"""Parse Content-Type header.
Returns a tuple of the parsed content type and a
MappingProxyType of parameters.
"""
msg = HeaderParser().parsestr(f"Content-Type: {raw}")
content_type = msg.get_content_type()
params = msg.get_params(())
content_dict = dict(params[1:]) # First element is content type again
return content_type, MappingProxyType(content_dict)


def guess_filename(obj: Any, default: Optional[str] = None) -> Optional[str]:
name = getattr(obj, "name", None)
if name and isinstance(name, str) and name[0] != "<" and name[-1] != ">":
@@ -710,10 +724,10 @@ def _parse_content_type(self, raw: Optional[str]) -> None:
self._content_type = "application/octet-stream"
self._content_dict = {}
else:
msg = HeaderParser().parsestr("Content-Type: " + raw)
self._content_type = msg.get_content_type()
params = msg.get_params(())
self._content_dict = dict(params[1:]) # First element is content type again
content_type, content_mapping_proxy = parse_content_type(raw)
self._content_type = content_type
# _content_dict needs to be mutable so we can update it
self._content_dict = content_mapping_proxy.copy()

@property
def content_type(self) -> str:
7 changes: 5 additions & 2 deletions aiohttp/resolver.py
Original file line number Diff line number Diff line change
@@ -18,6 +18,9 @@

_NUMERIC_SOCKET_FLAGS = socket.AI_NUMERICHOST | socket.AI_NUMERICSERV
_NAME_SOCKET_FLAGS = socket.NI_NUMERICHOST | socket.NI_NUMERICSERV
_AI_ADDRCONFIG = socket.AI_ADDRCONFIG
if hasattr(socket, "AI_MASK"):
_AI_ADDRCONFIG &= socket.AI_MASK


class ThreadedResolver(AbstractResolver):
@@ -38,7 +41,7 @@ async def resolve(
port,
type=socket.SOCK_STREAM,
family=family,
flags=socket.AI_ADDRCONFIG,
flags=_AI_ADDRCONFIG,
)

hosts: List[ResolveResult] = []
@@ -105,7 +108,7 @@ async def resolve(
port=port,
type=socket.SOCK_STREAM,
family=family,
flags=socket.AI_ADDRCONFIG,
flags=_AI_ADDRCONFIG,
)
except aiodns.error.DNSError as exc:
msg = exc.args[1] if len(exc.args) >= 1 else "DNS lookup failed"
14 changes: 11 additions & 3 deletions aiohttp/web_protocol.py
Original file line number Diff line number Diff line change
@@ -520,8 +520,6 @@ async def start(self) -> None:
keep_alive(True) specified.
"""
loop = self._loop
handler = asyncio.current_task(loop)
assert handler is not None
manager = self._manager
assert manager is not None
keepalive_timeout = self._keepalive_timeout
@@ -551,7 +549,16 @@ async def start(self) -> None:
else:
request_handler = self._request_handler

request = self._request_factory(message, payload, self, writer, handler)
# Important don't hold a reference to the current task
# as on traceback it will prevent the task from being
# collected and will cause a memory leak.
request = self._request_factory(
message,
payload,
self,
writer,
self._task_handler or asyncio.current_task(loop), # type: ignore[arg-type]
)
try:
# a new task is used for copy context vars (#3406)
coro = self._handle_request(request, start, request_handler)
@@ -617,6 +624,7 @@ async def start(self) -> None:
self.force_close()
raise
finally:
request._task = None # type: ignore[assignment] # Break reference cycle in case of exception
if self.transport is None and resp is not None:
self.log_debug("Ignored premature client disconnection.")

3 changes: 2 additions & 1 deletion aiohttp/web_ws.py
Original file line number Diff line number Diff line change
@@ -252,7 +252,8 @@ def _handshake(
else:
# No overlap found: Return no protocol as per spec
ws_logger.warning(
"Client protocols %r don’t overlap server-known ones %r",
"%s: Client protocols %r don’t overlap server-known ones %r",
request.remote,
req_protocols,
self._protocols,
)
8 changes: 6 additions & 2 deletions docs/contributing-admins.rst
Original file line number Diff line number Diff line change
@@ -21,9 +21,9 @@ To create a new release:
#. Run ``towncrier``.
#. Check and cleanup the changes in ``CHANGES.rst``.
#. Checkout a new branch: e.g. ``git checkout -b release/v3.8.6``
#. Commit and create a PR. Once PR is merged, continue.
#. Commit and create a PR. Verify the changelog and release notes look good on Read the Docs. Once PR is merged, continue.
#. Go back to the release branch: e.g. ``git checkout 3.8 && git pull``
#. Add a tag: e.g. ``git tag -a v3.8.6 -m 'Release 3.8.6'``
#. Add a tag: e.g. ``git tag -a v3.8.6 -m 'Release 3.8.6' -s``
#. Push the tag: e.g. ``git push origin v3.8.6``
#. Monitor CI to ensure release process completes without errors.

@@ -49,6 +49,10 @@ first merge into the newer release branch (e.g. 3.8 into 3.9) and then to master

Back on the original release branch, bump the version number and append ``.dev0`` in ``__init__.py``.

Post the release announcement to social media:
- BlueSky: https://bsky.app/profile/aiohttp.org and re-post to https://bsky.app/profile/aio-libs.org
- Mastodon: https://fosstodon.org/@aiohttp and re-post to https://fosstodon.org/@aio_libs

If doing a minor release:

#. Create a new release branch for future features to go to: e.g. ``git checkout -b 3.10 3.9 && git push``
47 changes: 47 additions & 0 deletions tests/isolated/check_for_client_response_leak.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
import asyncio
import contextlib
import gc
import sys

from aiohttp import ClientError, ClientSession, web
from aiohttp.test_utils import get_unused_port_socket

gc.set_debug(gc.DEBUG_LEAK)


async def main() -> None:
app = web.Application()

async def stream_handler(request: web.Request) -> web.Response:
assert request.transport is not None
request.transport.close() # Forcefully closing connection
return web.Response()

app.router.add_get("/stream", stream_handler)
sock = get_unused_port_socket("127.0.0.1")
port = sock.getsockname()[1]

runner = web.AppRunner(app)
await runner.setup()
site = web.SockSite(runner, sock)
await site.start()

session = ClientSession()

async def fetch_stream(url: str) -> None:
"""Fetch a stream and read a few bytes from it."""
with contextlib.suppress(ClientError):
await session.get(url)

client_task = asyncio.create_task(fetch_stream(f"http://localhost:{port}/stream"))
await client_task
gc.collect()
client_response_present = any(
type(obj).__name__ == "ClientResponse" for obj in gc.garbage
)
await session.close()
await runner.cleanup()
sys.exit(1 if client_response_present else 0)


asyncio.run(main())
41 changes: 41 additions & 0 deletions tests/isolated/check_for_request_leak.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import asyncio
import gc
import sys
from typing import NoReturn

from aiohttp import ClientSession, web
from aiohttp.test_utils import get_unused_port_socket

gc.set_debug(gc.DEBUG_LEAK)


async def main() -> None:
app = web.Application()

async def handler(request: web.Request) -> NoReturn:
await request.json()
assert False

app.router.add_route("GET", "/json", handler)
sock = get_unused_port_socket("127.0.0.1")
port = sock.getsockname()[1]

runner = web.AppRunner(app)
await runner.setup()
site = web.SockSite(runner, sock)
await site.start()

async with ClientSession() as session:
async with session.get(f"http://127.0.0.1:{port}/json") as resp:
await resp.read()

# Give time for the cancelled task to be collected
await asyncio.sleep(0.5)
gc.collect()
request_present = any(type(obj).__name__ == "Request" for obj in gc.garbage)
await session.close()
await runner.cleanup()
sys.exit(1 if request_present else 0)


asyncio.run(main())
Loading