mirror of
				https://github.com/Rapptz/discord.py.git
				synced 2025-10-24 18:13:00 +00:00 
			
		
		
		
	| @@ -55,6 +55,46 @@ from .appinfo import AppInfo | ||||
|  | ||||
| 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: | ||||
|     r"""Represents a client connection that connects to Discord. | ||||
|     This class is used to interact with the Discord WebSocket and API. | ||||
| @@ -463,43 +503,28 @@ class Client: | ||||
|     def _do_cleanup(self): | ||||
|         log.info('Cleaning up event loop.') | ||||
|         loop = self.loop | ||||
|         if loop.is_closed(): | ||||
|             return # we're already cleaning up | ||||
|  | ||||
|         task = asyncio.ensure_future(self.close(), loop=loop) | ||||
|  | ||||
|         def _silence_gathered(fut): | ||||
|         def stop_loop(fut): | ||||
|             try: | ||||
|                 fut.result() | ||||
|             except Exception: | ||||
|             except asyncio.CancelledError: | ||||
|                 pass | ||||
|             except Exception as e: | ||||
|                 loop.call_exception_handler({ | ||||
|                     'message': 'Unexpected exception during Client.close', | ||||
|                     'exception': e | ||||
|                 }) | ||||
|             finally: | ||||
|                 loop.stop() | ||||
|  | ||||
|         def when_future_is_done(fut): | ||||
|             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 | ||||
|  | ||||
|         task.add_done_callback(stop_loop) | ||||
|         try: | ||||
|             return task.result() # suppress unused task warning | ||||
|         except Exception: | ||||
|             return None | ||||
|             loop.run_forever() | ||||
|         finally: | ||||
|             _cleanup_loop(loop) | ||||
|             loop.close() | ||||
|  | ||||
|     def run(self, *args, **kwargs): | ||||
|         """A blocking call that abstracts away the `event loop`_ | ||||
| @@ -528,29 +553,15 @@ class Client: | ||||
|         is_windows = sys.platform == 'win32' | ||||
|         loop = self.loop | ||||
|         if not is_windows: | ||||
|             loop.add_signal_handler(signal.SIGINT, self._do_cleanup) | ||||
|             loop.add_signal_handler(signal.SIGTERM, self._do_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) | ||||
|             loop.add_signal_handler(signal.SIGINT, _raise_proper_cleanup) | ||||
|             loop.add_signal_handler(signal.SIGTERM, _raise_proper_cleanup) | ||||
|  | ||||
|         try: | ||||
|             loop.run_forever() | ||||
|         except KeyboardInterrupt: | ||||
|             loop.run_until_complete(self.start(*args, **kwargs)) | ||||
|         except (_ProperCleanup, KeyboardInterrupt): | ||||
|             log.info('Received signal to terminate bot and event loop.') | ||||
|         finally: | ||||
|             task.remove_done_callback(stop_loop_on_finish) | ||||
|             if is_windows: | ||||
|                 self._do_cleanup() | ||||
|  | ||||
|             loop.close() | ||||
|             if task.cancelled() or not task.done(): | ||||
|                 return None | ||||
|             return task.result() | ||||
|             self._do_cleanup() | ||||
|  | ||||
|     # properties | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user