From e37ea1b3a04e9d3cfd8d125f5c2af4ebb7db3eca Mon Sep 17 00:00:00 2001 From: richardsheridan Date: Sat, 28 Oct 2023 13:44:46 -0400 Subject: [PATCH] add abandon_on_cancel to to_thread_run_sync: new name for cancellable flag Replaced in code and docs --- docs/source/reference-core.rst | 6 +++--- trio/_socket.py | 6 +++--- trio/_subprocess_platform/waitid.py | 4 ++-- trio/_tests/test_threads.py | 29 ++++++++++++++++------------- trio/_threads.py | 21 +++++++++++++-------- trio/_wait_for_object.py | 2 +- 6 files changed, 38 insertions(+), 30 deletions(-) diff --git a/docs/source/reference-core.rst b/docs/source/reference-core.rst index 333cb0537b..98cf579399 100644 --- a/docs/source/reference-core.rst +++ b/docs/source/reference-core.rst @@ -1827,8 +1827,8 @@ to spawn a child thread, and then use a :ref:`memory channel The ``from_thread.run*`` functions reuse the host task that called :func:`trio.to_thread.run_sync` to run your provided function, as long as you're - using the default ``cancellable=False`` so Trio can be sure that the task will remain - around to perform the work. If you pass ``cancellable=True`` at the outset, or if + using the default ``abandon_on_cancel=False`` so Trio can be sure that the task will remain + around to perform the work. If you pass ``abandon_on_cancel=True`` at the outset, or if you provide a :class:`~trio.lowlevel.TrioToken` when calling back in to Trio, your functions will be executed in a new system task. Therefore, the :func:`~trio.lowlevel.current_task`, :func:`current_effective_deadline`, or other @@ -1836,7 +1836,7 @@ to spawn a child thread, and then use a :ref:`memory channel You can also use :func:`trio.from_thread.check_cancelled` to check for cancellation from a thread that was spawned by :func:`trio.to_thread.run_sync`. If the call to -:func:`~trio.to_thread.run_sync` was cancelled (even if ``cancellable=False``!), then +:func:`~trio.to_thread.run_sync` was cancelled (even if ``abandon_on_cancel=False``!), then :func:`~trio.from_thread.check_cancelled` will raise :func:`trio.Cancelled`. It's like ``trio.from_thread.run(trio.sleep, 0)``, but much faster. diff --git a/trio/_socket.py b/trio/_socket.py index efef2ff5ed..061cc09367 100644 --- a/trio/_socket.py +++ b/trio/_socket.py @@ -240,7 +240,7 @@ def numeric_only_failure(exc: BaseException) -> bool: type, proto, flags, - cancellable=True, + abandon_on_cancel=True, ) @@ -261,7 +261,7 @@ async def getnameinfo( return await hr.getnameinfo(sockaddr, flags) else: return await trio.to_thread.run_sync( - _stdlib_socket.getnameinfo, sockaddr, flags, cancellable=True + _stdlib_socket.getnameinfo, sockaddr, flags, abandon_on_cancel=True ) @@ -272,7 +272,7 @@ async def getprotobyname(name: str) -> int: """ return await trio.to_thread.run_sync( - _stdlib_socket.getprotobyname, name, cancellable=True + _stdlib_socket.getprotobyname, name, abandon_on_cancel=True ) diff --git a/trio/_subprocess_platform/waitid.py b/trio/_subprocess_platform/waitid.py index 7d941747e3..165e946d21 100644 --- a/trio/_subprocess_platform/waitid.py +++ b/trio/_subprocess_platform/waitid.py @@ -73,14 +73,14 @@ async def _waitid_system_task(pid: int, event: Event) -> None: """Spawn a thread that waits for ``pid`` to exit, then wake any tasks that were waiting on it. """ - # cancellable=True: if this task is cancelled, then we abandon the + # abandon_on_cancel=True: if this task is cancelled, then we abandon the # thread to keep running waitpid in the background. Since this is # always run as a system task, this will only happen if the whole # call to trio.run is shutting down. try: await to_thread_run_sync( - sync_wait_reapable, pid, cancellable=True, limiter=waitid_limiter + sync_wait_reapable, pid, abandon_on_cancel=True, limiter=waitid_limiter ) except OSError: # If waitid fails, waitpid will fail too, so it still makes diff --git a/trio/_tests/test_threads.py b/trio/_tests/test_threads.py index ff865d2ef0..34e71dcf6d 100644 --- a/trio/_tests/test_threads.py +++ b/trio/_tests/test_threads.py @@ -315,10 +315,10 @@ def f(q): q.get() register[0] = "finished" - async def child(q, cancellable): + async def child(q, abandon_on_cancel): record.append("start") try: - return await to_thread_run_sync(f, q, cancellable=cancellable) + return await to_thread_run_sync(f, q, abandon_on_cancel=abandon_on_cancel) finally: record.append("exit") @@ -378,7 +378,7 @@ def thread_fn(): async def main(): async def child(): - await to_thread_run_sync(thread_fn, cancellable=True) + await to_thread_run_sync(thread_fn, abandon_on_cancel=True) async with _core.open_nursery() as nursery: nursery.start_soon(child) @@ -462,7 +462,10 @@ def thread_fn(cancel_scope): async def run_thread(event): with _core.CancelScope() as cancel_scope: await to_thread_run_sync( - thread_fn, cancel_scope, limiter=limiter_arg, cancellable=cancel + thread_fn, + cancel_scope, + abandon_on_cancel=cancel, + limiter=limiter_arg, ) print("run_thread finished, cancelled:", cancel_scope.cancelled_caught) event.set() @@ -847,7 +850,7 @@ async def test_trio_token_weak_referenceable(): assert token is weak_reference() -async def test_unsafe_cancellable_kwarg(): +async def test_unsafe_abandon_on_cancel_kwarg(): # This is a stand in for a numpy ndarray or other objects # that (maybe surprisingly) lack a notion of truthiness class BadBool: @@ -855,7 +858,7 @@ def __bool__(self): raise NotImplementedError with pytest.raises(NotImplementedError): - await to_thread_run_sync(int, cancellable=BadBool()) + await to_thread_run_sync(int, abandon_on_cancel=BadBool()) async def test_from_thread_reuses_task(): @@ -898,7 +901,7 @@ def sync_check(): assert not queue.get_nowait() with _core.CancelScope() as cancel_scope: - await to_thread_run_sync(sync_check, cancellable=True) + await to_thread_run_sync(sync_check, abandon_on_cancel=True) assert cancel_scope.cancelled_caught assert not await to_thread_run_sync(partial(queue.get, timeout=1)) @@ -922,7 +925,7 @@ def async_check(): assert not queue.get_nowait() with _core.CancelScope() as cancel_scope: - await to_thread_run_sync(async_check, cancellable=True) + await to_thread_run_sync(async_check, abandon_on_cancel=True) assert cancel_scope.cancelled_caught assert not await to_thread_run_sync(partial(queue.get, timeout=1)) @@ -941,11 +944,11 @@ async def async_time_bomb(): async def test_from_thread_check_cancelled(): q = stdlib_queue.Queue() - async def child(cancellable, scope): + async def child(abandon_on_cancel, scope): with scope: record.append("start") try: - return await to_thread_run_sync(f, cancellable=cancellable) + return await to_thread_run_sync(f, abandon_on_cancel=abandon_on_cancel) except _core.Cancelled: record.append("cancel") raise @@ -974,7 +977,7 @@ def f(): # implicit assertion, Cancelled not raised via nursery assert record[1] == "exit" - # cancellable=False case: a cancel will pop out but be handled by + # abandon_on_cancel=False case: a cancel will pop out but be handled by # the appropriate cancel scope record = [] ev = threading.Event() @@ -990,7 +993,7 @@ def f(): assert "cancel" in record assert record[-1] == "exit" - # cancellable=True case: slightly different thread behavior needed + # abandon_on_cancel=True case: slightly different thread behavior needed # check thread is cancelled "soon" after abandonment def f(): # noqa: F811 ev.wait() @@ -1033,7 +1036,7 @@ async def test_reentry_doesnt_deadlock(): async def child() -> None: while True: - await to_thread_run_sync(from_thread_run, sleep, 0, cancellable=False) + await to_thread_run_sync(from_thread_run, sleep, 0, abandon_on_cancel=False) with move_on_after(2): async with _core.open_nursery() as nursery: diff --git a/trio/_threads.py b/trio/_threads.py index 023b1c198e..ecfcf9e258 100644 --- a/trio/_threads.py +++ b/trio/_threads.py @@ -175,6 +175,7 @@ async def to_thread_run_sync( # type: ignore[misc] sync_fn: Callable[..., RetT], *args: object, thread_name: str | None = None, + abandon_on_cancel: bool | None = None, cancellable: bool = False, limiter: CapacityLimiter | None = None, ) -> RetT: @@ -197,8 +198,10 @@ async def to_thread_run_sync( # type: ignore[misc] sync_fn: An arbitrary synchronous callable. *args: Positional arguments to pass to sync_fn. If you need keyword arguments, use :func:`functools.partial`. - cancellable (bool): Whether to allow cancellation of this operation. See - discussion below. + abandon_on_cancel (bool): Whether to abandon this thread upon + cancellation of this operation. See discussion below. + cancellable (bool): *Deprecated* synonym for ``abandon_on_cancel``. + Providing a value to ``abandon_on_cancel`` overrides this argument. thread_name (str): Optional string to set the name of the thread. Will always set `threading.Thread.name`, but only set the os name if pthread.h is available (i.e. most POSIX installations). @@ -224,17 +227,17 @@ async def to_thread_run_sync( # type: ignore[misc] starting the thread. But once the thread is running, there are two ways it can handle being cancelled: - * If ``cancellable=False``, the function ignores the cancellation and + * If ``abandon_on_cancel=False``, the function ignores the cancellation and keeps going, just like if we had called ``sync_fn`` synchronously. This is the default behavior. - * If ``cancellable=True``, then this function immediately raises + * If ``abandon_on_cancel=True``, then this function immediately raises `~trio.Cancelled`. In this case **the thread keeps running in background** – we just abandon it to do whatever it's going to do, and silently discard any return value or errors that it raises. Only use this if you know that the operation is safe and side-effect free. (For example: :func:`trio.socket.getaddrinfo` uses a thread with - ``cancellable=True``, because it doesn't really affect anything if a + ``abandon_on_cancel=True``, because it doesn't really affect anything if a stray hostname lookup keeps running in the background.) The ``limiter`` is only released after the thread has *actually* @@ -262,6 +265,8 @@ async def to_thread_run_sync( # type: ignore[misc] """ await trio.lowlevel.checkpoint_if_cancelled() + if abandon_on_cancel is not None: + cancellable = abandon_on_cancel abandon_on_cancel = bool(cancellable) # raise early if cancellable.__bool__ raises if limiter is None: limiter = current_default_thread_limiter() @@ -382,14 +387,14 @@ def from_thread_check_cancelled() -> None: """Raise `trio.Cancelled` if the associated Trio task entered a cancelled status. Only applicable to threads spawned by `trio.to_thread.run_sync`. Poll to allow - ``cancellable=False`` threads to raise :exc:`~trio.Cancelled` at a suitable - place, or to end abandoned ``cancellable=True`` threads sooner than they may + ``abandon_on_cancel=False`` threads to raise :exc:`~trio.Cancelled` at a suitable + place, or to end abandoned ``abandon_on_cancel=True`` threads sooner than they may otherwise. Raises: Cancelled: If the corresponding call to `trio.to_thread.run_sync` has had a delivery of cancellation attempted against it, regardless of the value of - ``cancellable`` supplied as an argument to it. + ``abandon_on_cancel`` supplied as an argument to it. RuntimeError: If this thread is not spawned from `trio.to_thread.run_sync`. .. note:: diff --git a/trio/_wait_for_object.py b/trio/_wait_for_object.py index d2193d9c86..53832513a3 100644 --- a/trio/_wait_for_object.py +++ b/trio/_wait_for_object.py @@ -45,7 +45,7 @@ async def WaitForSingleObject(obj: int | CData) -> None: WaitForMultipleObjects_sync, handle, cancel_handle, - cancellable=True, + abandon_on_cancel=True, limiter=trio.CapacityLimiter(math.inf), ) finally: