diff --git a/trio/_core/_run.py b/trio/_core/_run.py index 7251f7970c..8f93a77967 100644 --- a/trio/_core/_run.py +++ b/trio/_core/_run.py @@ -168,11 +168,13 @@ class CancelScope: _has_been_entered = attr.ib(default=False, init=False) _effective_deadline = attr.ib(default=inf, init=False) cancel_called = attr.ib(default=False, init=False) + graceful_cancel_called = attr.ib(default=False, init=False) cancelled_caught = attr.ib(default=False, init=False) # Constructor arguments: _deadline = attr.ib(default=inf, kw_only=True) _shield = attr.ib(default=False, kw_only=True) + _graceful = attr.ib(default=None, kw_only=True) @enable_ki_protection def __enter__(self): @@ -325,6 +327,19 @@ def shield(self, new_value): for task in self._tasks: task._attempt_delivery_of_any_pending_cancel() + @property + def graceful(self): + """If this is set to :data:`True`, then the code inside this + scope will receive :exc:`~trio.Cancelled` exceptions from scopes + that are outside this scope that have been gracefully canceled. + If this is set to :data:`None` then this scope will inherit its + graceful cancel behavior from its parent scope. + + Defaults to :data:`None`, though this can be overridden by the + ``graceful=`` argument to the :class:`~trio.CancelScope` constructor. + """ + return self._graceful + @enable_ki_protection def cancel(self): """Cancels this scope immediately. @@ -339,6 +354,27 @@ def cancel(self): for task in self._tasks: task._attempt_delivery_of_any_pending_cancel() + @enable_ki_protection + def graceful_cancel(self, seconds=None): + """Gracefully cancel this scope immediately. + + Args: + seconds (float): Optional. If provided, this will change the deadline + of the current scope to be the number of seconds in the future. + + Raises: + ValueError: if *seconds* is negative. + """ + if self.cancel_called or self.graceful_cancel_called: + return + if seconds is not None: + if seconds < 0: + raise ValueError("new deadline must be in the future") + self.deadline = current_time() + float(seconds) + self.graceful_cancel_called = True + for task in self._tasks: + task._attempt_delivery_of_any_pending_cancel() + def _add_task(self, task): self._tasks.add(task) task._cancel_stack.append(self) @@ -365,10 +401,7 @@ def _exc_filter(self, exc): def _close(self, exc): scope_task = current_task() - if ( - exc is not None and self.cancel_called - and scope_task._pending_cancel_scope() is self - ): + if exc is not None and scope_task._pending_cancel_scope() is self: exc = MultiError.filter(self._exc_filter, exc) with self._might_change_effective_deadline(): self._remove_task(scope_task) @@ -643,6 +676,30 @@ def __del__(self): ################################################################ +def _pending_graceful_cancel_scope(cancel_stack): + # Return the outermost scope that is graceful, graceful_cancelled, and not outside a shield + pending_scope = None + graceful_cancel_called = False + scope_is_graceful = False + + for scope in cancel_stack: + if scope.shield: + pending_scope = None + graceful_cancel_called = False + scope_is_graceful = False + if scope.graceful is False: # None implies inherit + pending_scope = None + scope_is_graceful = False + if scope.graceful: + scope_is_graceful = True + if scope.graceful_cancel_called: + graceful_cancel_called = True + if pending_scope is None and scope_is_graceful and graceful_cancel_called: + pending_scope = scope + + return pending_scope + + def _pending_cancel_scope(cancel_stack): # Return the outermost exception that is is not outside a shield. pending_scope = None @@ -653,6 +710,10 @@ def _pending_cancel_scope(cancel_stack): pending_scope = None if pending_scope is None and scope.cancel_called: pending_scope = scope + + if pending_scope is None: + pending_scope = _pending_graceful_cancel_scope(cancel_stack) + return pending_scope