|
1 | 1 | from __future__ import annotations |
2 | 2 |
|
3 | 3 | import asyncio as __asyncio |
| 4 | +import signal as __signal |
4 | 5 | import sys as __sys |
| 6 | +import threading as __threading |
5 | 7 | import typing as __typing |
6 | 8 |
|
7 | 9 | from ._loop_compat import Loop |
|
11 | 13 | _PREVIOUS_EVENT_LOOP_POLICY: __typing.Optional[__asyncio.AbstractEventLoopPolicy] = None |
12 | 14 |
|
13 | 15 |
|
| 16 | +def _noop() -> None: |
| 17 | + pass |
| 18 | + |
| 19 | + |
14 | 20 | def new_event_loop() -> Loop: |
15 | 21 | return Loop() |
16 | 22 |
|
@@ -48,6 +54,25 @@ def uninstall() -> None: |
48 | 54 | _PREVIOUS_EVENT_LOOP_POLICY = None |
49 | 55 |
|
50 | 56 |
|
| 57 | +class _SigintHandler: |
| 58 | + def __init__(self, loop: Loop, main_task: __asyncio.Task[__typing.Any]) -> None: |
| 59 | + self._loop = loop |
| 60 | + self._main_task = main_task |
| 61 | + self.interrupt_count = 0 |
| 62 | + |
| 63 | + def __call__( |
| 64 | + self, |
| 65 | + signum: int, |
| 66 | + frame: __typing.Optional[__typing.Any], |
| 67 | + ) -> None: |
| 68 | + self.interrupt_count += 1 |
| 69 | + if self.interrupt_count == 1 and not self._main_task.done(): |
| 70 | + self._main_task.cancel() |
| 71 | + self._loop.call_soon_threadsafe(_noop) |
| 72 | + return |
| 73 | + raise KeyboardInterrupt() |
| 74 | + |
| 75 | + |
51 | 76 | if __typing.TYPE_CHECKING: |
52 | 77 |
|
53 | 78 | def run( |
@@ -86,7 +111,31 @@ async def wrapper(): |
86 | 111 | __asyncio.set_event_loop(loop) |
87 | 112 | if debug is not None: |
88 | 113 | loop.set_debug(debug) |
89 | | - return loop.run_until_complete(wrapper()) |
| 114 | + main_task = loop.create_task(wrapper()) |
| 115 | + sigint_handler = None |
| 116 | + if ( |
| 117 | + __threading.current_thread() is __threading.main_thread() |
| 118 | + and __signal.getsignal(__signal.SIGINT) is __signal.default_int_handler |
| 119 | + ): |
| 120 | + sigint_handler = _SigintHandler(loop, main_task) |
| 121 | + try: |
| 122 | + __signal.signal(__signal.SIGINT, sigint_handler) |
| 123 | + except ValueError: |
| 124 | + sigint_handler = None |
| 125 | + try: |
| 126 | + return loop.run_until_complete(main_task) |
| 127 | + except __asyncio.CancelledError: |
| 128 | + if sigint_handler is not None and sigint_handler.interrupt_count > 0: |
| 129 | + uncancel = getattr(main_task, "uncancel", None) |
| 130 | + if uncancel is None or uncancel() == 0: |
| 131 | + raise KeyboardInterrupt() |
| 132 | + raise |
| 133 | + finally: |
| 134 | + if ( |
| 135 | + sigint_handler is not None |
| 136 | + and __signal.getsignal(__signal.SIGINT) is sigint_handler |
| 137 | + ): |
| 138 | + __signal.signal(__signal.SIGINT, __signal.default_int_handler) |
90 | 139 | finally: |
91 | 140 | try: |
92 | 141 | __cancel_all_tasks(loop) |
|
0 commit comments