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

gh-117536: Fix asyncio _asyncgen_finalizer_hook() #117751

Closed
wants to merge 4 commits into from

Conversation

vstinner
Copy link
Member

@vstinner vstinner commented Apr 11, 2024

Make the finalization of asynchronous generators more reliable.

Store a strong reference to the asynchronous generator which is being closed to make sure that shutdown_asyncgens() can close it even if asyncio.run() cancels all tasks.

Make the finalization of asynchronous generators more reliable.

Store a strong reference to the asynchronous generator which is being
closed to make sure that shutdown_asyncgens() can close it even if
asyncio.run() cancels all tasks.
@vstinner
Copy link
Member Author

I'm not sure of my fix, I didn't touch asyncio for a long time.

I don't know if _asyncgen_close() works as expected, in my quick local tests, I never saw the task executed and I don't know why. It's always shutdown_asyncgens() which closed the async generator for me.

The subtle part is in runner.py:

            _cancel_all_tasks(loop)
            loop.run_until_complete(loop.shutdown_asyncgens())
            loop.run_until_complete(
                loop.shutdown_default_executor(constants.THREAD_JOIN_TIMEOUT))

First, all tasks are cancelled, including _asyncgen_close(agen) task.

@vstinner
Copy link
Member Author

@kumaraditya303: Would you mind to have a look at this asyncio change?

yield 0
yield 1
finally:
ns['state'] = 'finalized'
Copy link
Contributor

Choose a reason for hiding this comment

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

This should await something to prove it's getting aclosed properly

Suggested change
ns['state'] = 'finalized'
await asyncio.sleep(0)
ns['state'] = 'finalized'

Copy link
Member Author

Choose a reason for hiding this comment

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

Right. I added a second test. I prefer to have two tests, since having 'await' also changes the behavior in a subtle way.

@@ -555,10 +560,22 @@ def _check_default_executor(self):
if self._executor_shutdown_called:
raise RuntimeError('Executor shutdown has been called')

async def _asyncgen_close(self, agen):
await agen.aclose()
Copy link
Contributor

Choose a reason for hiding this comment

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

this is just side-stepping the actual bug by delaying the creation of the async_generator_aclose object to avoid the incorrect warning

@graingert graingert closed this Apr 13, 2024
@graingert graingert reopened this Apr 13, 2024
@graingert
Copy link
Contributor

sorry I accidentally closed this PR

@vstinner
Copy link
Member Author

I'm sorry, I don't understand this code enough to address reviews. I prefer to abandon my PR.

If someone wants to take it over, please go ahead :-)

@vstinner vstinner closed this Apr 15, 2024
@vstinner vstinner deleted the asyncio_shutdown_asyncgens branch April 15, 2024 20:00
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants