omlish 0.0.0.dev230__py3-none-any.whl → 0.0.0.dev232__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- omlish/.manifests.json +12 -0
- omlish/__about__.py +4 -4
- omlish/daemons/__init__.py +0 -0
- omlish/daemons/daemon.py +126 -0
- omlish/daemons/launching.py +121 -0
- omlish/daemons/reparent.py +16 -0
- omlish/daemons/spawning.py +166 -0
- omlish/daemons/targets.py +89 -0
- omlish/daemons/waiting.py +95 -0
- omlish/lang/__init__.py +0 -2
- omlish/lang/contextmanagers.py +15 -10
- omlish/lang/functions.py +10 -0
- omlish/libc.py +2 -4
- omlish/lite/timeouts.py +1 -1
- omlish/os/pidfiles/__main__.py +11 -0
- omlish/os/pidfiles/cli.py +88 -0
- omlish/os/pidfiles/pidfile.py +23 -1
- omlish/os/pidfiles/pinning.py +10 -1
- omlish/os/signals.py +15 -0
- {omlish-0.0.0.dev230.dist-info → omlish-0.0.0.dev232.dist-info}/METADATA +5 -5
- {omlish-0.0.0.dev230.dist-info → omlish-0.0.0.dev232.dist-info}/RECORD +25 -15
- {omlish-0.0.0.dev230.dist-info → omlish-0.0.0.dev232.dist-info}/LICENSE +0 -0
- {omlish-0.0.0.dev230.dist-info → omlish-0.0.0.dev232.dist-info}/WHEEL +0 -0
- {omlish-0.0.0.dev230.dist-info → omlish-0.0.0.dev232.dist-info}/entry_points.txt +0 -0
- {omlish-0.0.0.dev230.dist-info → omlish-0.0.0.dev232.dist-info}/top_level.txt +0 -0
omlish/.manifests.json
CHANGED
@@ -307,6 +307,18 @@
|
|
307
307
|
}
|
308
308
|
}
|
309
309
|
},
|
310
|
+
{
|
311
|
+
"module": ".os.pidfiles.__main__",
|
312
|
+
"attr": "_CLI_MODULE",
|
313
|
+
"file": "omlish/os/pidfiles/__main__.py",
|
314
|
+
"line": 1,
|
315
|
+
"value": {
|
316
|
+
"$omdev.cli.types.CliModule": {
|
317
|
+
"cmd_name": "pidfiles",
|
318
|
+
"mod_name": "omlish.os.pidfiles.__main__"
|
319
|
+
}
|
320
|
+
}
|
321
|
+
},
|
310
322
|
{
|
311
323
|
"module": ".secrets.pwgen",
|
312
324
|
"attr": "_CLI_MODULE",
|
omlish/__about__.py
CHANGED
@@ -1,5 +1,5 @@
|
|
1
|
-
__version__ = '0.0.0.
|
2
|
-
__revision__ = '
|
1
|
+
__version__ = '0.0.0.dev232'
|
2
|
+
__revision__ = '2f0435f277c869f9ecf426be2aeaf08eeb9792de'
|
3
3
|
|
4
4
|
|
5
5
|
#
|
@@ -37,7 +37,7 @@ class Project(ProjectBase):
|
|
37
37
|
|
38
38
|
'greenlet ~= 3.1',
|
39
39
|
|
40
|
-
'trio ~= 0.
|
40
|
+
'trio ~= 0.29',
|
41
41
|
'trio-asyncio ~= 0.15',
|
42
42
|
],
|
43
43
|
|
@@ -56,7 +56,7 @@ class Project(ProjectBase):
|
|
56
56
|
'asttokens ~= 3.0',
|
57
57
|
'executing ~= 2.2',
|
58
58
|
|
59
|
-
'psutil ~=
|
59
|
+
'psutil ~= 7.0',
|
60
60
|
],
|
61
61
|
|
62
62
|
'formats': [
|
File without changes
|
omlish/daemons/daemon.py
ADDED
@@ -0,0 +1,126 @@
|
|
1
|
+
"""
|
2
|
+
TODO:
|
3
|
+
- OK.. this is useful for non-system-daemons too... which have no pidfile..
|
4
|
+
- split out pidfile concern to.. identity?
|
5
|
+
- async[io] support, really just waiting
|
6
|
+
- helpers for http, json, etc
|
7
|
+
- heartbeat? status checks? heartbeat file?
|
8
|
+
- zmq
|
9
|
+
- cli, cfg reload
|
10
|
+
- bootstrap
|
11
|
+
- rsrc limit
|
12
|
+
- logs
|
13
|
+
- https://github.com/Homebrew/homebrew-services
|
14
|
+
- deathpacts
|
15
|
+
- timebomb
|
16
|
+
- pickle protocol, revision / venv check, multiprocessing manager support
|
17
|
+
"""
|
18
|
+
import logging
|
19
|
+
import os.path
|
20
|
+
import time
|
21
|
+
|
22
|
+
from .. import check
|
23
|
+
from .. import dataclasses as dc
|
24
|
+
from .. import lang
|
25
|
+
from ..os.pidfiles.pidfile import Pidfile
|
26
|
+
from .launching import Launcher
|
27
|
+
from .spawning import Spawning
|
28
|
+
from .targets import Target
|
29
|
+
from .waiting import Wait
|
30
|
+
from .waiting import waiter_for
|
31
|
+
|
32
|
+
|
33
|
+
log = logging.getLogger(__name__)
|
34
|
+
|
35
|
+
|
36
|
+
##
|
37
|
+
|
38
|
+
|
39
|
+
class Daemon:
|
40
|
+
@dc.dataclass(frozen=True, kw_only=True)
|
41
|
+
class Config:
|
42
|
+
target: Target
|
43
|
+
spawning: Spawning
|
44
|
+
|
45
|
+
#
|
46
|
+
|
47
|
+
pid_file: str | None = None
|
48
|
+
|
49
|
+
#
|
50
|
+
|
51
|
+
reparent_process: bool = False
|
52
|
+
launched_timeout_s: float = 5.
|
53
|
+
|
54
|
+
#
|
55
|
+
|
56
|
+
wait: Wait | None = None
|
57
|
+
|
58
|
+
wait_timeout: lang.TimeoutLike = 10.
|
59
|
+
wait_sleep_s: float = .1
|
60
|
+
|
61
|
+
#
|
62
|
+
|
63
|
+
def __post_init__(self) -> None:
|
64
|
+
check.isinstance(self.pid_file, (str, None))
|
65
|
+
|
66
|
+
def __init__(self, config: Config) -> None:
|
67
|
+
super().__init__()
|
68
|
+
|
69
|
+
self._config = config
|
70
|
+
|
71
|
+
@property
|
72
|
+
def config(self) -> Config:
|
73
|
+
return self._config
|
74
|
+
|
75
|
+
#
|
76
|
+
|
77
|
+
@property
|
78
|
+
def has_pidfile(self) -> bool:
|
79
|
+
return self._config.pid_file is not None
|
80
|
+
|
81
|
+
def _non_inheritable_pidfile(self) -> Pidfile:
|
82
|
+
check.state(self.has_pidfile)
|
83
|
+
return Pidfile(
|
84
|
+
check.non_empty_str(self._config.pid_file),
|
85
|
+
inheritable=False,
|
86
|
+
)
|
87
|
+
|
88
|
+
def is_running(self) -> bool:
|
89
|
+
check.state(self.has_pidfile)
|
90
|
+
|
91
|
+
if not os.path.isfile(check.non_empty_str(self._config.pid_file)):
|
92
|
+
return False
|
93
|
+
|
94
|
+
with self._non_inheritable_pidfile() as pf:
|
95
|
+
return not pf.try_acquire_lock()
|
96
|
+
|
97
|
+
#
|
98
|
+
|
99
|
+
def wait_sync(self, timeout: lang.TimeoutLike = lang.Timeout.Default) -> None:
|
100
|
+
if self._config.wait is None:
|
101
|
+
return
|
102
|
+
|
103
|
+
timeout = lang.Timeout.of(timeout, self._config.wait_timeout)
|
104
|
+
waiter = waiter_for(self._config.wait)
|
105
|
+
while not waiter.do_wait():
|
106
|
+
timeout()
|
107
|
+
time.sleep(self._config.wait_sleep_s or 0.)
|
108
|
+
|
109
|
+
#
|
110
|
+
|
111
|
+
def launch_no_wait(self) -> None:
|
112
|
+
launcher = Launcher(
|
113
|
+
target=self._config.target,
|
114
|
+
spawning=self._config.spawning,
|
115
|
+
|
116
|
+
pid_file=self._config.pid_file,
|
117
|
+
reparent_process=self._config.reparent_process,
|
118
|
+
launched_timeout_s=self._config.launched_timeout_s,
|
119
|
+
)
|
120
|
+
|
121
|
+
launcher.launch()
|
122
|
+
|
123
|
+
def launch(self, timeout: lang.TimeoutLike = lang.Timeout.Default) -> None:
|
124
|
+
self.launch_no_wait()
|
125
|
+
|
126
|
+
self.wait_sync(timeout)
|
@@ -0,0 +1,121 @@
|
|
1
|
+
"""
|
2
|
+
TODO:
|
3
|
+
- Config? dedupe defaults with Daemon
|
4
|
+
- ExitStacked? hold ref to Spawner, which holds refs to thread/proc - which will likely outlive it, but still
|
5
|
+
"""
|
6
|
+
import contextlib
|
7
|
+
import functools
|
8
|
+
import logging
|
9
|
+
import threading
|
10
|
+
import typing as ta
|
11
|
+
|
12
|
+
from .. import check
|
13
|
+
from .. import lang
|
14
|
+
from ..os.pidfiles.manager import open_inheritable_pidfile
|
15
|
+
from ..os.pidfiles.pidfile import Pidfile
|
16
|
+
from .reparent import reparent_process
|
17
|
+
from .spawning import InProcessSpawner
|
18
|
+
from .spawning import Spawn
|
19
|
+
from .spawning import Spawner
|
20
|
+
from .spawning import Spawning
|
21
|
+
from .spawning import spawner_for
|
22
|
+
from .targets import Target
|
23
|
+
from .targets import target_runner_for
|
24
|
+
|
25
|
+
|
26
|
+
log = logging.getLogger(__name__)
|
27
|
+
|
28
|
+
|
29
|
+
##
|
30
|
+
|
31
|
+
|
32
|
+
class Launcher:
|
33
|
+
def __init__(
|
34
|
+
self,
|
35
|
+
*,
|
36
|
+
target: Target,
|
37
|
+
spawning: Spawning,
|
38
|
+
|
39
|
+
pid_file: str | None = None,
|
40
|
+
reparent_process: bool = False, # noqa
|
41
|
+
launched_timeout_s: float = 5.,
|
42
|
+
) -> None:
|
43
|
+
super().__init__()
|
44
|
+
|
45
|
+
self._target = target
|
46
|
+
self._spawning = spawning
|
47
|
+
|
48
|
+
self._pid_file = pid_file
|
49
|
+
self._reparent_process = reparent_process
|
50
|
+
self._launched_timeout_s = launched_timeout_s
|
51
|
+
|
52
|
+
def _inner_launch(
|
53
|
+
self,
|
54
|
+
*,
|
55
|
+
pidfile_manager: ta.ContextManager | None,
|
56
|
+
launched_callback: ta.Callable[[], None] | None = None,
|
57
|
+
) -> None:
|
58
|
+
try:
|
59
|
+
if self._reparent_process:
|
60
|
+
log.info('Reparenting')
|
61
|
+
reparent_process()
|
62
|
+
|
63
|
+
with contextlib.ExitStack() as es:
|
64
|
+
pidfile: Pidfile | None = None # noqa
|
65
|
+
if pidfile_manager is not None:
|
66
|
+
pidfile = check.isinstance(es.enter_context(pidfile_manager), Pidfile)
|
67
|
+
pidfile.write()
|
68
|
+
|
69
|
+
if launched_callback is not None:
|
70
|
+
launched_callback()
|
71
|
+
|
72
|
+
runner = target_runner_for(self._target)
|
73
|
+
runner.run()
|
74
|
+
|
75
|
+
finally:
|
76
|
+
if launched_callback is not None:
|
77
|
+
launched_callback()
|
78
|
+
|
79
|
+
def launch(self) -> bool:
|
80
|
+
with contextlib.ExitStack() as es:
|
81
|
+
spawner: Spawner = es.enter_context(spawner_for(self._spawning))
|
82
|
+
|
83
|
+
#
|
84
|
+
|
85
|
+
inherit_fds: set[int] = set()
|
86
|
+
launched_event: threading.Event | None = None
|
87
|
+
|
88
|
+
pidfile: Pidfile | None = None # noqa
|
89
|
+
pidfile_manager: ta.ContextManager | None = None
|
90
|
+
|
91
|
+
if (pid_file := self._pid_file) is not None:
|
92
|
+
if not isinstance(spawner, InProcessSpawner):
|
93
|
+
pidfile = es.enter_context(open_inheritable_pidfile(pid_file))
|
94
|
+
pidfile_manager = lang.NopContextManager(pidfile)
|
95
|
+
|
96
|
+
else:
|
97
|
+
check.state(not self._reparent_process)
|
98
|
+
pidfile = es.enter_context(Pidfile(pid_file))
|
99
|
+
pidfile_manager = pidfile.dup()
|
100
|
+
launched_event = threading.Event()
|
101
|
+
|
102
|
+
if not pidfile.try_acquire_lock():
|
103
|
+
return False
|
104
|
+
|
105
|
+
inherit_fds.add(check.isinstance(pidfile.fileno(), int))
|
106
|
+
|
107
|
+
#
|
108
|
+
|
109
|
+
spawner.spawn(Spawn(
|
110
|
+
functools.partial(
|
111
|
+
self._inner_launch,
|
112
|
+
pidfile_manager=pidfile_manager,
|
113
|
+
launched_callback=launched_event.set if launched_event is not None else None,
|
114
|
+
),
|
115
|
+
inherit_fds=inherit_fds,
|
116
|
+
))
|
117
|
+
|
118
|
+
if launched_event is not None:
|
119
|
+
check.state(launched_event.wait(timeout=self._launched_timeout_s))
|
120
|
+
|
121
|
+
return True
|
@@ -0,0 +1,16 @@
|
|
1
|
+
import os
|
2
|
+
import sys
|
3
|
+
|
4
|
+
|
5
|
+
def reparent_process() -> None:
|
6
|
+
if (pid := os.fork()): # noqa
|
7
|
+
sys.exit(0)
|
8
|
+
raise RuntimeError('Unreachable') # noqa
|
9
|
+
|
10
|
+
os.setsid()
|
11
|
+
|
12
|
+
if (pid := os.fork()): # noqa
|
13
|
+
sys.exit(0)
|
14
|
+
|
15
|
+
sys.stdout.flush()
|
16
|
+
sys.stderr.flush()
|
@@ -0,0 +1,166 @@
|
|
1
|
+
import abc
|
2
|
+
import functools
|
3
|
+
import os
|
4
|
+
import sys
|
5
|
+
import threading
|
6
|
+
import typing as ta
|
7
|
+
|
8
|
+
from .. import check
|
9
|
+
from .. import dataclasses as dc
|
10
|
+
from .. import lang
|
11
|
+
from ..diag import pydevd
|
12
|
+
|
13
|
+
|
14
|
+
if ta.TYPE_CHECKING:
|
15
|
+
import multiprocessing as mp
|
16
|
+
import multiprocessing.context
|
17
|
+
import multiprocessing.process # noqa
|
18
|
+
|
19
|
+
from ..multiprocessing import spawn as omp_spawn
|
20
|
+
|
21
|
+
else:
|
22
|
+
mp = lang.proxy_import('multiprocessing', extras=['context', 'process'])
|
23
|
+
subprocess = lang.proxy_import('subprocess')
|
24
|
+
|
25
|
+
omp_spawn = lang.proxy_import('..multiprocessing.spawn', __package__)
|
26
|
+
|
27
|
+
|
28
|
+
##
|
29
|
+
|
30
|
+
|
31
|
+
class Spawning(dc.Case):
|
32
|
+
pass
|
33
|
+
|
34
|
+
|
35
|
+
class Spawn(dc.Frozen, final=True):
|
36
|
+
fn: ta.Callable[[], None]
|
37
|
+
|
38
|
+
_: dc.KW_ONLY
|
39
|
+
|
40
|
+
inherit_fds: ta.Collection[int] | None = None
|
41
|
+
|
42
|
+
|
43
|
+
class Spawner(lang.ContextManaged, abc.ABC):
|
44
|
+
@abc.abstractmethod
|
45
|
+
def spawn(self, spawn: Spawn) -> None:
|
46
|
+
raise NotImplementedError
|
47
|
+
|
48
|
+
|
49
|
+
class InProcessSpawner(Spawner, abc.ABC):
|
50
|
+
pass
|
51
|
+
|
52
|
+
|
53
|
+
@functools.singledispatch
|
54
|
+
def spawner_for(spawning: Spawning) -> Spawner:
|
55
|
+
raise TypeError(spawning)
|
56
|
+
|
57
|
+
|
58
|
+
##
|
59
|
+
|
60
|
+
|
61
|
+
class MultiprocessingSpawning(Spawning, kw_only=True):
|
62
|
+
# Defaults to 'fork' if under pydevd, else 'spawn'
|
63
|
+
start_method: str | None = None
|
64
|
+
|
65
|
+
non_daemon: bool = False
|
66
|
+
|
67
|
+
|
68
|
+
class MultiprocessingSpawner(Spawner):
|
69
|
+
def __init__(self, spawning: MultiprocessingSpawning) -> None:
|
70
|
+
super().__init__()
|
71
|
+
|
72
|
+
self._spawning = spawning
|
73
|
+
self._process: ta.Optional['mp.process.BaseProcess'] = None # noqa
|
74
|
+
|
75
|
+
def _process_cls(self, spawn: Spawn) -> type['mp.process.BaseProcess']:
|
76
|
+
if (start_method := self._spawning.start_method) is None:
|
77
|
+
# Unfortunately, pydevd forces the use of the 'fork' start_method, which cannot be mixed with 'spawn':
|
78
|
+
# https://github.com/python/cpython/blob/a7427f2db937adb4c787754deb4c337f1894fe86/Lib/multiprocessing/spawn.py#L102 # noqa
|
79
|
+
if pydevd.is_running():
|
80
|
+
start_method = 'fork'
|
81
|
+
else:
|
82
|
+
start_method = 'spawn'
|
83
|
+
|
84
|
+
ctx: 'mp.context.BaseContext' # noqa
|
85
|
+
if start_method == 'fork':
|
86
|
+
ctx = mp.get_context(check.non_empty_str(start_method))
|
87
|
+
|
88
|
+
elif start_method == 'spawn':
|
89
|
+
ctx = omp_spawn.ExtrasSpawnContext(omp_spawn.SpawnExtras(
|
90
|
+
pass_fds=frozenset(spawn.inherit_fds) if spawn.inherit_fds is not None else None,
|
91
|
+
))
|
92
|
+
|
93
|
+
else:
|
94
|
+
raise ValueError(start_method)
|
95
|
+
|
96
|
+
return ctx.Process # type: ignore
|
97
|
+
|
98
|
+
def spawn(self, spawn: Spawn) -> None:
|
99
|
+
check.none(self._process)
|
100
|
+
self._process = self._process_cls(spawn)(
|
101
|
+
target=spawn.fn,
|
102
|
+
daemon=not self._spawning.non_daemon,
|
103
|
+
)
|
104
|
+
self._process.start()
|
105
|
+
|
106
|
+
|
107
|
+
@spawner_for.register
|
108
|
+
def _(spawning: MultiprocessingSpawning) -> MultiprocessingSpawner:
|
109
|
+
return MultiprocessingSpawner(spawning)
|
110
|
+
|
111
|
+
|
112
|
+
##
|
113
|
+
|
114
|
+
|
115
|
+
class ForkSpawning(Spawning):
|
116
|
+
pass
|
117
|
+
|
118
|
+
|
119
|
+
class ForkSpawner(Spawner, dc.Frozen):
|
120
|
+
spawning: ForkSpawning
|
121
|
+
|
122
|
+
def spawn(self, spawn: Spawn) -> None:
|
123
|
+
if (pid := os.fork()): # noqa
|
124
|
+
return
|
125
|
+
|
126
|
+
try:
|
127
|
+
spawn.fn()
|
128
|
+
except BaseException: # noqa
|
129
|
+
sys.exit(1)
|
130
|
+
else:
|
131
|
+
sys.exit(0)
|
132
|
+
|
133
|
+
raise RuntimeError('Unreachable') # noqa
|
134
|
+
|
135
|
+
|
136
|
+
@spawner_for.register
|
137
|
+
def _(spawning: ForkSpawning) -> ForkSpawner:
|
138
|
+
return ForkSpawner(spawning)
|
139
|
+
|
140
|
+
|
141
|
+
##
|
142
|
+
|
143
|
+
|
144
|
+
class ThreadSpawning(Spawning, kw_only=True):
|
145
|
+
non_daemon: bool = False
|
146
|
+
|
147
|
+
|
148
|
+
class ThreadSpawner(InProcessSpawner):
|
149
|
+
def __init__(self, spawning: ThreadSpawning) -> None:
|
150
|
+
super().__init__()
|
151
|
+
|
152
|
+
self._spawning = spawning
|
153
|
+
self._thread: threading.Thread | None = None
|
154
|
+
|
155
|
+
def spawn(self, spawn: Spawn) -> None:
|
156
|
+
check.none(self._thread)
|
157
|
+
self._thread = threading.Thread(
|
158
|
+
target=spawn.fn,
|
159
|
+
daemon=not self._spawning.non_daemon,
|
160
|
+
)
|
161
|
+
self._thread.start()
|
162
|
+
|
163
|
+
|
164
|
+
@spawner_for.register
|
165
|
+
def _(spawning: ThreadSpawning) -> ThreadSpawner:
|
166
|
+
return ThreadSpawner(spawning)
|
@@ -0,0 +1,89 @@
|
|
1
|
+
import abc
|
2
|
+
import functools
|
3
|
+
import os
|
4
|
+
import typing as ta
|
5
|
+
|
6
|
+
from .. import check
|
7
|
+
from .. import dataclasses as dc
|
8
|
+
from .. import lang
|
9
|
+
|
10
|
+
|
11
|
+
if ta.TYPE_CHECKING:
|
12
|
+
import runpy
|
13
|
+
else:
|
14
|
+
runpy = lang.proxy_import('runpy')
|
15
|
+
|
16
|
+
|
17
|
+
##
|
18
|
+
|
19
|
+
|
20
|
+
class Target(dc.Case):
|
21
|
+
pass
|
22
|
+
|
23
|
+
|
24
|
+
class TargetRunner(abc.ABC):
|
25
|
+
@abc.abstractmethod
|
26
|
+
def run(self) -> None:
|
27
|
+
raise NotImplementedError
|
28
|
+
|
29
|
+
|
30
|
+
@functools.singledispatch
|
31
|
+
def target_runner_for(target: Target) -> TargetRunner:
|
32
|
+
raise TypeError(target)
|
33
|
+
|
34
|
+
|
35
|
+
##
|
36
|
+
|
37
|
+
|
38
|
+
class FnTarget(Target):
|
39
|
+
fn: ta.Callable[[], None]
|
40
|
+
|
41
|
+
|
42
|
+
class FnTargetRunner(TargetRunner, dc.Frozen):
|
43
|
+
target: FnTarget
|
44
|
+
|
45
|
+
def run(self) -> None:
|
46
|
+
self.target.fn()
|
47
|
+
|
48
|
+
|
49
|
+
@target_runner_for.register
|
50
|
+
def _(target: FnTarget) -> FnTargetRunner:
|
51
|
+
return FnTargetRunner(target)
|
52
|
+
|
53
|
+
|
54
|
+
##
|
55
|
+
|
56
|
+
|
57
|
+
class NameTarget(Target):
|
58
|
+
name: str
|
59
|
+
|
60
|
+
|
61
|
+
class NameTargetRunner(TargetRunner, dc.Frozen):
|
62
|
+
target: NameTarget
|
63
|
+
|
64
|
+
def run(self) -> None:
|
65
|
+
name = self.target.name
|
66
|
+
if lang.can_import(name):
|
67
|
+
runpy._run_module_as_main(name) # type: ignore # noqa
|
68
|
+
else:
|
69
|
+
obj = lang.import_attr(self.target.name)
|
70
|
+
obj()
|
71
|
+
|
72
|
+
|
73
|
+
##
|
74
|
+
|
75
|
+
|
76
|
+
class ExecTarget(Target):
|
77
|
+
cmd: ta.Sequence[str] = dc.xfield(coerce=check.of_not_isinstance(str))
|
78
|
+
|
79
|
+
|
80
|
+
class ExecTargetRunner(TargetRunner, dc.Frozen):
|
81
|
+
target: ExecTarget
|
82
|
+
|
83
|
+
def run(self) -> None:
|
84
|
+
os.execl(*self.target.cmd)
|
85
|
+
|
86
|
+
|
87
|
+
@target_runner_for.register
|
88
|
+
def _(target: ExecTarget) -> ExecTargetRunner:
|
89
|
+
return ExecTargetRunner(target)
|
@@ -0,0 +1,95 @@
|
|
1
|
+
import abc
|
2
|
+
import functools
|
3
|
+
import socket
|
4
|
+
import typing as ta
|
5
|
+
|
6
|
+
from .. import dataclasses as dc
|
7
|
+
|
8
|
+
|
9
|
+
##
|
10
|
+
|
11
|
+
|
12
|
+
class Wait(dc.Case):
|
13
|
+
pass
|
14
|
+
|
15
|
+
|
16
|
+
class Waiter(abc.ABC):
|
17
|
+
@abc.abstractmethod
|
18
|
+
def do_wait(self) -> bool:
|
19
|
+
raise NotImplementedError
|
20
|
+
|
21
|
+
|
22
|
+
@functools.singledispatch
|
23
|
+
def waiter_for(wait: Wait) -> Waiter:
|
24
|
+
raise TypeError(wait)
|
25
|
+
|
26
|
+
|
27
|
+
##
|
28
|
+
|
29
|
+
|
30
|
+
class SequentialWait(Wait):
|
31
|
+
waits: ta.Sequence[Wait]
|
32
|
+
|
33
|
+
|
34
|
+
class SequentialWaiter(Waiter):
|
35
|
+
def __init__(self, waiters: ta.Sequence[Waiter]) -> None:
|
36
|
+
super().__init__()
|
37
|
+
|
38
|
+
self._waiters = waiters
|
39
|
+
self._idx = 0
|
40
|
+
|
41
|
+
def do_wait(self) -> bool:
|
42
|
+
while self._idx < len(self._waiters):
|
43
|
+
if not self._waiters[self._idx].do_wait():
|
44
|
+
return False
|
45
|
+
self._idx += 1
|
46
|
+
return True
|
47
|
+
|
48
|
+
|
49
|
+
@waiter_for.register
|
50
|
+
def _(wait: SequentialWait) -> SequentialWaiter:
|
51
|
+
return SequentialWaiter([waiter_for(c) for c in wait.waits])
|
52
|
+
|
53
|
+
|
54
|
+
##
|
55
|
+
|
56
|
+
|
57
|
+
class FnWait(Wait):
|
58
|
+
fn: ta.Callable[[], bool]
|
59
|
+
|
60
|
+
|
61
|
+
class FnWaiter(Waiter, dc.Frozen):
|
62
|
+
wait: FnWait
|
63
|
+
|
64
|
+
def do_wait(self) -> bool:
|
65
|
+
return self.wait.fn()
|
66
|
+
|
67
|
+
|
68
|
+
@waiter_for.register
|
69
|
+
def _(wait: FnWait) -> FnWaiter:
|
70
|
+
return FnWaiter(wait)
|
71
|
+
|
72
|
+
|
73
|
+
##
|
74
|
+
|
75
|
+
|
76
|
+
class ConnectWait(Wait):
|
77
|
+
address: ta.Any
|
78
|
+
|
79
|
+
|
80
|
+
class ConnectWaiter(Waiter, dc.Frozen):
|
81
|
+
wait: ConnectWait
|
82
|
+
|
83
|
+
def do_wait(self) -> bool:
|
84
|
+
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
85
|
+
try:
|
86
|
+
s.connect(self.wait.address)
|
87
|
+
except ConnectionRefusedError:
|
88
|
+
return False
|
89
|
+
else:
|
90
|
+
return True
|
91
|
+
|
92
|
+
|
93
|
+
@waiter_for.register
|
94
|
+
def _(wait: ConnectWait) -> ConnectWaiter:
|
95
|
+
return ConnectWaiter(wait)
|
omlish/lang/__init__.py
CHANGED
omlish/lang/contextmanagers.py
CHANGED
@@ -18,6 +18,11 @@ T = ta.TypeVar('T')
|
|
18
18
|
##
|
19
19
|
|
20
20
|
|
21
|
+
class _NOT_SET: # noqa
|
22
|
+
def __new__(cls, *args, **kwargs): # noqa
|
23
|
+
raise TypeError
|
24
|
+
|
25
|
+
|
21
26
|
class ContextManaged:
|
22
27
|
def __enter__(self) -> ta.Self:
|
23
28
|
return self
|
@@ -31,21 +36,21 @@ class ContextManaged:
|
|
31
36
|
return None
|
32
37
|
|
33
38
|
|
34
|
-
class
|
35
|
-
def
|
36
|
-
|
37
|
-
|
39
|
+
class NopContextManager(ContextManaged):
|
40
|
+
def __init__(self, /, value: ta.Any = _NOT_SET) -> None:
|
41
|
+
super().__init__()
|
38
42
|
|
39
|
-
|
43
|
+
self._value = value
|
40
44
|
|
45
|
+
def __enter__(self):
|
46
|
+
if (value := self._value) is _NOT_SET:
|
47
|
+
return self
|
48
|
+
else:
|
49
|
+
return value
|
41
50
|
|
42
|
-
class NopContextManager:
|
43
51
|
def __init_subclass__(cls, **kwargs: ta.Any) -> None:
|
44
52
|
raise TypeError
|
45
53
|
|
46
|
-
def __call__(self, *args, **kwargs):
|
47
|
-
return NOP_CONTEXT_MANAGED
|
48
|
-
|
49
54
|
|
50
55
|
NOP_CONTEXT_MANAGER = NopContextManager()
|
51
56
|
|
@@ -350,7 +355,7 @@ def default_lock(value: DefaultLockable, default: DefaultLockable = None) -> Loc
|
|
350
355
|
return lambda: lock
|
351
356
|
|
352
357
|
elif value is False or value is None:
|
353
|
-
return NOP_CONTEXT_MANAGER
|
358
|
+
return lambda: NOP_CONTEXT_MANAGER
|
354
359
|
|
355
360
|
elif callable(value):
|
356
361
|
return value
|
omlish/lang/functions.py
CHANGED
@@ -190,6 +190,16 @@ class Args:
|
|
190
190
|
self.args = args
|
191
191
|
self.kwargs = kwargs
|
192
192
|
|
193
|
+
def update(self, *args: ta.Any, **kwargs: ta.Any) -> 'Args':
|
194
|
+
return Args(
|
195
|
+
*self.args,
|
196
|
+
*args,
|
197
|
+
**{
|
198
|
+
**self.kwargs,
|
199
|
+
**kwargs,
|
200
|
+
},
|
201
|
+
)
|
202
|
+
|
193
203
|
def __call__(self, fn: ta.Callable[..., T]) -> T:
|
194
204
|
return fn(*self.args, **self.kwargs)
|
195
205
|
|
omlish/libc.py
CHANGED
@@ -62,7 +62,6 @@ libc.free.argtypes = [ct.c_void_p]
|
|
62
62
|
|
63
63
|
|
64
64
|
class Malloc:
|
65
|
-
|
66
65
|
def __init__(self, sz: int) -> None:
|
67
66
|
super().__init__()
|
68
67
|
|
@@ -174,7 +173,6 @@ if LINUX:
|
|
174
173
|
|
175
174
|
|
176
175
|
class Mmap:
|
177
|
-
|
178
176
|
def __init__(
|
179
177
|
self,
|
180
178
|
length: int,
|
@@ -537,14 +535,14 @@ elif DARWIN:
|
|
537
535
|
|
538
536
|
|
539
537
|
if LINUX:
|
540
|
-
def gettid():
|
538
|
+
def gettid() -> int:
|
541
539
|
syscalls = {
|
542
540
|
'i386': 224, # unistd_32.h: #define __NR_gettid 224
|
543
541
|
'x86_64': 186, # unistd_64.h: #define __NR_gettid 186
|
544
542
|
'aarch64': 178, # asm-generic/unistd.h: #define __NR_gettid 178
|
545
543
|
}
|
546
544
|
try:
|
547
|
-
tid =
|
545
|
+
tid = libc.syscall(syscalls[platform.machine()])
|
548
546
|
except Exception: # noqa
|
549
547
|
tid = -1
|
550
548
|
return tid
|
omlish/lite/timeouts.py
CHANGED
@@ -8,7 +8,7 @@ import time
|
|
8
8
|
import typing as ta
|
9
9
|
|
10
10
|
|
11
|
-
TimeoutLike = ta.Union['Timeout', 'Timeout.Default', ta.Iterable['TimeoutLike'], float] # ta.TypeAlias
|
11
|
+
TimeoutLike = ta.Union['Timeout', ta.Type['Timeout.Default'], ta.Iterable['TimeoutLike'], float] # ta.TypeAlias
|
12
12
|
|
13
13
|
|
14
14
|
##
|
@@ -0,0 +1,88 @@
|
|
1
|
+
"""
|
2
|
+
TODO:
|
3
|
+
- F_SETLK mode
|
4
|
+
"""
|
5
|
+
import os
|
6
|
+
import typing as ta
|
7
|
+
|
8
|
+
from ... import lang
|
9
|
+
from ...argparse import all as ap
|
10
|
+
from ..pidfiles.pidfile import Pidfile
|
11
|
+
from ..pidfiles.pinning import PidfilePinner
|
12
|
+
from ..signals import parse_signal
|
13
|
+
|
14
|
+
|
15
|
+
class Cli(ap.Cli):
|
16
|
+
_PIDFILE_ARGS: ta.ClassVar[ta.Sequence[ap.Arg]] = [
|
17
|
+
ap.arg('pid-file'),
|
18
|
+
ap.arg('--create', action='store_true'),
|
19
|
+
]
|
20
|
+
|
21
|
+
def _pidfile_args(self) -> lang.Args:
|
22
|
+
return lang.Args(
|
23
|
+
self.args.pid_file,
|
24
|
+
inheritable=False,
|
25
|
+
no_create=not self._args.create,
|
26
|
+
)
|
27
|
+
|
28
|
+
def _args_pidfile(self) -> Pidfile:
|
29
|
+
return self._pidfile_args()(Pidfile)
|
30
|
+
|
31
|
+
#
|
32
|
+
|
33
|
+
@ap.cmd(*_PIDFILE_ARGS)
|
34
|
+
def read_no_verify(self) -> None:
|
35
|
+
with self._args_pidfile() as pidfile:
|
36
|
+
print(pidfile.read())
|
37
|
+
|
38
|
+
@ap.cmd(*_PIDFILE_ARGS)
|
39
|
+
def lock(self) -> None:
|
40
|
+
with self._args_pidfile() as pidfile:
|
41
|
+
pidfile.acquire_lock()
|
42
|
+
print(os.getpid())
|
43
|
+
input()
|
44
|
+
|
45
|
+
#
|
46
|
+
|
47
|
+
_PIDFILE_PINNER_ARGS: ta.ClassVar[ta.Sequence[ap.Arg]] = [
|
48
|
+
*_PIDFILE_ARGS,
|
49
|
+
ap.arg('--timeout', type=float),
|
50
|
+
]
|
51
|
+
|
52
|
+
def _pidfile_pinner_args(self) -> lang.Args:
|
53
|
+
return self._pidfile_args().update(
|
54
|
+
timeout=self._args.timeout,
|
55
|
+
)
|
56
|
+
|
57
|
+
def _args_pidfile_pinner(self) -> ta.ContextManager[int]:
|
58
|
+
return self._pidfile_pinner_args()(PidfilePinner.default_impl()().pin_pidfile_owner)
|
59
|
+
|
60
|
+
#
|
61
|
+
|
62
|
+
@ap.cmd(*_PIDFILE_PINNER_ARGS)
|
63
|
+
def read(self) -> None:
|
64
|
+
with self._args_pidfile_pinner() as pid:
|
65
|
+
print(pid)
|
66
|
+
|
67
|
+
@ap.cmd(*_PIDFILE_PINNER_ARGS)
|
68
|
+
def pin(self) -> None:
|
69
|
+
with self._args_pidfile_pinner() as pid:
|
70
|
+
print(pid)
|
71
|
+
input()
|
72
|
+
|
73
|
+
@ap.cmd(
|
74
|
+
*_PIDFILE_PINNER_ARGS,
|
75
|
+
ap.arg('signal'),
|
76
|
+
)
|
77
|
+
def kill(self) -> None:
|
78
|
+
sig = parse_signal(self._args.signal)
|
79
|
+
with self._args_pidfile_pinner() as pid:
|
80
|
+
os.kill(pid, sig)
|
81
|
+
|
82
|
+
|
83
|
+
def _main() -> None:
|
84
|
+
Cli().cli_run_and_exit()
|
85
|
+
|
86
|
+
|
87
|
+
if __name__ == '__main__':
|
88
|
+
_main()
|
omlish/os/pidfiles/pidfile.py
CHANGED
@@ -27,11 +27,13 @@ class Pidfile:
|
|
27
27
|
path: str,
|
28
28
|
*,
|
29
29
|
inheritable: bool = True,
|
30
|
+
no_create: bool = False,
|
30
31
|
) -> None:
|
31
32
|
super().__init__()
|
32
33
|
|
33
34
|
self._path = path
|
34
35
|
self._inheritable = inheritable
|
36
|
+
self._no_create = no_create
|
35
37
|
|
36
38
|
@property
|
37
39
|
def path(self) -> str:
|
@@ -56,8 +58,28 @@ class Pidfile:
|
|
56
58
|
|
57
59
|
#
|
58
60
|
|
61
|
+
_fd_to_dup: int
|
62
|
+
|
63
|
+
def dup(self) -> 'Pidfile':
|
64
|
+
fd = self._f.fileno()
|
65
|
+
dup = Pidfile(
|
66
|
+
self._path,
|
67
|
+
inheritable=self._inheritable,
|
68
|
+
)
|
69
|
+
dup._fd_to_dup = fd # noqa
|
70
|
+
return dup
|
71
|
+
|
72
|
+
#
|
73
|
+
|
59
74
|
def __enter__(self) -> 'Pidfile':
|
60
|
-
|
75
|
+
if hasattr(self, '_fd_to_dup'):
|
76
|
+
fd = os.dup(self._fd_to_dup)
|
77
|
+
del self._fd_to_dup
|
78
|
+
else:
|
79
|
+
ofl = os.O_RDWR
|
80
|
+
if not self._no_create:
|
81
|
+
ofl |= os.O_CREAT
|
82
|
+
fd = os.open(self._path, ofl, 0o600)
|
61
83
|
|
62
84
|
try:
|
63
85
|
if self._inheritable:
|
omlish/os/pidfiles/pinning.py
CHANGED
@@ -27,6 +27,7 @@ import typing as ta
|
|
27
27
|
|
28
28
|
from ...diag.lslocks import LslocksCommand
|
29
29
|
from ...diag.lsof import LsofCommand
|
30
|
+
from ...lite.check import check
|
30
31
|
from ...lite.timeouts import Timeout
|
31
32
|
from ...lite.timeouts import TimeoutLike
|
32
33
|
from ...subprocesses.sync import subprocesses # noqa
|
@@ -64,13 +65,21 @@ class PidfilePinner(abc.ABC):
|
|
64
65
|
path: str,
|
65
66
|
*,
|
66
67
|
timeout: ta.Optional[TimeoutLike] = None,
|
68
|
+
inheritable: bool = False, # Present to match Pidfile kwargs for convenience, but enforced to be False.
|
69
|
+
**kwargs: ta.Any,
|
67
70
|
) -> ta.Iterator[int]:
|
71
|
+
check.arg(not inheritable)
|
72
|
+
|
68
73
|
timeout = Timeout.of(timeout)
|
69
74
|
|
70
75
|
if not os.path.isfile(path):
|
71
76
|
raise self.NoOwnerError
|
72
77
|
|
73
|
-
with Pidfile(
|
78
|
+
with Pidfile(
|
79
|
+
path,
|
80
|
+
inheritable=False,
|
81
|
+
**kwargs,
|
82
|
+
) as pf:
|
74
83
|
try:
|
75
84
|
with self._pin_pidfile_owner(pf, timeout) as pid:
|
76
85
|
yield pid
|
omlish/os/signals.py
ADDED
@@ -0,0 +1,15 @@
|
|
1
|
+
# ruff: noqa: UP006 UP007
|
2
|
+
# @omlish-lite
|
3
|
+
import signal
|
4
|
+
|
5
|
+
|
6
|
+
def parse_signal(s: str) -> int:
|
7
|
+
try:
|
8
|
+
return int(s)
|
9
|
+
except ValueError:
|
10
|
+
pass
|
11
|
+
|
12
|
+
s = s.upper()
|
13
|
+
if not s.startswith('SIG'):
|
14
|
+
s = 'SIG' + s
|
15
|
+
return signal.Signals[s] # noqa
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.2
|
2
2
|
Name: omlish
|
3
|
-
Version: 0.0.0.
|
3
|
+
Version: 0.0.0.dev232
|
4
4
|
Summary: omlish
|
5
5
|
Author: wrmsr
|
6
6
|
License: BSD-3-Clause
|
@@ -16,7 +16,7 @@ Provides-Extra: all
|
|
16
16
|
Requires-Dist: anyio~=4.8; extra == "all"
|
17
17
|
Requires-Dist: sniffio~=1.3; extra == "all"
|
18
18
|
Requires-Dist: greenlet~=3.1; extra == "all"
|
19
|
-
Requires-Dist: trio~=0.
|
19
|
+
Requires-Dist: trio~=0.29; extra == "all"
|
20
20
|
Requires-Dist: trio-asyncio~=0.15; extra == "all"
|
21
21
|
Requires-Dist: lz4~=4.4; extra == "all"
|
22
22
|
Requires-Dist: python-snappy~=0.7; extra == "all"
|
@@ -24,7 +24,7 @@ Requires-Dist: zstandard~=0.23; extra == "all"
|
|
24
24
|
Requires-Dist: brotli~=1.1; extra == "all"
|
25
25
|
Requires-Dist: asttokens~=3.0; extra == "all"
|
26
26
|
Requires-Dist: executing~=2.2; extra == "all"
|
27
|
-
Requires-Dist: psutil~=
|
27
|
+
Requires-Dist: psutil~=7.0; extra == "all"
|
28
28
|
Requires-Dist: orjson~=3.10; extra == "all"
|
29
29
|
Requires-Dist: ujson~=5.10; extra == "all"
|
30
30
|
Requires-Dist: pyyaml~=6.0; extra == "all"
|
@@ -54,7 +54,7 @@ Provides-Extra: async
|
|
54
54
|
Requires-Dist: anyio~=4.8; extra == "async"
|
55
55
|
Requires-Dist: sniffio~=1.3; extra == "async"
|
56
56
|
Requires-Dist: greenlet~=3.1; extra == "async"
|
57
|
-
Requires-Dist: trio~=0.
|
57
|
+
Requires-Dist: trio~=0.29; extra == "async"
|
58
58
|
Requires-Dist: trio-asyncio~=0.15; extra == "async"
|
59
59
|
Provides-Extra: compress
|
60
60
|
Requires-Dist: lz4~=4.4; extra == "compress"
|
@@ -64,7 +64,7 @@ Requires-Dist: brotli~=1.1; extra == "compress"
|
|
64
64
|
Provides-Extra: diag
|
65
65
|
Requires-Dist: asttokens~=3.0; extra == "diag"
|
66
66
|
Requires-Dist: executing~=2.2; extra == "diag"
|
67
|
-
Requires-Dist: psutil~=
|
67
|
+
Requires-Dist: psutil~=7.0; extra == "diag"
|
68
68
|
Provides-Extra: formats
|
69
69
|
Requires-Dist: orjson~=3.10; extra == "formats"
|
70
70
|
Requires-Dist: ujson~=5.10; extra == "formats"
|
@@ -1,5 +1,5 @@
|
|
1
|
-
omlish/.manifests.json,sha256=
|
2
|
-
omlish/__about__.py,sha256
|
1
|
+
omlish/.manifests.json,sha256=vQTAIvR8OblSq-uP2GUfnbei0RnmAnM5j0T1-OToh9E,8253
|
2
|
+
omlish/__about__.py,sha256=-wogzlxSt20e8KuTBCH4MRnaUQhIUXLRZdyQ82uUp4E,3380
|
3
3
|
omlish/__init__.py,sha256=SsyiITTuK0v74XpKV8dqNaCmjOlan1JZKrHQv5rWKPA,253
|
4
4
|
omlish/c3.py,sha256=ubu7lHwss5V4UznbejAI0qXhXahrU01MysuHOZI9C4U,8116
|
5
5
|
omlish/cached.py,sha256=UI-XTFBwA6YXWJJJeBn-WkwBkfzDjLBBaZf4nIJA9y0,510
|
@@ -7,7 +7,7 @@ omlish/check.py,sha256=THqm6jD1a0skAO5EC8SOVg58yq96Vk5wcuruBkCYxyU,2016
|
|
7
7
|
omlish/datetimes.py,sha256=HajeM1kBvwlTa-uR1TTZHmZ3zTPnnUr1uGGQhiO1XQ0,2152
|
8
8
|
omlish/defs.py,sha256=9uUjJuVIbCBL3g14fyzAp-9gH935MFofvlfOGwcBIaM,4913
|
9
9
|
omlish/dynamic.py,sha256=kIZokHHid8a0pIAPXMNiXrVJvJJyBnY49WP1a2m-HUQ,6525
|
10
|
-
omlish/libc.py,sha256=
|
10
|
+
omlish/libc.py,sha256=8K4c66YV1ziJerl5poAAYCmsV-VSsHkT3EHhPW04ufg,15639
|
11
11
|
omlish/outcome.py,sha256=ABIE0zjjTyTNtn-ZqQ_9_mUzLiBQ3sDAyqc9JVD8N2k,7852
|
12
12
|
omlish/runmodule.py,sha256=PWvuAaJ9wQQn6bx9ftEL3_d04DyotNn8dR_twm2pgw0,700
|
13
13
|
omlish/shlex.py,sha256=bsW2XUD8GiMTUTDefJejZ5AyqT1pTgWMPD0BMoF02jE,248
|
@@ -175,6 +175,13 @@ omlish/configs/processing/matching.py,sha256=R64RxpPB1uX5Ztvvk2dQ2xi_xwlaxkxQgZw
|
|
175
175
|
omlish/configs/processing/names.py,sha256=weHmaTclzgM9lUn3aBtw-kwZ3mc2N-CZlFg3Kd_UsKo,1093
|
176
176
|
omlish/configs/processing/rewriting.py,sha256=v7PfHtuTn5v_5Y6Au7oMN2Z0nxAMy1iYyO5CXnTvZhs,4226
|
177
177
|
omlish/configs/processing/strings.py,sha256=qFS2oh6z02IaM_q4lTKLdufzkJqAJ6J-Qjrz5S-QJoM,826
|
178
|
+
omlish/daemons/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
179
|
+
omlish/daemons/daemon.py,sha256=lnMgGrPeZ3pjhuXPU-7DX2XGEdNG0kP6HAejCQPSGtk,2976
|
180
|
+
omlish/daemons/launching.py,sha256=mhtkuAO16STcznUl3rrX9pacfrKbPQRCP2AllKL4B70,3664
|
181
|
+
omlish/daemons/reparent.py,sha256=UaG2X6VJHJPOlUwHPNRH3aWGgF0Fg771jjO9IRPLlyY,280
|
182
|
+
omlish/daemons/spawning.py,sha256=cx00xeqSrfhlFbjCtKqaBHvMuHwB9hdjuKNHzAAo_dw,4030
|
183
|
+
omlish/daemons/targets.py,sha256=scq6BYgzi0H2apfgD74U0D8_msQuYAel3Qiij74YWVo,1501
|
184
|
+
omlish/daemons/waiting.py,sha256=RfgD1L33QQVbD2431dkKZGE4w6DUcGvYeRXXi8puAP4,1676
|
178
185
|
omlish/dataclasses/__init__.py,sha256=D7I6ZJjEFeLN2re8oOZ_7JKWiG2plrPnaUFq3iEXYmQ,1553
|
179
186
|
omlish/dataclasses/utils.py,sha256=N2seT8cJtfOv-41D7F3E-q4us-FCTQmnxxPv3dt1OcI,3796
|
180
187
|
omlish/dataclasses/impl/LICENSE,sha256=Oy-B_iHRgcSZxZolbI4ZaEVdZonSaaqFNzv7avQdo78,13936
|
@@ -385,15 +392,15 @@ omlish/iterators/iterators.py,sha256=ghI4dO6WPyyFOLTIIMaHQ_IOy2xXaFpGPqveZ5YGIBU
|
|
385
392
|
omlish/iterators/recipes.py,sha256=53mkexitMhkwXQZbL6DrhpT0WePQ_56uXd5Jaw3DfzI,467
|
386
393
|
omlish/iterators/tools.py,sha256=Pi4ybXytUXVZ3xwK89xpPImQfYYId9p1vIFQvVqVLqA,2551
|
387
394
|
omlish/iterators/unique.py,sha256=0jAX3kwzVfRNhe0Tmh7kVP_Q2WBIn8POo_O-rgFV0rQ,1390
|
388
|
-
omlish/lang/__init__.py,sha256=
|
395
|
+
omlish/lang/__init__.py,sha256=bE8ra_QM5M4k8v3qFtreTjQswaZClIOaqXt9fE43MD0,4079
|
389
396
|
omlish/lang/cached.py,sha256=tQaqMu1LID0q4NSTk5vPXsgxIBWSFAmjs5AhQoEHoCQ,7833
|
390
397
|
omlish/lang/clsdct.py,sha256=sJYadm-fwzti-gsi98knR5qQUxriBmOqQE_qz3RopNk,1743
|
391
398
|
omlish/lang/cmp.py,sha256=5vbzWWbqdzDmNKAGL19z6ZfUKe5Ci49e-Oegf9f4BsE,1346
|
392
|
-
omlish/lang/contextmanagers.py,sha256=
|
399
|
+
omlish/lang/contextmanagers.py,sha256=Mrn8NJ3pP0Zxi-IoGqSjZDdWUctsyee2vrZ2FtZvNmo,10529
|
393
400
|
omlish/lang/datetimes.py,sha256=ehI_DhQRM-bDxAavnp470XcekbbXc4Gdw9y1KpHDJT0,223
|
394
401
|
omlish/lang/descriptors.py,sha256=njkYDS1gn5p4-3v1jr-s_srauC7tvvt571RjE7Q4LXE,6616
|
395
402
|
omlish/lang/exceptions.py,sha256=qJBo3NU1mOWWm-NhQUHCY5feYXR3arZVyEHinLsmRH4,47
|
396
|
-
omlish/lang/functions.py,sha256=
|
403
|
+
omlish/lang/functions.py,sha256=0ql9EXA_gEEhvUVzMJCjVhEnVtHecsLKmfmAXuQqeGY,4388
|
397
404
|
omlish/lang/generators.py,sha256=5LX17j-Ej3QXhwBgZvRTm_dq3n9veC4IOUcVmvSu2vU,5243
|
398
405
|
omlish/lang/imports.py,sha256=FMPz1lIDmuoapM88uabXeZbeJX1uCT8pkHeukC3X4qc,10129
|
399
406
|
omlish/lang/iterables.py,sha256=HOjcxOwyI5bBApDLsxRAGGhTTmw7fdZl2kEckxRVl-0,1994
|
@@ -434,7 +441,7 @@ omlish/lite/resources.py,sha256=YNSmX1Ohck1aoWRs55a-o5ChVbFJIQhtbqE-XwF55Oc,326
|
|
434
441
|
omlish/lite/runtime.py,sha256=XQo408zxTdJdppUZqOWHyeUR50VlCpNIExNGHz4U6O4,459
|
435
442
|
omlish/lite/secrets.py,sha256=3Mz3V2jf__XU9qNHcH56sBSw95L3U2UPL24bjvobG0c,816
|
436
443
|
omlish/lite/strings.py,sha256=QGxT1Yh4oI8ycsfeobxnjEhvDob_GiAKLeIhZwo1j24,1986
|
437
|
-
omlish/lite/timeouts.py,sha256=
|
444
|
+
omlish/lite/timeouts.py,sha256=lhXo0zwpLM7nr2-AliBRui2BO9jh7B8ALxetwOq6hYI,4968
|
438
445
|
omlish/lite/timing.py,sha256=aVu3hEDB_jyTF_ryZI7iU-xg4q8CNwqpp9Apfru_iwY,196
|
439
446
|
omlish/lite/types.py,sha256=fP5EMyBdEp2LmDxcHjUDtwAMdR06ISr9lKOL7smWfHM,140
|
440
447
|
omlish/lite/typing.py,sha256=U3-JaEnkDSYxK4tsu_MzUn3RP6qALBe5FXQXpD-licE,1090
|
@@ -512,6 +519,7 @@ omlish/os/forkhooks.py,sha256=yjodOvs90ClXskv5oBIJbHn0Y7dzajLmZmOpRMKbyxM,5656
|
|
512
519
|
omlish/os/journald.py,sha256=2nI8Res1poXkbLc31--MPUlzYMESnCcPUkIxDOCjZW0,3903
|
513
520
|
omlish/os/linux.py,sha256=whJ6scwMKSFBdXiVhJW0BCpJV4jOGMr-a_a3Bhwz6Ls,18938
|
514
521
|
omlish/os/paths.py,sha256=hqPiyg_eYaRoIVPdAeX4oeLEV4Kpln_XsH0tHvbOf8Q,844
|
522
|
+
omlish/os/signals.py,sha256=FtzkovLb58N3vNdfxflUeXWFCqqKzseCjk5kBdWT-ds,267
|
515
523
|
omlish/os/sizes.py,sha256=ohkALLvqSqBX4iR-7DMKJ4pfOCRdZXV8htH4QywUNM0,152
|
516
524
|
omlish/os/temp.py,sha256=P97KiVeNB7rfGn4tlgU5ro86JUxAsiphLMlxsjQgfB0,1198
|
517
525
|
omlish/os/deathpacts/__init__.py,sha256=IFJkHVWff-VhBbQX38th1RlmjUF2ptKh5TPIzP9Ei2M,229
|
@@ -519,9 +527,11 @@ omlish/os/deathpacts/base.py,sha256=EGN3BWSXPv0s9kl_QLrWE31hTybDHCmsLc_w3U2VyHc,
|
|
519
527
|
omlish/os/deathpacts/heartbeatfile.py,sha256=OybdvhM2kxBTuoJWOJJ5LcX-0lg3jTOvvD2HUunxDWU,1731
|
520
528
|
omlish/os/deathpacts/pipe.py,sha256=ZH-l-fIKyurocCehqOgvaYRurxIEMWe8D7l2dsJeGws,3214
|
521
529
|
omlish/os/pidfiles/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
530
|
+
omlish/os/pidfiles/__main__.py,sha256=AF8TwjK4xgHVnoLAP9dIWgKvT0vGhHJlfDW0tKZ7tx4,200
|
531
|
+
omlish/os/pidfiles/cli.py,sha256=2SSsP4O3VdpsDIMAkWgWSjh_YNIPzCD9l5LNN2qrIjo,2074
|
522
532
|
omlish/os/pidfiles/manager.py,sha256=qSEwNaWT1KOAnU0KxliwvU_uowme5jyf1FyIPsGwnTY,2391
|
523
|
-
omlish/os/pidfiles/pidfile.py,sha256=
|
524
|
-
omlish/os/pidfiles/pinning.py,sha256=
|
533
|
+
omlish/os/pidfiles/pidfile.py,sha256=yy14NCjvsCdXzlJjZzkw4vbzjt_FXEaUS5SbYT3dGY0,4277
|
534
|
+
omlish/os/pidfiles/pinning.py,sha256=v9RlJ4BnJZcaZZXiiRqbmzLluaSOkeeEb_WrbKEClBQ,6643
|
525
535
|
omlish/reflect/__init__.py,sha256=Er2yBHibVO16hFNA1szQF2_f43Y3HRCBWtS-fjsOIYc,798
|
526
536
|
omlish/reflect/inspect.py,sha256=WCo2YpBYauKw6k758FLlZ_H4Q05rgVPs96fEv9w6zHQ,1538
|
527
537
|
omlish/reflect/ops.py,sha256=RJ6jzrM4ieFsXzWyNXWV43O_WgzEaUvlHSc5N2ezW2A,2044
|
@@ -695,9 +705,9 @@ omlish/text/indent.py,sha256=YjtJEBYWuk8--b9JU_T6q4yxV85_TR7VEVr5ViRCFwk,1336
|
|
695
705
|
omlish/text/minja.py,sha256=jZC-fp3Xuhx48ppqsf2Sf1pHbC0t8XBB7UpUUoOk2Qw,5751
|
696
706
|
omlish/text/parts.py,sha256=JkNZpyR2tv2CNcTaWJJhpQ9E4F0yPR8P_YfDbZfMtwQ,6182
|
697
707
|
omlish/text/random.py,sha256=jNWpqiaKjKyTdMXC-pWAsSC10AAP-cmRRPVhm59ZWLk,194
|
698
|
-
omlish-0.0.0.
|
699
|
-
omlish-0.0.0.
|
700
|
-
omlish-0.0.0.
|
701
|
-
omlish-0.0.0.
|
702
|
-
omlish-0.0.0.
|
703
|
-
omlish-0.0.0.
|
708
|
+
omlish-0.0.0.dev232.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
|
709
|
+
omlish-0.0.0.dev232.dist-info/METADATA,sha256=q9xYAS00xDATKtVjZBhWNKwkSnMAIV0By4EIl_2vqFs,4176
|
710
|
+
omlish-0.0.0.dev232.dist-info/WHEEL,sha256=In9FTNxeP60KnTkGw7wk6mJPYd_dQSjEZmXdBdMCI-8,91
|
711
|
+
omlish-0.0.0.dev232.dist-info/entry_points.txt,sha256=Lt84WvRZJskWCAS7xnQGZIeVWksprtUHj0llrvVmod8,35
|
712
|
+
omlish-0.0.0.dev232.dist-info/top_level.txt,sha256=pePsKdLu7DvtUiecdYXJ78iO80uDNmBlqe-8hOzOmfs,7
|
713
|
+
omlish-0.0.0.dev232.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|