-
Notifications
You must be signed in to change notification settings - Fork 27
/
exithooks.py
68 lines (53 loc) · 2.41 KB
/
exithooks.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
"""
This module enable registering exit hooks that are called both on successful exits and exits
due to signals. See :py:class:`ExitHooksManager` for details. Use this through the singleton
value :py:data:`EXIT_HOOKS_MGR` (its singleton nature is not enforced).
"""
import atexit
import signal
from typing import Callable
####################################################################################################
class ExitHooksManager:
"""
Enables registering exit hooks that are called on exit (both successful exit and exit due to
signals). Does support all signals that normally cause termination of the program (SIGTERM,
SIGINT and SIGQUIT) but not SIGKILL that cannot be caught by design.
This will install signal handlers that kill the process (just like the default handlers) in
addition to running the hooks. Do not use this if you want to handle the signals yourself
without exiting.
"""
def __init__(self):
self.hooks = []
"""List of functions to call when exiting."""
self.signal_fired = False
"""
Record when a signal has been fired to avoid calling hooks again with the atexit handler.
"""
atexit.register(self._run_hooks)
signal.signal(signal.SIGTERM, self._signal_handler)
signal.signal(signal.SIGINT, self._signal_handler)
signal.signal(signal.SIGQUIT, self._signal_handler)
def _run_hooks(self):
if self.signal_fired:
return
for hook in self.hooks:
hook(0)
def _signal_handler(self, signum, frame):
try:
for hook in self.hooks:
hook(signum)
finally:
if signum == signal.SIGINT:
print("Interrupted by SIGINT (most likely Ctrl-C).")
self.signal_fired = True
exit(signum)
def register(self, hook: Callable[[int], None]):
"""
Registers a hook to be called on exit (both successful exit and exit due to signal).
The hook is passed 0 on successful exit, and the signal number on exit due to signal.
"""
self.hooks.append(hook)
####################################################################################################
EXIT_HOOKS_MGR = ExitHooksManager()
"""Singleton instance of :py:class:`ExitHooksManager`."""
####################################################################################################