ominfra 0.0.0.dev118__py3-none-any.whl → 0.0.0.dev119__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- ominfra/deploy/_executor.py +6 -2
- ominfra/deploy/poly/_main.py +6 -2
- ominfra/journald/fields.py +187 -0
- ominfra/journald/tailer.py +375 -312
- ominfra/pyremote/_runcommands.py +6 -2
- ominfra/scripts/journald2aws.py +381 -314
- ominfra/scripts/supervisor.py +649 -425
- ominfra/supervisor/__main__.py +1 -1
- ominfra/supervisor/context.py +50 -25
- ominfra/supervisor/dispatchers.py +8 -9
- ominfra/supervisor/main.py +68 -0
- ominfra/supervisor/poller.py +1 -3
- ominfra/supervisor/process.py +2 -4
- ominfra/supervisor/supervisor.py +100 -141
- {ominfra-0.0.0.dev118.dist-info → ominfra-0.0.0.dev119.dist-info}/METADATA +3 -3
- {ominfra-0.0.0.dev118.dist-info → ominfra-0.0.0.dev119.dist-info}/RECORD +20 -18
- {ominfra-0.0.0.dev118.dist-info → ominfra-0.0.0.dev119.dist-info}/LICENSE +0 -0
- {ominfra-0.0.0.dev118.dist-info → ominfra-0.0.0.dev119.dist-info}/WHEEL +0 -0
- {ominfra-0.0.0.dev118.dist-info → ominfra-0.0.0.dev119.dist-info}/entry_points.txt +0 -0
- {ominfra-0.0.0.dev118.dist-info → ominfra-0.0.0.dev119.dist-info}/top_level.txt +0 -0
ominfra/supervisor/__main__.py
CHANGED
ominfra/supervisor/context.py
CHANGED
@@ -2,7 +2,6 @@
|
|
2
2
|
import errno
|
3
3
|
import fcntl
|
4
4
|
import grp
|
5
|
-
import logging
|
6
5
|
import os
|
7
6
|
import pwd
|
8
7
|
import re
|
@@ -12,6 +11,8 @@ import stat
|
|
12
11
|
import typing as ta
|
13
12
|
import warnings
|
14
13
|
|
14
|
+
from omlish.lite.logs import log
|
15
|
+
|
15
16
|
from .compat import SignalReceiver
|
16
17
|
from .compat import close_fd
|
17
18
|
from .compat import mktempfile
|
@@ -23,6 +24,7 @@ from .datatypes import name_to_uid
|
|
23
24
|
from .exceptions import NoPermissionError
|
24
25
|
from .exceptions import NotExecutableError
|
25
26
|
from .exceptions import NotFoundError
|
27
|
+
from .poller import BasePoller
|
26
28
|
from .poller import Poller
|
27
29
|
from .states import SupervisorState
|
28
30
|
from .states import SupervisorStates
|
@@ -30,41 +32,47 @@ from .types import AbstractServerContext
|
|
30
32
|
from .types import AbstractSubprocess
|
31
33
|
|
32
34
|
|
33
|
-
log = logging.getLogger(__name__)
|
34
|
-
|
35
|
-
|
36
35
|
class ServerContext(AbstractServerContext):
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
36
|
+
def __init__(
|
37
|
+
self,
|
38
|
+
config: ServerConfig,
|
39
|
+
*,
|
40
|
+
epoch: int = 0,
|
41
|
+
) -> None:
|
43
42
|
super().__init__()
|
44
43
|
|
45
44
|
self._config = config
|
45
|
+
self._epoch = epoch
|
46
46
|
|
47
47
|
self._pid_history: ta.Dict[int, AbstractSubprocess] = {}
|
48
48
|
self._state: SupervisorState = SupervisorStates.RUNNING
|
49
49
|
|
50
|
-
self.
|
50
|
+
self._signal_receiver = SignalReceiver()
|
51
51
|
|
52
|
-
self.
|
52
|
+
self._poller: BasePoller = Poller()
|
53
53
|
|
54
|
-
if
|
55
|
-
uid = name_to_uid(
|
56
|
-
self.
|
57
|
-
self.
|
54
|
+
if config.user is not None:
|
55
|
+
uid = name_to_uid(config.user)
|
56
|
+
self._uid: ta.Optional[int] = uid
|
57
|
+
self._gid: ta.Optional[int] = gid_for_uid(uid)
|
58
58
|
else:
|
59
|
-
self.
|
60
|
-
self.
|
59
|
+
self._uid = None
|
60
|
+
self._gid = None
|
61
61
|
|
62
|
-
self.
|
62
|
+
self._unlink_pidfile = False
|
63
63
|
|
64
64
|
@property
|
65
65
|
def config(self) -> ServerConfig:
|
66
66
|
return self._config
|
67
67
|
|
68
|
+
@property
|
69
|
+
def epoch(self) -> int:
|
70
|
+
return self._epoch
|
71
|
+
|
72
|
+
@property
|
73
|
+
def first(self) -> bool:
|
74
|
+
return not self._epoch
|
75
|
+
|
68
76
|
@property
|
69
77
|
def state(self) -> SupervisorState:
|
70
78
|
return self._state
|
@@ -72,17 +80,26 @@ class ServerContext(AbstractServerContext):
|
|
72
80
|
def set_state(self, state: SupervisorState) -> None:
|
73
81
|
self._state = state
|
74
82
|
|
83
|
+
@property
|
84
|
+
def poller(self) -> BasePoller:
|
85
|
+
return self._poller
|
86
|
+
|
75
87
|
@property
|
76
88
|
def pid_history(self) -> ta.Dict[int, AbstractSubprocess]:
|
77
89
|
return self._pid_history
|
78
90
|
|
79
|
-
|
80
|
-
|
91
|
+
@property
|
92
|
+
def uid(self) -> ta.Optional[int]:
|
93
|
+
return self._uid
|
94
|
+
|
95
|
+
@property
|
96
|
+
def gid(self) -> ta.Optional[int]:
|
97
|
+
return self._gid
|
81
98
|
|
82
99
|
##
|
83
100
|
|
84
101
|
def set_signals(self) -> None:
|
85
|
-
self.
|
102
|
+
self._signal_receiver.install(
|
86
103
|
signal.SIGTERM,
|
87
104
|
signal.SIGINT,
|
88
105
|
signal.SIGQUIT,
|
@@ -193,7 +210,7 @@ class ServerContext(AbstractServerContext):
|
|
193
210
|
))
|
194
211
|
|
195
212
|
def cleanup(self) -> None:
|
196
|
-
if self.
|
213
|
+
if self._unlink_pidfile:
|
197
214
|
try_unlink(self.config.pidfile)
|
198
215
|
self.poller.close()
|
199
216
|
|
@@ -246,6 +263,7 @@ class ServerContext(AbstractServerContext):
|
|
246
263
|
# Parent
|
247
264
|
log.debug('supervisord forked; parent exiting')
|
248
265
|
real_exit(0)
|
266
|
+
|
249
267
|
# Child
|
250
268
|
log.info('daemonizing the supervisord process')
|
251
269
|
if self.config.directory:
|
@@ -255,11 +273,15 @@ class ServerContext(AbstractServerContext):
|
|
255
273
|
log.critical("can't chdir into %r: %s", self.config.directory, err)
|
256
274
|
else:
|
257
275
|
log.info('set current directory: %r', self.config.directory)
|
276
|
+
|
258
277
|
os.dup2(0, os.open('/dev/null', os.O_RDONLY))
|
259
278
|
os.dup2(1, os.open('/dev/null', os.O_WRONLY))
|
260
279
|
os.dup2(2, os.open('/dev/null', os.O_WRONLY))
|
280
|
+
|
261
281
|
os.setsid()
|
282
|
+
|
262
283
|
os.umask(self.config.umask)
|
284
|
+
|
263
285
|
# XXX Stevens, in his Advanced Unix book, section 13.3 (page 417) recommends calling umask(0) and closing unused
|
264
286
|
# file descriptors. In his Network Programming book, he additionally recommends ignoring SIGHUP and forking
|
265
287
|
# again after the setsid() call, for obscure SVR4 reasons.
|
@@ -274,7 +296,7 @@ class ServerContext(AbstractServerContext):
|
|
274
296
|
return logfile
|
275
297
|
|
276
298
|
def get_signal(self) -> ta.Optional[int]:
|
277
|
-
return self.
|
299
|
+
return self._signal_receiver.get_signal()
|
278
300
|
|
279
301
|
def write_pidfile(self) -> None:
|
280
302
|
pid = os.getpid()
|
@@ -284,7 +306,7 @@ class ServerContext(AbstractServerContext):
|
|
284
306
|
except OSError:
|
285
307
|
log.critical('could not write pidfile %s', self.config.pidfile)
|
286
308
|
else:
|
287
|
-
self.
|
309
|
+
self._unlink_pidfile = True
|
288
310
|
log.info('supervisord started with pid %s', pid)
|
289
311
|
|
290
312
|
|
@@ -337,11 +359,14 @@ def drop_privileges(user: ta.Union[int, str, None]) -> ta.Optional[str]:
|
|
337
359
|
os.setgroups(groups)
|
338
360
|
except OSError:
|
339
361
|
return 'Could not set groups of effective user'
|
362
|
+
|
340
363
|
try:
|
341
364
|
os.setgid(gid)
|
342
365
|
except OSError:
|
343
366
|
return 'Could not set group id of effective user'
|
367
|
+
|
344
368
|
os.setuid(uid)
|
369
|
+
|
345
370
|
return None
|
346
371
|
|
347
372
|
|
@@ -5,6 +5,8 @@ import logging
|
|
5
5
|
import os
|
6
6
|
import typing as ta
|
7
7
|
|
8
|
+
from omlish.lite.logs import log
|
9
|
+
|
8
10
|
from .compat import as_bytes
|
9
11
|
from .compat import compact_traceback
|
10
12
|
from .compat import find_prefix_at_end
|
@@ -17,9 +19,6 @@ from .events import notify_event
|
|
17
19
|
from .types import AbstractSubprocess
|
18
20
|
|
19
21
|
|
20
|
-
log = logging.getLogger(__name__)
|
21
|
-
|
22
|
-
|
23
22
|
class Dispatcher(abc.ABC):
|
24
23
|
|
25
24
|
def __init__(self, process: AbstractSubprocess, channel: str, fd: int) -> None:
|
@@ -176,16 +175,16 @@ class OutputDispatcher(Dispatcher):
|
|
176
175
|
# )
|
177
176
|
|
178
177
|
def remove_logs(self):
|
179
|
-
for
|
180
|
-
if
|
181
|
-
for handler in
|
178
|
+
for l in (self._normal_log, self._capture_log):
|
179
|
+
if l is not None:
|
180
|
+
for handler in l.handlers:
|
182
181
|
handler.remove() # type: ignore
|
183
182
|
handler.reopen() # type: ignore
|
184
183
|
|
185
184
|
def reopen_logs(self):
|
186
|
-
for
|
187
|
-
if
|
188
|
-
for handler in
|
185
|
+
for l in (self._normal_log, self._capture_log):
|
186
|
+
if l is not None:
|
187
|
+
for handler in l.handlers:
|
189
188
|
handler.reopen() # type: ignore
|
190
189
|
|
191
190
|
def _log(self, data):
|
@@ -0,0 +1,68 @@
|
|
1
|
+
#!/usr/bin/env python3
|
2
|
+
# ruff: noqa: UP006 UP007
|
3
|
+
# @omlish-amalg ../scripts/supervisor.py
|
4
|
+
import itertools
|
5
|
+
import json
|
6
|
+
import typing as ta
|
7
|
+
|
8
|
+
from omlish.lite.journald import journald_log_handler_factory
|
9
|
+
from omlish.lite.logs import configure_standard_logging
|
10
|
+
from omlish.lite.marshal import unmarshal_obj
|
11
|
+
|
12
|
+
from .compat import ExitNow
|
13
|
+
from .configs import ServerConfig
|
14
|
+
from .context import ServerContext
|
15
|
+
from .states import SupervisorStates
|
16
|
+
from .supervisor import Supervisor
|
17
|
+
|
18
|
+
|
19
|
+
def main(
|
20
|
+
argv: ta.Optional[ta.Sequence[str]] = None,
|
21
|
+
*,
|
22
|
+
no_logging: bool = False,
|
23
|
+
) -> None:
|
24
|
+
import argparse
|
25
|
+
|
26
|
+
parser = argparse.ArgumentParser()
|
27
|
+
parser.add_argument('config_file', metavar='config-file')
|
28
|
+
parser.add_argument('--no-journald', action='store_true')
|
29
|
+
args = parser.parse_args(argv)
|
30
|
+
|
31
|
+
#
|
32
|
+
|
33
|
+
if not (cf := args.config_file):
|
34
|
+
raise RuntimeError('No config file specified')
|
35
|
+
|
36
|
+
if not no_logging:
|
37
|
+
configure_standard_logging(
|
38
|
+
'INFO',
|
39
|
+
handler_factory=journald_log_handler_factory if not args.no_journald else None,
|
40
|
+
)
|
41
|
+
|
42
|
+
#
|
43
|
+
|
44
|
+
# if we hup, restart by making a new Supervisor()
|
45
|
+
for epoch in itertools.count():
|
46
|
+
with open(cf) as f:
|
47
|
+
config_src = f.read()
|
48
|
+
|
49
|
+
config_dct = json.loads(config_src)
|
50
|
+
config: ServerConfig = unmarshal_obj(config_dct, ServerConfig)
|
51
|
+
|
52
|
+
context = ServerContext(
|
53
|
+
config,
|
54
|
+
epoch=epoch,
|
55
|
+
)
|
56
|
+
|
57
|
+
supervisor = Supervisor(context)
|
58
|
+
try:
|
59
|
+
supervisor.main()
|
60
|
+
except ExitNow:
|
61
|
+
pass
|
62
|
+
|
63
|
+
if context.state < SupervisorStates.RESTARTING:
|
64
|
+
break
|
65
|
+
|
66
|
+
|
67
|
+
if __name__ == '__main__':
|
68
|
+
main()
|
ominfra/supervisor/poller.py
CHANGED
ominfra/supervisor/process.py
CHANGED
@@ -1,7 +1,6 @@
|
|
1
1
|
# ruff: noqa: UP006 UP007
|
2
2
|
import errno
|
3
3
|
import functools
|
4
|
-
import logging
|
5
4
|
import os
|
6
5
|
import shlex
|
7
6
|
import signal
|
@@ -9,6 +8,8 @@ import time
|
|
9
8
|
import traceback
|
10
9
|
import typing as ta
|
11
10
|
|
11
|
+
from omlish.lite.logs import log
|
12
|
+
|
12
13
|
from .compat import as_bytes
|
13
14
|
from .compat import as_string
|
14
15
|
from .compat import close_fd
|
@@ -54,9 +55,6 @@ from .types import AbstractServerContext
|
|
54
55
|
from .types import AbstractSubprocess
|
55
56
|
|
56
57
|
|
57
|
-
log = logging.getLogger(__name__)
|
58
|
-
|
59
|
-
|
60
58
|
@functools.total_ordering
|
61
59
|
class Subprocess(AbstractSubprocess):
|
62
60
|
"""A class to manage a subprocess."""
|
ominfra/supervisor/supervisor.py
CHANGED
@@ -1,22 +1,17 @@
|
|
1
|
-
#!/usr/bin/env python3
|
2
1
|
# ruff: noqa: UP006 UP007
|
3
|
-
# @omlish-amalg ../scripts/supervisor.py
|
4
|
-
import json
|
5
|
-
import logging
|
6
2
|
import signal
|
7
3
|
import time
|
8
4
|
import typing as ta
|
9
5
|
|
6
|
+
from omlish.lite.cached import cached_nullary
|
10
7
|
from omlish.lite.check import check_not_none
|
11
|
-
from omlish.lite.logs import
|
12
|
-
from omlish.lite.marshal import unmarshal_obj
|
8
|
+
from omlish.lite.logs import log
|
13
9
|
|
14
10
|
from .compat import ExitNow
|
15
11
|
from .compat import as_string
|
16
12
|
from .compat import decode_wait_status
|
17
13
|
from .compat import signame
|
18
14
|
from .configs import ProcessGroupConfig
|
19
|
-
from .configs import ServerConfig
|
20
15
|
from .context import ServerContext
|
21
16
|
from .dispatchers import Dispatcher
|
22
17
|
from .events import TICK_EVENTS
|
@@ -33,7 +28,8 @@ from .states import SupervisorStates
|
|
33
28
|
from .states import get_process_state_description
|
34
29
|
|
35
30
|
|
36
|
-
|
31
|
+
def timeslice(period, when):
|
32
|
+
return int(when - (when % period))
|
37
33
|
|
38
34
|
|
39
35
|
class Supervisor:
|
@@ -56,6 +52,11 @@ class Supervisor:
|
|
56
52
|
return self._context.state
|
57
53
|
|
58
54
|
def main(self) -> None:
|
55
|
+
self.setup()
|
56
|
+
self.run()
|
57
|
+
|
58
|
+
@cached_nullary
|
59
|
+
def setup(self) -> None:
|
59
60
|
if not self._context.first:
|
60
61
|
# prevent crash on libdispatch-based systems, at least for the first request
|
61
62
|
self._context.cleanup_fds()
|
@@ -70,9 +71,11 @@ class Supervisor:
|
|
70
71
|
# clean up old automatic logs
|
71
72
|
self._context.clear_auto_child_logdir()
|
72
73
|
|
73
|
-
|
74
|
-
|
75
|
-
|
74
|
+
def run(
|
75
|
+
self,
|
76
|
+
*,
|
77
|
+
callback: ta.Optional[ta.Callable[['Supervisor'], bool]] = None,
|
78
|
+
) -> None:
|
76
79
|
self._process_groups = {} # clear
|
77
80
|
self._stop_groups = None # clear
|
78
81
|
|
@@ -90,12 +93,23 @@ class Supervisor:
|
|
90
93
|
# writing pid file needs to come *after* daemonizing or pid will be wrong
|
91
94
|
self._context.write_pidfile()
|
92
95
|
|
93
|
-
|
96
|
+
notify_event(SupervisorRunningEvent())
|
97
|
+
|
98
|
+
while True:
|
99
|
+
if callback is not None and not callback(self):
|
100
|
+
break
|
101
|
+
|
102
|
+
self._run_once()
|
94
103
|
|
95
104
|
finally:
|
96
105
|
self._context.cleanup()
|
97
106
|
|
98
|
-
|
107
|
+
class DiffToActive(ta.NamedTuple):
|
108
|
+
added: ta.List[ProcessGroupConfig]
|
109
|
+
changed: ta.List[ProcessGroupConfig]
|
110
|
+
removed: ta.List[ProcessGroupConfig]
|
111
|
+
|
112
|
+
def diff_to_active(self) -> DiffToActive:
|
99
113
|
new = self._context.config.groups or []
|
100
114
|
cur = [group.config for group in self._process_groups.values()]
|
101
115
|
|
@@ -107,7 +121,7 @@ class Supervisor:
|
|
107
121
|
|
108
122
|
changed = [cand for cand in new if cand != curdict.get(cand.name, cand)]
|
109
123
|
|
110
|
-
return added, changed, removed
|
124
|
+
return Supervisor.DiffToActive(added, changed, removed)
|
111
125
|
|
112
126
|
def add_process_group(self, config: ProcessGroupConfig) -> bool:
|
113
127
|
name = config.name
|
@@ -173,90 +187,84 @@ class Supervisor:
|
|
173
187
|
# down, so push it back on to the end of the stop group queue
|
174
188
|
self._stop_groups.append(group)
|
175
189
|
|
176
|
-
def
|
177
|
-
|
178
|
-
|
190
|
+
def _run_once(self) -> None:
|
191
|
+
combined_map = {}
|
192
|
+
combined_map.update(self.get_process_map())
|
193
|
+
|
194
|
+
pgroups = list(self._process_groups.values())
|
195
|
+
pgroups.sort()
|
196
|
+
|
197
|
+
if self._context.state < SupervisorStates.RUNNING:
|
198
|
+
if not self._stopping:
|
199
|
+
# first time, set the stopping flag, do a notification and set stop_groups
|
200
|
+
self._stopping = True
|
201
|
+
self._stop_groups = pgroups[:]
|
202
|
+
notify_event(SupervisorStoppingEvent())
|
203
|
+
|
204
|
+
self._ordered_stop_groups_phase_1()
|
205
|
+
|
206
|
+
if not self.shutdown_report():
|
207
|
+
# if there are no unstopped processes (we're done killing everything), it's OK to shutdown or reload
|
208
|
+
raise ExitNow
|
179
209
|
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
if not self.shutdown_report():
|
197
|
-
# if there are no unstopped processes (we're done killing everything), it's OK to shutdown or reload
|
198
|
-
raise ExitNow
|
199
|
-
|
200
|
-
for fd, dispatcher in combined_map.items():
|
201
|
-
if dispatcher.readable():
|
202
|
-
self._context.poller.register_readable(fd)
|
203
|
-
if dispatcher.writable():
|
204
|
-
self._context.poller.register_writable(fd)
|
205
|
-
|
206
|
-
r, w = self._context.poller.poll(timeout)
|
207
|
-
|
208
|
-
for fd in r:
|
209
|
-
if fd in combined_map:
|
210
|
-
try:
|
211
|
-
dispatcher = combined_map[fd]
|
212
|
-
log.debug('read event caused by %r', dispatcher)
|
213
|
-
dispatcher.handle_read_event()
|
214
|
-
if not dispatcher.readable():
|
215
|
-
self._context.poller.unregister_readable(fd)
|
216
|
-
except ExitNow:
|
217
|
-
raise
|
218
|
-
except Exception: # noqa
|
219
|
-
combined_map[fd].handle_error()
|
220
|
-
else:
|
221
|
-
# if the fd is not in combined_map, we should unregister it. otherwise, it will be polled every
|
222
|
-
# time, which may cause 100% cpu usage
|
223
|
-
log.debug('unexpected read event from fd %r', fd)
|
224
|
-
try:
|
210
|
+
for fd, dispatcher in combined_map.items():
|
211
|
+
if dispatcher.readable():
|
212
|
+
self._context.poller.register_readable(fd)
|
213
|
+
if dispatcher.writable():
|
214
|
+
self._context.poller.register_writable(fd)
|
215
|
+
|
216
|
+
timeout = 1 # this cannot be fewer than the smallest TickEvent (5)
|
217
|
+
r, w = self._context.poller.poll(timeout)
|
218
|
+
|
219
|
+
for fd in r:
|
220
|
+
if fd in combined_map:
|
221
|
+
try:
|
222
|
+
dispatcher = combined_map[fd]
|
223
|
+
log.debug('read event caused by %r', dispatcher)
|
224
|
+
dispatcher.handle_read_event()
|
225
|
+
if not dispatcher.readable():
|
225
226
|
self._context.poller.unregister_readable(fd)
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
237
|
-
|
238
|
-
|
239
|
-
|
240
|
-
|
241
|
-
|
242
|
-
|
243
|
-
|
227
|
+
except ExitNow:
|
228
|
+
raise
|
229
|
+
except Exception: # noqa
|
230
|
+
combined_map[fd].handle_error()
|
231
|
+
else:
|
232
|
+
# if the fd is not in combined_map, we should unregister it. otherwise, it will be polled every
|
233
|
+
# time, which may cause 100% cpu usage
|
234
|
+
log.debug('unexpected read event from fd %r', fd)
|
235
|
+
try:
|
236
|
+
self._context.poller.unregister_readable(fd)
|
237
|
+
except Exception: # noqa
|
238
|
+
pass
|
239
|
+
|
240
|
+
for fd in w:
|
241
|
+
if fd in combined_map:
|
242
|
+
try:
|
243
|
+
dispatcher = combined_map[fd]
|
244
|
+
log.debug('write event caused by %r', dispatcher)
|
245
|
+
dispatcher.handle_write_event()
|
246
|
+
if not dispatcher.writable():
|
244
247
|
self._context.poller.unregister_writable(fd)
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
248
|
+
except ExitNow:
|
249
|
+
raise
|
250
|
+
except Exception: # noqa
|
251
|
+
combined_map[fd].handle_error()
|
252
|
+
else:
|
253
|
+
log.debug('unexpected write event from fd %r', fd)
|
254
|
+
try:
|
255
|
+
self._context.poller.unregister_writable(fd)
|
256
|
+
except Exception: # noqa
|
257
|
+
pass
|
250
258
|
|
251
|
-
|
252
|
-
|
253
|
-
self._tick()
|
259
|
+
for group in pgroups:
|
260
|
+
group.transition()
|
254
261
|
|
255
|
-
|
256
|
-
|
262
|
+
self._reap()
|
263
|
+
self._handle_signal()
|
264
|
+
self._tick()
|
257
265
|
|
258
|
-
|
259
|
-
|
266
|
+
if self._context.state < SupervisorStates.RUNNING:
|
267
|
+
self._ordered_stop_groups_phase_2()
|
260
268
|
|
261
269
|
def _tick(self, now: ta.Optional[float] = None) -> None:
|
262
270
|
"""Send one or more 'tick' events when the timeslice related to the period for the event type rolls over"""
|
@@ -325,52 +333,3 @@ class Supervisor:
|
|
325
333
|
|
326
334
|
else:
|
327
335
|
log.debug('received %s indicating nothing', signame(sig))
|
328
|
-
|
329
|
-
|
330
|
-
def timeslice(period, when):
|
331
|
-
return int(when - (when % period))
|
332
|
-
|
333
|
-
|
334
|
-
def main(args=None, test=False):
|
335
|
-
import argparse
|
336
|
-
|
337
|
-
parser = argparse.ArgumentParser()
|
338
|
-
parser.add_argument('config_file', metavar='config-file')
|
339
|
-
args = parser.parse_args()
|
340
|
-
|
341
|
-
configure_standard_logging('INFO')
|
342
|
-
|
343
|
-
if not (cf := args.config_file):
|
344
|
-
raise RuntimeError('No config file specified')
|
345
|
-
|
346
|
-
# if we hup, restart by making a new Supervisor()
|
347
|
-
first = True
|
348
|
-
while True:
|
349
|
-
with open(cf) as f:
|
350
|
-
config_src = f.read()
|
351
|
-
config_dct = json.loads(config_src)
|
352
|
-
config: ServerConfig = unmarshal_obj(config_dct, ServerConfig)
|
353
|
-
|
354
|
-
context = ServerContext(
|
355
|
-
config,
|
356
|
-
)
|
357
|
-
|
358
|
-
context.first = first
|
359
|
-
context.test = test
|
360
|
-
go(context)
|
361
|
-
# options.close_logger()
|
362
|
-
first = False
|
363
|
-
if test or (context.state < SupervisorStates.RESTARTING):
|
364
|
-
break
|
365
|
-
|
366
|
-
|
367
|
-
def go(context): # pragma: no cover
|
368
|
-
d = Supervisor(context)
|
369
|
-
try:
|
370
|
-
d.main()
|
371
|
-
except ExitNow:
|
372
|
-
pass
|
373
|
-
|
374
|
-
|
375
|
-
if __name__ == '__main__':
|
376
|
-
main()
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.1
|
2
2
|
Name: ominfra
|
3
|
-
Version: 0.0.0.
|
3
|
+
Version: 0.0.0.dev119
|
4
4
|
Summary: ominfra
|
5
5
|
Author: wrmsr
|
6
6
|
License: BSD-3-Clause
|
@@ -12,8 +12,8 @@ Classifier: Operating System :: OS Independent
|
|
12
12
|
Classifier: Operating System :: POSIX
|
13
13
|
Requires-Python: >=3.12
|
14
14
|
License-File: LICENSE
|
15
|
-
Requires-Dist: omdev ==0.0.0.
|
16
|
-
Requires-Dist: omlish ==0.0.0.
|
15
|
+
Requires-Dist: omdev ==0.0.0.dev119
|
16
|
+
Requires-Dist: omlish ==0.0.0.dev119
|
17
17
|
Provides-Extra: all
|
18
18
|
Requires-Dist: paramiko ~=3.5 ; extra == 'all'
|
19
19
|
Requires-Dist: asyncssh ~=2.18 ; extra == 'all'
|