Simplify bot loop cleanup routine.

Attempted solution for #2040
This commit is contained in:
Rapptz
2019-04-06 21:27:32 -04:00
parent 7d385b23a1
commit 239d430539

View File

@@ -55,6 +55,46 @@ from .appinfo import AppInfo
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
class _ProperCleanup(Exception):
pass
def _raise_proper_cleanup():
raise _ProperCleanup
def _cancel_tasks(loop, tasks):
if not tasks:
return
log.info('Cleaning up after %d tasks.', len(tasks))
gathered = asyncio.gather(*tasks, loop=loop, return_exceptions=True)
gathered.cancel()
gathered.add_done_callback(lambda fut: loop.stop())
while not gathered.done():
loop.run_forever()
for task in tasks:
if task.cancelled():
continue
if task.exception() is not None:
loop.call_exception_handler({
'message': 'Unhandled exception during Client.run shutdown.',
'exception': task.exception(),
'task': task
})
def _cleanup_loop(loop):
try:
task_retriever = asyncio.Task.all_tasks
except AttributeError:
# future proofing for 3.9 I guess
task_retriever = asyncio.all_tasks
all_tasks = {t for t in task_retriever(loop=loop) if not t.done()}
_cancel_tasks(loop, all_tasks)
if sys.version_info >= (3, 6):
loop.run_until_complete(loop.shutdown_asyncgens())
class Client: class Client:
r"""Represents a client connection that connects to Discord. r"""Represents a client connection that connects to Discord.
This class is used to interact with the Discord WebSocket and API. This class is used to interact with the Discord WebSocket and API.
@@ -463,43 +503,28 @@ class Client:
def _do_cleanup(self): def _do_cleanup(self):
log.info('Cleaning up event loop.') log.info('Cleaning up event loop.')
loop = self.loop loop = self.loop
if loop.is_closed():
return # we're already cleaning up
task = asyncio.ensure_future(self.close(), loop=loop) task = asyncio.ensure_future(self.close(), loop=loop)
def _silence_gathered(fut): def stop_loop(fut):
try: try:
fut.result() fut.result()
except Exception: except asyncio.CancelledError:
pass pass
except Exception as e:
loop.call_exception_handler({
'message': 'Unexpected exception during Client.close',
'exception': e
})
finally: finally:
loop.stop() loop.stop()
def when_future_is_done(fut): task.add_done_callback(stop_loop)
pending = asyncio.Task.all_tasks(loop=loop)
if pending:
log.info('Cleaning up after %s tasks', len(pending))
gathered = asyncio.gather(*pending, loop=loop)
gathered.cancel()
gathered.add_done_callback(_silence_gathered)
else:
loop.stop()
task.add_done_callback(when_future_is_done)
if not loop.is_running():
loop.run_forever()
else:
# on Linux, we're still running because we got triggered via
# the signal handler rather than the natural KeyboardInterrupt
# Since that's the case, we're going to return control after
# registering the task for the event loop to handle later
return None
try: try:
return task.result() # suppress unused task warning loop.run_forever()
except Exception: finally:
return None _cleanup_loop(loop)
loop.close()
def run(self, *args, **kwargs): def run(self, *args, **kwargs):
"""A blocking call that abstracts away the `event loop`_ """A blocking call that abstracts away the `event loop`_
@@ -528,29 +553,15 @@ class Client:
is_windows = sys.platform == 'win32' is_windows = sys.platform == 'win32'
loop = self.loop loop = self.loop
if not is_windows: if not is_windows:
loop.add_signal_handler(signal.SIGINT, self._do_cleanup) loop.add_signal_handler(signal.SIGINT, _raise_proper_cleanup)
loop.add_signal_handler(signal.SIGTERM, self._do_cleanup) loop.add_signal_handler(signal.SIGTERM, _raise_proper_cleanup)
task = asyncio.ensure_future(self.start(*args, **kwargs), loop=loop)
def stop_loop_on_finish(fut):
loop.stop()
task.add_done_callback(stop_loop_on_finish)
try: try:
loop.run_forever() loop.run_until_complete(self.start(*args, **kwargs))
except KeyboardInterrupt: except (_ProperCleanup, KeyboardInterrupt):
log.info('Received signal to terminate bot and event loop.') log.info('Received signal to terminate bot and event loop.')
finally: finally:
task.remove_done_callback(stop_loop_on_finish) self._do_cleanup()
if is_windows:
self._do_cleanup()
loop.close()
if task.cancelled() or not task.done():
return None
return task.result()
# properties # properties