diff options
author | Guido van Rossum <guido@python.org> | 2015-11-19 21:28:47 (GMT) |
---|---|---|
committer | Guido van Rossum <guido@python.org> | 2015-11-19 21:28:47 (GMT) |
commit | 41f69f4cc7c977bd202545b8ada01b80a278d0e2 (patch) | |
tree | 10b1e25f8802dd9b5469acf5002c69d6996400c1 /Lib/asyncio | |
parent | 01a65af4a150a9a81cd92923adef76810e41895a (diff) | |
download | cpython-41f69f4cc7c977bd202545b8ada01b80a278d0e2.zip cpython-41f69f4cc7c977bd202545b8ada01b80a278d0e2.tar.gz cpython-41f69f4cc7c977bd202545b8ada01b80a278d0e2.tar.bz2 |
Issue #25593: Change semantics of EventLoop.stop().
Diffstat (limited to 'Lib/asyncio')
-rw-r--r-- | Lib/asyncio/base_events.py | 25 | ||||
-rw-r--r-- | Lib/asyncio/test_utils.py | 11 |
2 files changed, 15 insertions, 21 deletions
diff --git a/Lib/asyncio/base_events.py b/Lib/asyncio/base_events.py index af9c881..c5ffad4 100644 --- a/Lib/asyncio/base_events.py +++ b/Lib/asyncio/base_events.py @@ -70,10 +70,6 @@ def _format_pipe(fd): return repr(fd) -class _StopError(BaseException): - """Raised to stop the event loop.""" - - def _check_resolved_address(sock, address): # Ensure that the address is already resolved to avoid the trap of hanging # the entire event loop when the address requires doing a DNS lookup. @@ -118,9 +114,6 @@ def _check_resolved_address(sock, address): "got host %r: %s" % (host, err)) -def _raise_stop_error(*args): - raise _StopError - def _run_until_complete_cb(fut): exc = fut._exception @@ -129,7 +122,7 @@ def _run_until_complete_cb(fut): # Issue #22429: run_forever() already finished, no need to # stop it. return - _raise_stop_error() + fut._loop.stop() class Server(events.AbstractServer): @@ -184,6 +177,7 @@ class BaseEventLoop(events.AbstractEventLoop): def __init__(self): self._timer_cancelled_count = 0 self._closed = False + self._stopping = False self._ready = collections.deque() self._scheduled = [] self._default_executor = None @@ -298,11 +292,11 @@ class BaseEventLoop(events.AbstractEventLoop): self._thread_id = threading.get_ident() try: while True: - try: - self._run_once() - except _StopError: + self._run_once() + if self._stopping: break finally: + self._stopping = False self._thread_id = None self._set_coroutine_wrapper(False) @@ -345,11 +339,10 @@ class BaseEventLoop(events.AbstractEventLoop): def stop(self): """Stop running the event loop. - Every callback scheduled before stop() is called will run. Callbacks - scheduled after stop() is called will not run. However, those callbacks - will run if run_forever is called again later. + Every callback already scheduled will still run. This simply informs + run_forever to stop looping after a complete iteration. """ - self.call_soon(_raise_stop_error) + self._stopping = True def close(self): """Close the event loop. @@ -1194,7 +1187,7 @@ class BaseEventLoop(events.AbstractEventLoop): handle._scheduled = False timeout = None - if self._ready: + if self._ready or self._stopping: timeout = 0 elif self._scheduled: # Compute the desired timeout. diff --git a/Lib/asyncio/test_utils.py b/Lib/asyncio/test_utils.py index 8cee95b..e06ac06 100644 --- a/Lib/asyncio/test_utils.py +++ b/Lib/asyncio/test_utils.py @@ -71,12 +71,13 @@ def run_until(loop, pred, timeout=30): def run_once(loop): - """loop.stop() schedules _raise_stop_error() - and run_forever() runs until _raise_stop_error() callback. - this wont work if test waits for some IO events, because - _raise_stop_error() runs before any of io events callbacks. + """Legacy API to run once through the event loop. + + This is the recommended pattern for test code. It will poll the + selector once and run all callbacks scheduled in response to I/O + events. """ - loop.stop() + loop.call_soon(loop.stop) loop.run_forever() |