Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

avoid refcycles in tracebacks from happy eyeballs exceptions #809

Open
wants to merge 18 commits into
base: master
Choose a base branch
from

Conversation

graingert
Copy link
Collaborator

@graingert graingert commented Oct 13, 2024

Changes

Checklist

If this is a user-facing code change, like a bugfix or a new feature, please ensure that
you've fulfilled the following conditions (where applicable):

  • You've added tests (in tests/) added which would fail without your patch
  • You've updated the documentation (in docs/, in case of behavior changes or new
    features)
  • You've added a new changelog entry (in docs/versionhistory.rst).

If this is a trivial change, like a typo fix or a code reformatting, then you can ignore
these instructions.

Updating the changelog

If there are no entries after the last release, use **UNRELEASED** as the version.
If, say, your patch fixes issue #123, the entry should look like this:

* Fix big bad boo-boo in task groups (#123 <https://github.com/agronholm/anyio/issues/123>_; PR by @yourgithubaccount)

If there's no issue linked, just link to your pull request instead by updating the
changelog after you've created the PR.

@graingert graingert force-pushed the happy-eyeballs-cyclic-garbage branch from 41264d3 to c217d95 Compare October 13, 2024 11:22
tests/test_sockets.py Outdated Show resolved Hide resolved
@graingert
Copy link
Collaborator Author

This is failing on Python3.9 because anyio is using asyncio.get_running_loop().create_connection() which in 3.9 didn't delete its reference to exceptions I think it should use sock.bind() + asyncio.get_running_loop().sock_connect instead.

src/anyio/_core/_sockets.py Outdated Show resolved Hide resolved
src/anyio/_core/_sockets.py Outdated Show resolved Hide resolved
pyproject.toml Outdated Show resolved Hide resolved
tests/test_sockets.py Outdated Show resolved Hide resolved
tests/test_sockets.py Outdated Show resolved Hide resolved
@graingert graingert marked this pull request as ready for review January 7, 2025 09:51
@graingert graingert requested a review from agronholm January 7, 2025 09:52
"asyncio.BaseEventLoop.create_connection creates refcycles on py 3.9"
)
ip = "127.0.0.1"
port = ephemeral_port_reserve.reserve(ip=ip)
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We have plenty of tests that need ephemeral ports. Why do we have to add a new test dependency for this specific test? (I'd be happy to add such a fixture to anyio's pytest plugin though!)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh is there already a tool for this? I'll have a look

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I didn't say there was a tool yet (that's what my suggestion in parentheses was about). Rather, we currently do this in a rather manual fashion.

@@ -133,6 +135,31 @@ def check_asyncio_bug(anyio_backend_name: str, family: AnyIPAddressFamily) -> No
pytest.skip("Does not work due to a known bug (39148)")


if sys.version_info >= (3, 14):

async def no_other_refs() -> list[object]:
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We don't really need to duplicate this from test_taskgroups.py, do we? Why not just move it to a utility module in the test suite, or something?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I could put it in conftest.py ? happy to have a utils.py

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've usually gone with a utils.py in the test suite. So long as it's structured as a package (which it is in anyio's case), it works fine.

tests/test_sockets.py Outdated Show resolved Hide resolved
Comment on lines +360 to +363
ip = "127.0.0.1"
with socket.socket(AddressFamily.AF_INET) as dummy_socket:
dummy_socket.bind(("0.0.0.0", 0))
free_port = dummy_socket.getsockname()[1]
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
ip = "127.0.0.1"
with socket.socket(AddressFamily.AF_INET) as dummy_socket:
dummy_socket.bind(("0.0.0.0", 0))
free_port = dummy_socket.getsockname()[1]
ip = "127.0.0.1"
with socket.socket(AddressFamily.AF_INET) as dummy_socket:
dummy_socket.bind((ip, 0))
free_port = dummy_socket.getsockname()[1]

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants