Skip to content

Commit ca9b856

Browse files
revise and document cancellation semantics
in short, cancellable threads always use system tasks. normal threads use the host task, unless passed a token
1 parent eab30c4 commit ca9b856

6 files changed

+105
-97
lines changed

docs/source/reference-core.rst

+12-1
Original file line numberDiff line numberDiff line change
@@ -1823,9 +1823,20 @@ to spawn a child thread, and then use a :ref:`memory channel
18231823

18241824
.. literalinclude:: reference-core/from-thread-example.py
18251825

1826+
.. note::
1827+
1828+
The ``from_thread.run*`` functions reuse the host task that called
1829+
:func:`trio.to_thread.run_sync` to run your provided function in the typical case,
1830+
namely when ``cancellable=False`` so Trio can be sure that the task will always be
1831+
around to perform the work. If you pass ``cancellable=True`` at the outset, or if
1832+
you provide a :class:`~trio.lowlevel.TrioToken` when calling back in to Trio, your
1833+
functions will be executed in a new system task. Therefore, the
1834+
:func:`~trio.lowlevel.current_task`, :func:`current_effective_deadline`, or other
1835+
task-tree specific values may differ depending on keyword argument values.
1836+
18261837
You can also use :func:`trio.from_thread.check_cancelled` to check for cancellation from
18271838
a thread that was spawned by :func:`trio.to_thread.run_sync`. If the call to
1828-
:func:`~trio.to_thread.run_sync` was cancelled, then
1839+
:func:`~trio.to_thread.run_sync` was cancelled (even if ``cancellable=False``!), then
18291840
:func:`~trio.from_thread.check_cancelled` will raise :func:`trio.Cancelled`.
18301841
It's like ``trio.from_thread.run(trio.sleep, 0)``, but much faster.
18311842

trio/_tests/test_threads.py

+12-8
Original file line numberDiff line numberDiff line change
@@ -933,12 +933,14 @@ async def async_time_bomb():
933933
async def test_from_thread_check_cancelled():
934934
q = stdlib_queue.Queue()
935935

936-
async def child(cancellable):
936+
async def child(cancellable, scope):
937+
with scope:
937938
record.append("start")
938939
try:
939940
return await to_thread_run_sync(f, cancellable=cancellable)
940941
except _core.Cancelled:
941942
record.append("cancel")
943+
raise
942944
finally:
943945
record.append("exit")
944946

@@ -956,7 +958,7 @@ def f():
956958
record = []
957959
ev = threading.Event()
958960
async with _core.open_nursery() as nursery:
959-
nursery.start_soon(child, False)
961+
nursery.start_soon(child, False, _core.CancelScope())
960962
await wait_all_tasks_blocked()
961963
assert record[0] == "start"
962964
assert q.get(timeout=1) == "Not Cancelled"
@@ -968,14 +970,15 @@ def f():
968970
# the appropriate cancel scope
969971
record = []
970972
ev = threading.Event()
973+
scope = _core.CancelScope() # Nursery cancel scope gives false positives
971974
async with _core.open_nursery() as nursery:
972-
nursery.start_soon(child, False)
975+
nursery.start_soon(child, False, scope)
973976
await wait_all_tasks_blocked()
974977
assert record[0] == "start"
975978
assert q.get(timeout=1) == "Not Cancelled"
976-
nursery.cancel_scope.cancel()
979+
scope.cancel()
977980
ev.set()
978-
assert nursery.cancel_scope.cancelled_caught
981+
assert scope.cancelled_caught
979982
assert "cancel" in record
980983
assert record[-1] == "exit"
981984

@@ -992,13 +995,14 @@ def f(): # noqa: F811
992995

993996
record = []
994997
ev = threading.Event()
998+
scope = _core.CancelScope()
995999
async with _core.open_nursery() as nursery:
996-
nursery.start_soon(child, True)
1000+
nursery.start_soon(child, True, scope)
9971001
await wait_all_tasks_blocked()
9981002
assert record[0] == "start"
999-
nursery.cancel_scope.cancel()
1003+
scope.cancel()
10001004
ev.set()
1001-
assert nursery.cancel_scope.cancelled_caught
1005+
assert scope.cancelled_caught
10021006
assert "cancel" in record
10031007
assert record[-1] == "exit"
10041008
assert q.get(timeout=1) == "Cancelled"

trio/_tests/verify_types_darwin.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@
4040
],
4141
"exportedSymbolCounts": {
4242
"withAmbiguousType": 0,
43-
"withKnownType": 631,
43+
"withKnownType": 630,
4444
"withUnknownType": 0
4545
},
4646
"ignoreUnknownTypesFromImports": true,

trio/_tests/verify_types_linux.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@
2828
],
2929
"exportedSymbolCounts": {
3030
"withAmbiguousType": 0,
31-
"withKnownType": 628,
31+
"withKnownType": 627,
3232
"withUnknownType": 0
3333
},
3434
"ignoreUnknownTypesFromImports": true,

trio/_tests/verify_types_windows.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,7 @@
6464
],
6565
"exportedSymbolCounts": {
6666
"withAmbiguousType": 0,
67-
"withKnownType": 631,
67+
"withKnownType": 630,
6868
"withUnknownType": 0
6969
},
7070
"ignoreUnknownTypesFromImports": true,

0 commit comments

Comments
 (0)