| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | """A Future class similar to the one in PEP 3148.""" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-10 18:36:12 -05:00
										 |  |  | __all__ = ( | 
					
						
							|  |  |  |     'Future', 'wrap_future', 'isfuture', | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |     'future_add_to_awaited_by', 'future_discard_from_awaited_by', | 
					
						
							| 
									
										
										
										
											2017-12-10 18:36:12 -05:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | import concurrent.futures | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  | import contextvars | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2013-12-19 22:42:40 +01:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2022-01-22 16:58:53 +05:30
										 |  |  | from types import GenericAlias | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | from . import base_futures | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | from . import events | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  | from . import exceptions | 
					
						
							| 
									
										
										
										
											2017-12-15 07:04:38 +02:00
										 |  |  | from . import format_helpers | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | isfuture = base_futures.isfuture | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | _PENDING = base_futures._PENDING | 
					
						
							|  |  |  | _CANCELLED = base_futures._CANCELLED | 
					
						
							|  |  |  | _FINISHED = base_futures._FINISHED | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | STACK_DEBUG = logging.DEBUG - 1  # heavy-duty debugging | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Future: | 
					
						
							|  |  |  |     """This class is *almost* compatible with concurrent.futures.Future.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Differences: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-07 17:03:28 +01:00
										 |  |  |     - This class is not thread-safe. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     - result() and exception() do not take a timeout argument and | 
					
						
							|  |  |  |       raise an exception when the future isn't done yet. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     - Callbacks registered with add_done_callback() are always called | 
					
						
							| 
									
										
										
										
											2017-11-07 17:03:28 +01:00
										 |  |  |       via the event loop's call_soon(). | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     - This class is not compatible with the wait() and as_completed() | 
					
						
							|  |  |  |       methods in the concurrent.futures package. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Class variables serving as defaults for instance variables. | 
					
						
							|  |  |  |     _state = _PENDING | 
					
						
							|  |  |  |     _result = None | 
					
						
							|  |  |  |     _exception = None | 
					
						
							|  |  |  |     _loop = None | 
					
						
							| 
									
										
										
										
											2014-12-04 23:00:13 +01:00
										 |  |  |     _source_traceback = None | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |     _cancel_message = None | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |     # A saved CancelledError for later chaining as an exception context. | 
					
						
							|  |  |  |     _cancelled_exc = None | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 12:54:54 -07:00
										 |  |  |     # This field is used for a dual purpose: | 
					
						
							|  |  |  |     # - Its presence is a marker to declare that a class implements | 
					
						
							|  |  |  |     #   the Future protocol (i.e. is intended to be duck-type compatible). | 
					
						
							|  |  |  |     #   The value must also be not-None, to enable a subclass to declare | 
					
						
							|  |  |  |     #   that it is not compatible by setting this to None. | 
					
						
							| 
									
										
										
										
											2024-07-15 11:29:19 +05:30
										 |  |  |     # - It is set by __iter__() below so that Task.__step() can tell | 
					
						
							| 
									
										
										
										
											2017-12-11 17:35:49 +02:00
										 |  |  |     #   the difference between | 
					
						
							| 
									
										
										
										
											2025-01-15 20:24:31 +08:00
										 |  |  |     #   `await Future()` or `yield from Future()` (correct) vs. | 
					
						
							| 
									
										
										
										
											2016-09-09 12:54:54 -07:00
										 |  |  |     #   `yield Future()` (incorrect). | 
					
						
							|  |  |  |     _asyncio_future_blocking = False | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |     # Used by the capture_call_stack() API. | 
					
						
							|  |  |  |     __asyncio_awaited_by = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |     __log_traceback = False | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, *, loop=None): | 
					
						
							|  |  |  |         """Initialize the future.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-10 05:44:01 +00:00
										 |  |  |         The optional event_loop argument allows explicitly setting the event | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         loop object used by the future. If it's not provided, the future uses | 
					
						
							|  |  |  |         the default event loop. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if loop is None: | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |             self._loop = events.get_event_loop() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self._loop = loop | 
					
						
							|  |  |  |         self._callbacks = [] | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  |         if self._loop.get_debug(): | 
					
						
							| 
									
										
										
										
											2017-12-15 07:04:38 +02:00
										 |  |  |             self._source_traceback = format_helpers.extract_stack( | 
					
						
							|  |  |  |                 sys._getframe(1)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-29 12:58:23 +02:00
										 |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2022-03-17 03:03:09 +02:00
										 |  |  |         return base_futures._future_repr(self) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-25 10:57:18 +09:00
										 |  |  |     def __del__(self): | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |         if not self.__log_traceback: | 
					
						
							| 
									
										
										
										
											2017-04-25 10:57:18 +09:00
										 |  |  |             # set_exception() was not called, or result() or exception() | 
					
						
							|  |  |  |             # has consumed the exception | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         exc = self._exception | 
					
						
							|  |  |  |         context = { | 
					
						
							| 
									
										
										
										
											2017-12-10 18:36:12 -05:00
										 |  |  |             'message': | 
					
						
							|  |  |  |                 f'{self.__class__.__name__} exception was never retrieved', | 
					
						
							| 
									
										
										
										
											2017-04-25 10:57:18 +09:00
										 |  |  |             'exception': exc, | 
					
						
							|  |  |  |             'future': self, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if self._source_traceback: | 
					
						
							|  |  |  |             context['source_traceback'] = self._source_traceback | 
					
						
							|  |  |  |         self._loop.call_exception_handler(context) | 
					
						
							| 
									
										
										
										
											2013-12-19 22:42:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-22 16:58:53 +05:30
										 |  |  |     __class_getitem__ = classmethod(GenericAlias) | 
					
						
							| 
									
										
										
										
											2019-12-07 14:05:07 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def _log_traceback(self): | 
					
						
							|  |  |  |         return self.__log_traceback | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @_log_traceback.setter | 
					
						
							|  |  |  |     def _log_traceback(self, val): | 
					
						
							| 
									
										
										
										
											2020-11-28 20:37:08 +09:00
										 |  |  |         if val: | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |             raise ValueError('_log_traceback can only be set to False') | 
					
						
							|  |  |  |         self.__log_traceback = False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def _asyncio_awaited_by(self): | 
					
						
							|  |  |  |         if self.__asyncio_awaited_by is None: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  |         return frozenset(self.__asyncio_awaited_by) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  |     def get_loop(self): | 
					
						
							|  |  |  |         """Return the event loop the Future is bound to.""" | 
					
						
							| 
									
										
										
										
											2019-11-13 23:36:46 +02:00
										 |  |  |         loop = self._loop | 
					
						
							|  |  |  |         if loop is None: | 
					
						
							|  |  |  |             raise RuntimeError("Future object is not initialized.") | 
					
						
							|  |  |  |         return loop | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |     def _make_cancelled_error(self): | 
					
						
							|  |  |  |         """Create the CancelledError to raise if the Future is cancelled.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This should only be called once when handling a cancellation since | 
					
						
							|  |  |  |         it erases the saved context exception value. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |         if self._cancelled_exc is not None: | 
					
						
							|  |  |  |             exc = self._cancelled_exc | 
					
						
							|  |  |  |             self._cancelled_exc = None | 
					
						
							|  |  |  |             return exc | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |         if self._cancel_message is None: | 
					
						
							|  |  |  |             exc = exceptions.CancelledError() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             exc = exceptions.CancelledError(self._cancel_message) | 
					
						
							|  |  |  |         return exc | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |     def cancel(self, msg=None): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         """Cancel the future and schedule callbacks.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         If the future is already done or cancelled, return False.  Otherwise, | 
					
						
							|  |  |  |         change the future's state to cancelled, schedule the callbacks and | 
					
						
							|  |  |  |         return True. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |         self.__log_traceback = False | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         if self._state != _PENDING: | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         self._state = _CANCELLED | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |         self._cancel_message = msg | 
					
						
							| 
									
										
										
										
											2018-01-24 11:31:01 -05:00
										 |  |  |         self.__schedule_callbacks() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-24 11:31:01 -05:00
										 |  |  |     def __schedule_callbacks(self): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         """Internal: Ask the event loop to call all callbacks.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The callbacks are scheduled to be called as soon as possible. Also | 
					
						
							|  |  |  |         clears the callback list. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         callbacks = self._callbacks[:] | 
					
						
							|  |  |  |         if not callbacks: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._callbacks[:] = [] | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |         for callback, ctx in callbacks: | 
					
						
							|  |  |  |             self._loop.call_soon(callback, self, context=ctx) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def cancelled(self): | 
					
						
							|  |  |  |         """Return True if the future was cancelled.""" | 
					
						
							|  |  |  |         return self._state == _CANCELLED | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Don't implement running(); see http://bugs.python.org/issue18699 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def done(self): | 
					
						
							|  |  |  |         """Return True if the future is done.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Done means either that a result / exception are available, or that the | 
					
						
							|  |  |  |         future was cancelled. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self._state != _PENDING | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def result(self): | 
					
						
							|  |  |  |         """Return the result this future represents.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         If the future has been cancelled, raises CancelledError.  If the | 
					
						
							|  |  |  |         future's result isn't yet available, raises InvalidStateError.  If | 
					
						
							|  |  |  |         the future is done and has an exception set, this exception is raised. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if self._state == _CANCELLED: | 
					
						
							| 
									
										
										
										
											2024-10-14 23:45:58 +01:00
										 |  |  |             raise self._make_cancelled_error() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         if self._state != _FINISHED: | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  |             raise exceptions.InvalidStateError('Result is not ready.') | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |         self.__log_traceback = False | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         if self._exception is not None: | 
					
						
							| 
									
										
										
										
											2022-07-11 18:02:11 +05:30
										 |  |  |             raise self._exception.with_traceback(self._exception_tb) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         return self._result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def exception(self): | 
					
						
							|  |  |  |         """Return the exception that was set on this future.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The exception (or None if no exception was set) is returned only if | 
					
						
							|  |  |  |         the future is done.  If the future has been cancelled, raises | 
					
						
							|  |  |  |         CancelledError.  If the future isn't done yet, raises | 
					
						
							|  |  |  |         InvalidStateError. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if self._state == _CANCELLED: | 
					
						
							| 
									
										
										
										
											2024-10-14 23:45:58 +01:00
										 |  |  |             raise self._make_cancelled_error() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         if self._state != _FINISHED: | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  |             raise exceptions.InvalidStateError('Exception is not set.') | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |         self.__log_traceback = False | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         return self._exception | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |     def add_done_callback(self, fn, *, context=None): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         """Add a callback to be run when the future becomes done.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The callback is called with a single argument - the future object. If | 
					
						
							|  |  |  |         the future is already done when this is called, the callback is | 
					
						
							|  |  |  |         scheduled with call_soon. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if self._state != _PENDING: | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |             self._loop.call_soon(fn, self, context=context) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |             if context is None: | 
					
						
							|  |  |  |                 context = contextvars.copy_context() | 
					
						
							|  |  |  |             self._callbacks.append((fn, context)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # New method not in PEP 3148. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def remove_done_callback(self, fn): | 
					
						
							|  |  |  |         """Remove all instances of a callback from the "call when done" list.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Returns the number of callbacks removed. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |         filtered_callbacks = [(f, ctx) | 
					
						
							|  |  |  |                               for (f, ctx) in self._callbacks | 
					
						
							|  |  |  |                               if f != fn] | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         removed_count = len(self._callbacks) - len(filtered_callbacks) | 
					
						
							|  |  |  |         if removed_count: | 
					
						
							|  |  |  |             self._callbacks[:] = filtered_callbacks | 
					
						
							|  |  |  |         return removed_count | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # So-called internal methods (note: no set_running_or_notify_cancel()). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_result(self, result): | 
					
						
							|  |  |  |         """Mark the future done and set its result.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         If the future is already done when this method is called, raises | 
					
						
							|  |  |  |         InvalidStateError. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if self._state != _PENDING: | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  |             raise exceptions.InvalidStateError(f'{self._state}: {self!r}') | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self._result = result | 
					
						
							|  |  |  |         self._state = _FINISHED | 
					
						
							| 
									
										
										
										
											2018-01-24 11:31:01 -05:00
										 |  |  |         self.__schedule_callbacks() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def set_exception(self, exception): | 
					
						
							|  |  |  |         """Mark the future done and set an exception.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         If the future is already done when this method is called, raises | 
					
						
							|  |  |  |         InvalidStateError. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if self._state != _PENDING: | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  |             raise exceptions.InvalidStateError(f'{self._state}: {self!r}') | 
					
						
							| 
									
										
										
										
											2014-01-30 16:01:54 -08:00
										 |  |  |         if isinstance(exception, type): | 
					
						
							|  |  |  |             exception = exception() | 
					
						
							| 
									
										
										
										
											2024-01-10 16:21:00 +11:00
										 |  |  |         if isinstance(exception, StopIteration): | 
					
						
							|  |  |  |             new_exc = RuntimeError("StopIteration interacts badly with " | 
					
						
							|  |  |  |                                    "generators and cannot be raised into a " | 
					
						
							|  |  |  |                                    "Future") | 
					
						
							|  |  |  |             new_exc.__cause__ = exception | 
					
						
							|  |  |  |             new_exc.__context__ = exception | 
					
						
							|  |  |  |             exception = new_exc | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self._exception = exception | 
					
						
							| 
									
										
										
										
											2022-07-11 18:02:11 +05:30
										 |  |  |         self._exception_tb = exception.__traceback__ | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self._state = _FINISHED | 
					
						
							| 
									
										
										
										
											2018-01-24 11:31:01 -05:00
										 |  |  |         self.__schedule_callbacks() | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |         self.__log_traceback = True | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     def __await__(self): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         if not self.done(): | 
					
						
							| 
									
										
										
										
											2016-09-09 12:54:54 -07:00
										 |  |  |             self._asyncio_future_blocking = True | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             yield self  # This tells Task to wait for completion. | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |         if not self.done(): | 
					
						
							|  |  |  |             raise RuntimeError("await wasn't used with future") | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         return self.result()  # May raise too. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     __iter__ = __await__  # make compatible with 'yield from'. | 
					
						
							| 
									
										
										
										
											2015-05-11 22:27:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-23 22:34:35 -04:00
										 |  |  | # Needed for testing purposes. | 
					
						
							|  |  |  | _PyFuture = Future | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  | def _get_loop(fut): | 
					
						
							|  |  |  |     # Tries to call Future.get_loop() if it's available. | 
					
						
							|  |  |  |     # Otherwise fallbacks to using the old '_loop' property. | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         get_loop = fut.get_loop | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return get_loop() | 
					
						
							|  |  |  |     return fut._loop | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-17 12:19:41 -05:00
										 |  |  | def _set_result_unless_cancelled(fut, result): | 
					
						
							|  |  |  |     """Helper setting the result only if the future was not cancelled.""" | 
					
						
							|  |  |  |     if fut.cancelled(): | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     fut.set_result(result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  | def _convert_future_exc(exc): | 
					
						
							|  |  |  |     exc_class = type(exc) | 
					
						
							|  |  |  |     if exc_class is concurrent.futures.CancelledError: | 
					
						
							| 
									
										
										
										
											2024-04-04 11:13:32 +08:00
										 |  |  |         return exceptions.CancelledError(*exc.args).with_traceback(exc.__traceback__) | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  |     elif exc_class is concurrent.futures.InvalidStateError: | 
					
						
							| 
									
										
										
										
											2024-04-04 11:13:32 +08:00
										 |  |  |         return exceptions.InvalidStateError(*exc.args).with_traceback(exc.__traceback__) | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  |     else: | 
					
						
							|  |  |  |         return exc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  | def _set_concurrent_future_state(concurrent, source): | 
					
						
							|  |  |  |     """Copy state from a future to a concurrent.futures.Future.""" | 
					
						
							|  |  |  |     assert source.done() | 
					
						
							|  |  |  |     if source.cancelled(): | 
					
						
							|  |  |  |         concurrent.cancel() | 
					
						
							|  |  |  |     if not concurrent.set_running_or_notify_cancel(): | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     exception = source.exception() | 
					
						
							|  |  |  |     if exception is not None: | 
					
						
							| 
									
										
										
										
											2018-09-11 10:13:04 -07:00
										 |  |  |         concurrent.set_exception(_convert_future_exc(exception)) | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |     else: | 
					
						
							|  |  |  |         result = source.result() | 
					
						
							|  |  |  |         concurrent.set_result(result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-17 12:19:41 -05:00
										 |  |  | def _copy_future_state(source, dest): | 
					
						
							|  |  |  |     """Internal helper to copy state from another Future.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-18 11:56:20 -04:00
										 |  |  |     The other Future must be a concurrent.futures.Future. | 
					
						
							| 
									
										
										
										
											2015-11-17 12:19:41 -05:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     if dest.cancelled(): | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     assert not dest.done() | 
					
						
							| 
									
										
										
										
											2025-05-18 11:56:20 -04:00
										 |  |  |     done, cancelled, result, exception = source._get_snapshot() | 
					
						
							|  |  |  |     assert done | 
					
						
							|  |  |  |     if cancelled: | 
					
						
							| 
									
										
										
										
											2015-11-17 12:19:41 -05:00
										 |  |  |         dest.cancel() | 
					
						
							| 
									
										
										
										
											2025-05-18 11:56:20 -04:00
										 |  |  |     elif exception is not None: | 
					
						
							|  |  |  |         dest.set_exception(_convert_future_exc(exception)) | 
					
						
							| 
									
										
										
										
											2015-11-17 12:19:41 -05:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2025-05-18 11:56:20 -04:00
										 |  |  |         dest.set_result(result) | 
					
						
							| 
									
										
										
										
											2015-11-17 12:19:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  | def _chain_future(source, destination): | 
					
						
							|  |  |  |     """Chain two futures so that when one completes, so does the other.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     The result (or exception) of source will be copied to destination. | 
					
						
							|  |  |  |     If destination is cancelled, source gets cancelled too. | 
					
						
							|  |  |  |     Compatible with both asyncio.Future and concurrent.futures.Future. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2016-09-09 14:26:31 -07:00
										 |  |  |     if not isfuture(source) and not isinstance(source, | 
					
						
							|  |  |  |                                                concurrent.futures.Future): | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         raise TypeError('A future is required for source argument') | 
					
						
							| 
									
										
										
										
											2016-09-09 14:26:31 -07:00
										 |  |  |     if not isfuture(destination) and not isinstance(destination, | 
					
						
							|  |  |  |                                                     concurrent.futures.Future): | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         raise TypeError('A future is required for destination argument') | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  |     source_loop = _get_loop(source) if isfuture(source) else None | 
					
						
							|  |  |  |     dest_loop = _get_loop(destination) if isfuture(destination) else None | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _set_state(future, other): | 
					
						
							| 
									
										
										
										
											2016-09-09 14:26:31 -07:00
										 |  |  |         if isfuture(future): | 
					
						
							| 
									
										
										
										
											2015-11-17 12:19:41 -05:00
										 |  |  |             _copy_future_state(other, future) | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         else: | 
					
						
							|  |  |  |             _set_concurrent_future_state(future, other) | 
					
						
							| 
									
										
										
										
											2013-11-22 11:47:22 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |     def _call_check_cancel(destination): | 
					
						
							|  |  |  |         if destination.cancelled(): | 
					
						
							|  |  |  |             if source_loop is None or source_loop is dest_loop: | 
					
						
							|  |  |  |                 source.cancel() | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 source_loop.call_soon_threadsafe(source.cancel) | 
					
						
							| 
									
										
										
										
											2013-11-22 11:47:22 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |     def _call_set_state(source): | 
					
						
							| 
									
										
										
										
											2018-05-28 17:10:20 -04:00
										 |  |  |         if (destination.cancelled() and | 
					
						
							|  |  |  |                 dest_loop is not None and dest_loop.is_closed()): | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         if dest_loop is None or dest_loop is source_loop: | 
					
						
							|  |  |  |             _set_state(destination, source) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2022-09-30 12:55:40 -07:00
										 |  |  |             if dest_loop.is_closed(): | 
					
						
							|  |  |  |                 return | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |             dest_loop.call_soon_threadsafe(_set_state, destination, source) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     destination.add_done_callback(_call_check_cancel) | 
					
						
							|  |  |  |     source.add_done_callback(_call_set_state) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def wrap_future(future, *, loop=None): | 
					
						
							|  |  |  |     """Wrap concurrent.futures.Future object.""" | 
					
						
							| 
									
										
										
										
											2016-09-09 14:26:31 -07:00
										 |  |  |     if isfuture(future): | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         return future | 
					
						
							|  |  |  |     assert isinstance(future, concurrent.futures.Future), \ | 
					
						
							| 
									
										
										
										
											2017-12-10 18:36:12 -05:00
										 |  |  |         f'concurrent.futures.Future is expected, got {future!r}' | 
					
						
							| 
									
										
										
										
											2016-05-16 15:38:39 -04:00
										 |  |  |     if loop is None: | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         loop = events.get_event_loop() | 
					
						
							| 
									
										
										
										
											2016-05-16 15:38:39 -04:00
										 |  |  |     new_future = loop.create_future() | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |     _chain_future(future, new_future) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     return new_future | 
					
						
							| 
									
										
										
										
											2016-10-18 11:48:14 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | def future_add_to_awaited_by(fut, waiter, /): | 
					
						
							|  |  |  |     """Record that `fut` is awaited on by `waiter`.""" | 
					
						
							|  |  |  |     # For the sake of keeping the implementation minimal and assuming | 
					
						
							|  |  |  |     # that most of asyncio users use the built-in Futures and Tasks | 
					
						
							|  |  |  |     # (or their subclasses), we only support native Future objects | 
					
						
							|  |  |  |     # and their subclasses. | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     # Longer version: tracking requires storing the caller-callee | 
					
						
							|  |  |  |     # dependency somewhere. One obvious choice is to store that | 
					
						
							|  |  |  |     # information right in the future itself in a dedicated attribute. | 
					
						
							|  |  |  |     # This means that we'd have to require all duck-type compatible | 
					
						
							|  |  |  |     # futures to implement a specific attribute used by asyncio for | 
					
						
							|  |  |  |     # the book keeping. Another solution would be to store that in | 
					
						
							|  |  |  |     # a global dictionary. The downside here is that that would create | 
					
						
							|  |  |  |     # strong references and any scenario where the "add" call isn't | 
					
						
							|  |  |  |     # followed by a "discard" call would lead to a memory leak. | 
					
						
							|  |  |  |     # Using WeakDict would resolve that issue, but would complicate | 
					
						
							|  |  |  |     # the C code (_asynciomodule.c). The bottom line here is that | 
					
						
							|  |  |  |     # it's not clear that all this work would be worth the effort. | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     # Note that there's an accelerated version of this function | 
					
						
							|  |  |  |     # shadowing this implementation later in this file. | 
					
						
							|  |  |  |     if isinstance(fut, _PyFuture) and isinstance(waiter, _PyFuture): | 
					
						
							|  |  |  |         if fut._Future__asyncio_awaited_by is None: | 
					
						
							|  |  |  |             fut._Future__asyncio_awaited_by = set() | 
					
						
							|  |  |  |         fut._Future__asyncio_awaited_by.add(waiter) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def future_discard_from_awaited_by(fut, waiter, /): | 
					
						
							|  |  |  |     """Record that `fut` is no longer awaited on by `waiter`.""" | 
					
						
							|  |  |  |     # See the comment in "future_add_to_awaited_by()" body for | 
					
						
							|  |  |  |     # details on implementation. | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     # Note that there's an accelerated version of this function | 
					
						
							|  |  |  |     # shadowing this implementation later in this file. | 
					
						
							|  |  |  |     if isinstance(fut, _PyFuture) and isinstance(waiter, _PyFuture): | 
					
						
							|  |  |  |         if fut._Future__asyncio_awaited_by is not None: | 
					
						
							|  |  |  |             fut._Future__asyncio_awaited_by.discard(waiter) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _py_future_add_to_awaited_by = future_add_to_awaited_by | 
					
						
							|  |  |  | _py_future_discard_from_awaited_by = future_discard_from_awaited_by | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-18 11:48:14 +09:00
										 |  |  | try: | 
					
						
							|  |  |  |     import _asyncio | 
					
						
							|  |  |  | except ImportError: | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | else: | 
					
						
							| 
									
										
										
										
											2016-10-23 22:34:35 -04:00
										 |  |  |     # _CFuture is needed for tests. | 
					
						
							|  |  |  |     Future = _CFuture = _asyncio.Future | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |     future_add_to_awaited_by = _asyncio.future_add_to_awaited_by | 
					
						
							|  |  |  |     future_discard_from_awaited_by = _asyncio.future_discard_from_awaited_by | 
					
						
							|  |  |  |     _c_future_add_to_awaited_by = future_add_to_awaited_by | 
					
						
							|  |  |  |     _c_future_discard_from_awaited_by = future_discard_from_awaited_by |