-
-
Notifications
You must be signed in to change notification settings - Fork 30.9k
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
Confusing CancelError message if multiple cancellations are scheduled #90985
Comments
Suppose multiple What message ( As of Python 3.10 it is the message from the last This makes use of cancellation message not robust: a task can be cancelled by many sources, task-groups adds even more mess. Guido van Rossum suggested that messages should be collected in a list and raised altogether. There is a possibility to do it in a backward-compatible way: construct the exception as
Working with exc.args[0] / exc.args[1] is tedious and error-prone. I propose adding The single message is not very useful but a list of messages can be used in timeouts implementation as cancellation count alternative. I don't have a strong preference now but want to carefully discuss possible opportunities before making the final decision. |
What about |
I would like to go on record (again) as preferring to get rid of the cancel-message parameter altogether. Let's deprecate it. When we initially designed things without a cancel message we did it on purpose -- "cancellation" is a single flag, not a collection of data. |
Deprecation is a good answer. |
I don't really understand all the hate around the idea of a cancel message. One reason it's useful is that it provides a simple way to say *why* something is being cancelled or *what* is cancelling it. It provides additional context to the exception, in the same way that a normal exception's message can provide additional helpful context. Take for example this chunk of code: import asyncio
import random
async def job():
await asyncio.sleep(5)
async def main():
task = asyncio.create_task(job())
await asyncio.sleep(1)
r = random.random()
if r < 0.5:
task.cancel()
else:
task.cancel()
await task
if __name__=="__main__":
asyncio.run(main()) Without passing a message, there's no way to tell which of the two lines called cancel, or why. The tracebacks are identical in both cases. This is even in the simplest case where only one cancellation is occurring. Passing a message lets one disambiguate the two call sites. And if there is truly a race condition where it's up in the air between two things cancelling it, I think it would be okay for the chosen message to be indeterminate, as either would have instigated the cancel in the absence of the other. |
But that example is made-up. Is there a real-world situation where you need to know the call site, and it wouldn't be obvious from other log messages? Directly cancelling a task without also somehow catching the cancellation (like in the timeout or task group cases) feels like an odd practice to me. And passing the cancel message through is complex (as we've seen in recent PRs). |
For reference, the msg parameter of Task.cancel() was added in bpo-31033. It seems that the initial use case was for debugging. I do not see how it differs from the following example: r = random.random()
if r < 0.5:
x = 0
else:
x = 0
1/x In the traceback we see the line where an error occurred but we do not see a line which lead to this error. |
Before we land #76021 we should have a somewhat more public discussion (e.g. on python-dev or maybe in Async-SIG, https://discuss.python.org/c/async-sig/20; or at least here) about deprecating the cancel message. I'm all for it but certainly Chris Jerdonek (who wrote the original code, see bpo-31033) needs to have a say, and from a comment on #64150 it looks Yury Selivanov also really liked it. |
If the cancellation message should be kept it needs improvements anyway, the single message doesn't work well with multiple I can imagine a 'CancelledError(*msgs)' and 'raise exc.drop_msg(msg)' as a function equivalent of task cancellation counter and The counter is easier to understand I guess. Both multi-message and counter require new APIs, timeout() can be adapted to any solution. |
Andrew asked me for my opinion on the matter -- I think we should get rid of the message. Exception messages for "signals" can be easily lost if an exception was re-raised. If the reason of why something is being cancelled is important (in my experience it never is) it should be recorded elsewhere. |
IOW I think that supporting custom messages is a needless complication of our API. Given how complex task trees can become with TaskGroups collecting those messages and presenting them all to the user isn't trivial, showing just first/last defeats the purpose. So i'm in favor of dropping it. |
Fixed by deprecating the message argument to cancel(). It will be removed in |
Reason: we were too hasty in deprecating this. We shouldn't deprecate it before we have a replacement.
…ythonGH-97999) Reason: we were too hasty in deprecating this. We shouldn't deprecate it before we have a replacement. (cherry picked from commit 09de8d7) Co-authored-by: Guido van Rossum <[email protected]>
Reason: we were too hasty in deprecating this. We shouldn't deprecate it before we have a replacement. (cherry picked from commit 09de8d7) Co-authored-by: Guido van Rossum <[email protected]>
* main: pythongh-86298: Ensure that __loader__ and __spec__.loader agree in warnings.warn_explicit() (pythonGH-97803) pythongh-82874: Convert remaining importlib format uses to f-str. (python#98005) Docs: Fix backtick errors found by sphinx-lint (python#97998) pythongh-97850: Remove deprecated functions from `importlib.utils` (python#97898) Remove extra spaces in custom openSSL documentation. (python#93568) pythonGH-90985: Revert "Deprecate passing a message into cancel()" (python#97999)
…ython#97999) Reason: we were too hasty in deprecating this. We shouldn't deprecate it before we have a replacement.
Reason: we were too hasty in deprecating this. We shouldn't deprecate it before we have a replacement. (cherry picked from commit 09de8d7) Co-authored-by: Guido van Rossum <[email protected]>
Late to the party, but an alternative to make both party happy might be to let Task.cancel accept an exception object that’s derived from asyncio.CancelledError? |
Could the problem discussed here be solved with exceptions notes? The PEP-678 mentions adding notes to caught exceptions, but I tested that it is possible to create an exception instance with several notes added and raise it (I did not test throwing it into a coroutine though). |
Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.
Show more details
GitHub fields:
bugs.python.org fields:
The text was updated successfully, but these errors were encountered: