ominfra 0.0.0.dev166__py3-none-any.whl → 0.0.0.dev167__py3-none-any.whl

Sign up to get free protection for your applications and to get access to all the features.
@@ -5,7 +5,9 @@ import typing as ta
5
5
 
6
6
  from omlish.lite.cached import cached_nullary
7
7
  from omlish.lite.check import check
8
+ from omlish.os.paths import relative_symlink
8
9
 
10
+ from .conf import DeployConfManager
9
11
  from .git import DeployGitManager
10
12
  from .paths import DeployPath
11
13
  from .paths import DeployPathOwner
@@ -36,12 +38,16 @@ class DeployAppManager(DeployPathOwner):
36
38
  self,
37
39
  *,
38
40
  deploy_home: ta.Optional[DeployHome] = None,
41
+
42
+ conf: DeployConfManager,
39
43
  git: DeployGitManager,
40
44
  venvs: DeployVenvManager,
41
45
  ) -> None:
42
46
  super().__init__()
43
47
 
44
48
  self._deploy_home = deploy_home
49
+
50
+ self._conf = conf
45
51
  self._git = git
46
52
  self._venvs = venvs
47
53
 
@@ -51,24 +57,69 @@ class DeployAppManager(DeployPathOwner):
51
57
 
52
58
  def get_owned_deploy_paths(self) -> ta.AbstractSet[DeployPath]:
53
59
  return {
54
- DeployPath.parse('apps/@app/@tag/'),
60
+ DeployPath.parse('apps/@app/current'),
61
+ DeployPath.parse('apps/@app/deploying'),
62
+
63
+ DeployPath.parse('apps/@app/tags/@tag/conf/'),
64
+ DeployPath.parse('apps/@app/tags/@tag/git/'),
65
+ DeployPath.parse('apps/@app/tags/@tag/venv/'),
55
66
  }
56
67
 
57
68
  async def prepare_app(
58
69
  self,
59
70
  spec: DeploySpec,
60
71
  ) -> None:
61
- app_tag = DeployAppTag(spec.app, make_deploy_tag(spec.checkout.rev, spec.key()))
62
- app_dir = os.path.join(self._dir(), spec.app, app_tag.tag)
72
+ app_tag = DeployAppTag(spec.app, make_deploy_tag(spec.git.rev, spec.key()))
73
+
74
+ #
75
+
76
+ app_dir = os.path.join(self._dir(), spec.app)
77
+ os.makedirs(app_dir, exist_ok=True)
63
78
 
64
79
  #
65
80
 
81
+ tag_dir = os.path.join(app_dir, 'tags', app_tag.tag)
82
+ os.makedirs(tag_dir)
83
+
84
+ #
85
+
86
+ deploying_file = os.path.join(app_dir, 'deploying')
87
+ current_file = os.path.join(app_dir, 'current')
88
+
89
+ if os.path.exists(deploying_file):
90
+ os.unlink(deploying_file)
91
+ relative_symlink(tag_dir, deploying_file, target_is_directory=True)
92
+
93
+ #
94
+
95
+ git_dir = os.path.join(tag_dir, 'git')
66
96
  await self._git.checkout(
67
- spec.checkout,
68
- app_dir,
97
+ spec.git,
98
+ git_dir,
69
99
  )
70
100
 
71
101
  #
72
102
 
73
103
  if spec.venv is not None:
74
- await self._venvs.setup_app_venv(app_tag, spec.venv)
104
+ venv_dir = os.path.join(tag_dir, 'venv')
105
+ await self._venvs.setup_venv(
106
+ spec.venv,
107
+ git_dir,
108
+ venv_dir,
109
+ )
110
+
111
+ #
112
+
113
+ if spec.conf is not None:
114
+ conf_dir = os.path.join(tag_dir, 'conf')
115
+ conf_link_dir = os.path.join(current_file, 'conf')
116
+ await self._conf.write_conf(
117
+ spec.conf,
118
+ conf_dir,
119
+ app_tag,
120
+ conf_link_dir,
121
+ )
122
+
123
+ #
124
+
125
+ os.replace(deploying_file, current_file)
@@ -0,0 +1,143 @@
1
+ # ruff: noqa: UP006 UP007
2
+ """
3
+ TODO:
4
+ - @conf DeployPathPlaceholder? :|
5
+ - post-deploy: remove any dir_links not present in new spec
6
+ - * only if succeeded * - otherwise, remove any dir_links present in new spec but not previously present?
7
+ - no such thing as 'previously present'.. build a 'deploy state' and pass it back?
8
+ - ** whole thing can be atomic **
9
+ - 1) new atomic temp dir
10
+ - 2) for each subdir not needing modification, hardlink into temp dir
11
+ - 3) for each subdir needing modification, new subdir, hardlink all files not needing modification
12
+ - 4) write (or if deleting, omit) new files
13
+ - 5) swap top level
14
+ - ** whole deploy can be atomic(-ish) - do this for everything **
15
+ - just a '/deploy/current' dir
16
+ - some things (venvs) cannot be moved, thus the /deploy/venvs dir
17
+ - ** ensure (enforce) equivalent relpath nesting
18
+ """
19
+ import os.path
20
+ import typing as ta
21
+
22
+ from omlish.lite.check import check
23
+ from omlish.os.paths import is_path_in_dir
24
+ from omlish.os.paths import relative_symlink
25
+
26
+ from .paths import SingleDirDeployPathOwner
27
+ from .specs import AppDeployConfLink
28
+ from .specs import DeployConfFile
29
+ from .specs import DeployConfLink
30
+ from .specs import DeployConfSpec
31
+ from .specs import TagDeployConfLink
32
+ from .types import DeployAppTag
33
+ from .types import DeployHome
34
+
35
+
36
+ class DeployConfManager(SingleDirDeployPathOwner):
37
+ def __init__(
38
+ self,
39
+ *,
40
+ deploy_home: ta.Optional[DeployHome] = None,
41
+ ) -> None:
42
+ super().__init__(
43
+ owned_dir='conf',
44
+ deploy_home=deploy_home,
45
+ )
46
+
47
+ async def _write_conf_file(
48
+ self,
49
+ cf: DeployConfFile,
50
+ conf_dir: str,
51
+ ) -> None:
52
+ conf_file = os.path.join(conf_dir, cf.path)
53
+ check.arg(is_path_in_dir(conf_dir, conf_file))
54
+
55
+ os.makedirs(os.path.dirname(conf_file), exist_ok=True)
56
+
57
+ with open(conf_file, 'w') as f: # noqa
58
+ f.write(cf.body)
59
+
60
+ async def _make_conf_link(
61
+ self,
62
+ link: DeployConfLink,
63
+ conf_dir: str,
64
+ app_tag: DeployAppTag,
65
+ link_dir: str,
66
+ ) -> None:
67
+ link_src = os.path.join(conf_dir, link.src)
68
+ check.arg(is_path_in_dir(conf_dir, link_src))
69
+
70
+ is_link_dir = link.src.endswith('/')
71
+ if is_link_dir:
72
+ check.arg(link.src.count('/') == 1)
73
+ check.arg(os.path.isdir(link_src))
74
+ link_dst_pfx = link.src
75
+ link_dst_sfx = ''
76
+
77
+ elif '/' in link.src:
78
+ check.arg(os.path.isfile(link_src))
79
+ d, f = os.path.split(link.src)
80
+ # TODO: check filename :|
81
+ link_dst_pfx = d + '/'
82
+ link_dst_sfx = '-' + f
83
+
84
+ else:
85
+ check.arg(os.path.isfile(link_src))
86
+ if '.' in link.src:
87
+ l, _, r = link.src.partition('.')
88
+ link_dst_pfx = l + '/'
89
+ link_dst_sfx = '.' + r
90
+ else:
91
+ link_dst_pfx = link.src + '/'
92
+ link_dst_sfx = ''
93
+
94
+ if isinstance(link, AppDeployConfLink):
95
+ link_dst_mid = str(app_tag.app)
96
+ sym_root = link_dir
97
+ elif isinstance(link, TagDeployConfLink):
98
+ link_dst_mid = '-'.join([app_tag.app, app_tag.tag])
99
+ sym_root = conf_dir
100
+ else:
101
+ raise TypeError(link)
102
+
103
+ link_dst = ''.join([
104
+ link_dst_pfx,
105
+ link_dst_mid,
106
+ link_dst_sfx,
107
+ ])
108
+
109
+ root_conf_dir = self._make_dir()
110
+ sym_src = os.path.join(sym_root, link.src)
111
+ sym_dst = os.path.join(root_conf_dir, link_dst)
112
+ check.arg(is_path_in_dir(root_conf_dir, sym_dst))
113
+
114
+ os.makedirs(os.path.dirname(sym_dst), exist_ok=True)
115
+ relative_symlink(sym_src, sym_dst, target_is_directory=is_link_dir)
116
+
117
+ async def write_conf(
118
+ self,
119
+ spec: DeployConfSpec,
120
+ conf_dir: str,
121
+ app_tag: DeployAppTag,
122
+ link_dir: str,
123
+ ) -> None:
124
+ conf_dir = os.path.abspath(conf_dir)
125
+ os.makedirs(conf_dir)
126
+
127
+ #
128
+
129
+ for cf in spec.files or []:
130
+ await self._write_conf_file(
131
+ cf,
132
+ conf_dir,
133
+ )
134
+
135
+ #
136
+
137
+ for link in spec.links or []:
138
+ await self._make_conf_link(
139
+ link,
140
+ conf_dir,
141
+ app_tag,
142
+ link_dir,
143
+ )
@@ -18,8 +18,8 @@ from omlish.lite.check import check
18
18
  from omlish.os.atomics import AtomicPathSwapping
19
19
 
20
20
  from .paths import SingleDirDeployPathOwner
21
- from .specs import DeployGitCheckout
22
21
  from .specs import DeployGitRepo
22
+ from .specs import DeployGitSpec
23
23
  from .types import DeployHome
24
24
  from .types import DeployRev
25
25
 
@@ -95,7 +95,7 @@ class DeployGitManager(SingleDirDeployPathOwner):
95
95
 
96
96
  #
97
97
 
98
- async def checkout(self, checkout: DeployGitCheckout, dst_dir: str) -> None:
98
+ async def checkout(self, spec: DeployGitSpec, dst_dir: str) -> None:
99
99
  check.state(not os.path.exists(dst_dir))
100
100
  with self._git._atomics.begin_atomic_path_swap( # noqa
101
101
  'dir',
@@ -103,14 +103,14 @@ class DeployGitManager(SingleDirDeployPathOwner):
103
103
  auto_commit=True,
104
104
  make_dirs=True,
105
105
  ) as dst_swap:
106
- await self.fetch(checkout.rev)
106
+ await self.fetch(spec.rev)
107
107
 
108
108
  dst_call = functools.partial(asyncio_subprocesses.check_call, cwd=dst_swap.tmp_path)
109
109
  await dst_call('git', 'init')
110
110
 
111
111
  await dst_call('git', 'remote', 'add', 'local', self._dir)
112
- await dst_call('git', 'fetch', '--depth=1', 'local', checkout.rev)
113
- await dst_call('git', 'checkout', checkout.rev, *(checkout.subtrees or []))
112
+ await dst_call('git', 'fetch', '--depth=1', 'local', spec.rev)
113
+ await dst_call('git', 'checkout', spec.rev, *(spec.subtrees or []))
114
114
 
115
115
  def get_repo_dir(self, repo: DeployGitRepo) -> RepoDir:
116
116
  try:
@@ -119,5 +119,9 @@ class DeployGitManager(SingleDirDeployPathOwner):
119
119
  repo_dir = self._repo_dirs[repo] = DeployGitManager.RepoDir(self, repo)
120
120
  return repo_dir
121
121
 
122
- async def checkout(self, checkout: DeployGitCheckout, dst_dir: str) -> None:
123
- await self.get_repo_dir(checkout.repo).checkout(checkout, dst_dir)
122
+ async def checkout(
123
+ self,
124
+ spec: DeployGitSpec,
125
+ dst_dir: str,
126
+ ) -> None:
127
+ await self.get_repo_dir(spec.repo).checkout(spec, dst_dir)
@@ -11,10 +11,13 @@ from ..commands.inject import bind_command
11
11
  from .apps import DeployAppManager
12
12
  from .commands import DeployCommand
13
13
  from .commands import DeployCommandExecutor
14
+ from .conf import DeployConfManager
14
15
  from .config import DeployConfig
15
16
  from .git import DeployGitManager
16
17
  from .interp import InterpCommand
17
18
  from .interp import InterpCommandExecutor
19
+ from .paths import DeployPathOwner
20
+ from .paths import DeployPathOwners
18
21
  from .tmp import DeployTmpManager
19
22
  from .types import DeployHome
20
23
  from .venvs import DeployVenvManager
@@ -26,23 +29,45 @@ def bind_deploy(
26
29
  ) -> InjectorBindings:
27
30
  lst: ta.List[InjectorBindingOrBindings] = [
28
31
  inj.bind(deploy_config),
32
+ ]
33
+
34
+ #
35
+
36
+ def bind_manager(cls: type) -> InjectorBindings:
37
+ return inj.as_bindings(
38
+ inj.bind(cls, singleton=True),
39
+
40
+ *([inj.bind(DeployPathOwner, to_key=cls, array=True)] if issubclass(cls, DeployPathOwner) else []),
41
+ )
29
42
 
30
- #
43
+ lst.extend([
44
+ inj.bind_array(DeployPathOwner),
45
+ inj.bind_array_type(DeployPathOwner, DeployPathOwners),
46
+ ])
31
47
 
32
- inj.bind(DeployAppManager, singleton=True),
48
+ #
33
49
 
34
- inj.bind(DeployGitManager, singleton=True),
50
+ lst.extend([
51
+ bind_manager(DeployAppManager),
35
52
 
36
- inj.bind(DeployTmpManager, singleton=True),
53
+ bind_manager(DeployConfManager),
54
+
55
+ bind_manager(DeployGitManager),
56
+
57
+ bind_manager(DeployTmpManager),
37
58
  inj.bind(AtomicPathSwapping, to_key=DeployTmpManager),
38
59
 
39
- inj.bind(DeployVenvManager, singleton=True),
60
+ bind_manager(DeployVenvManager),
61
+ ])
40
62
 
41
- #
63
+ #
42
64
 
65
+ lst.extend([
43
66
  bind_command(DeployCommand, DeployCommandExecutor),
44
67
  bind_command(InterpCommand, InterpCommandExecutor),
45
- ]
68
+ ])
69
+
70
+ #
46
71
 
47
72
  if (dh := deploy_config.deploy_home) is not None:
48
73
  dh = os.path.abspath(os.path.expanduser(dh))
@@ -1,11 +1,12 @@
1
1
  # ruff: noqa: UP006 UP007
2
2
  """
3
3
  TODO:
4
- - run/pidfile
4
+ - run/{.pid,.sock}
5
5
  - logs/...
6
6
  - current symlink
7
7
  - conf/{nginx,supervisor}
8
8
  - env/?
9
+ - apps/<app>/shared
9
10
  """
10
11
  import abc
11
12
  import dataclasses as dc
@@ -208,6 +209,9 @@ class DeployPathOwner(abc.ABC):
208
209
  raise NotImplementedError
209
210
 
210
211
 
212
+ DeployPathOwners = ta.NewType('DeployPathOwners', ta.Sequence[DeployPathOwner])
213
+
214
+
211
215
  class SingleDirDeployPathOwner(DeployPathOwner, abc.ABC):
212
216
  def __init__(
213
217
  self,
@@ -1,4 +1,5 @@
1
1
  # ruff: noqa: UP006 UP007
2
+ import abc
2
3
  import dataclasses as dc
3
4
  import hashlib
4
5
  import typing as ta
@@ -14,6 +15,17 @@ from .types import DeployRev
14
15
  ##
15
16
 
16
17
 
18
+ def check_valid_deploy_spec_path(s: str) -> str:
19
+ check.non_empty_str(s)
20
+ for c in ['..', '//']:
21
+ check.not_in(c, s)
22
+ check.arg(not s.startswith('/'))
23
+ return s
24
+
25
+
26
+ ##
27
+
28
+
17
29
  @dc.dataclass(frozen=True)
18
30
  class DeployGitRepo:
19
31
  host: ta.Optional[str] = None
@@ -26,7 +38,7 @@ class DeployGitRepo:
26
38
 
27
39
 
28
40
  @dc.dataclass(frozen=True)
29
- class DeployGitCheckout:
41
+ class DeployGitSpec:
30
42
  repo: DeployGitRepo
31
43
  rev: DeployRev
32
44
 
@@ -52,8 +64,62 @@ class DeployVenvSpec:
52
64
 
53
65
  use_uv: bool = False
54
66
 
67
+
68
+ ##
69
+
70
+
71
+ @dc.dataclass(frozen=True)
72
+ class DeployConfFile:
73
+ path: str
74
+ body: str
75
+
55
76
  def __post_init__(self) -> None:
56
- hash(self)
77
+ check_valid_deploy_spec_path(self.path)
78
+
79
+
80
+ #
81
+
82
+
83
+ @dc.dataclass(frozen=True)
84
+ class DeployConfLink(abc.ABC): # noqa
85
+ """
86
+ May be either:
87
+ - @conf(.ext)* - links a single file in root of app conf dir to conf/@conf/@dst(.ext)*
88
+ - @conf/file - links a single file in a single subdir to conf/@conf/@dst-file
89
+ - @conf/ - links a directory in root of app conf dir to conf/@conf/@dst/
90
+ """
91
+
92
+ src: str
93
+
94
+ def __post_init__(self) -> None:
95
+ check_valid_deploy_spec_path(self.src)
96
+ if '/' in self.src:
97
+ check.equal(self.src.count('/'), 1)
98
+
99
+
100
+ class AppDeployConfLink(DeployConfLink):
101
+ pass
102
+
103
+
104
+ class TagDeployConfLink(DeployConfLink):
105
+ pass
106
+
107
+
108
+ #
109
+
110
+
111
+ @dc.dataclass(frozen=True)
112
+ class DeployConfSpec:
113
+ files: ta.Optional[ta.Sequence[DeployConfFile]] = None
114
+
115
+ links: ta.Optional[ta.Sequence[DeployConfLink]] = None
116
+
117
+ def __post_init__(self) -> None:
118
+ if self.files:
119
+ seen: ta.Set[str] = set()
120
+ for f in self.files:
121
+ check.not_in(f.path, seen)
122
+ seen.add(f.path)
57
123
 
58
124
 
59
125
  ##
@@ -62,12 +128,12 @@ class DeployVenvSpec:
62
128
  @dc.dataclass(frozen=True)
63
129
  class DeploySpec:
64
130
  app: DeployApp
65
- checkout: DeployGitCheckout
131
+
132
+ git: DeployGitSpec
66
133
 
67
134
  venv: ta.Optional[DeployVenvSpec] = None
68
135
 
69
- def __post_init__(self) -> None:
70
- hash(self)
136
+ conf: ta.Optional[DeployConfSpec] = None
71
137
 
72
138
  @cached_nullary
73
139
  def key(self) -> DeployKey:
@@ -5,46 +5,28 @@ TODO:
5
5
  - share more code with pyproject?
6
6
  """
7
7
  import os.path
8
- import typing as ta
9
8
 
10
9
  from omlish.asyncs.asyncio.subprocesses import asyncio_subprocesses
11
- from omlish.lite.cached import cached_nullary
12
- from omlish.lite.check import check
13
10
  from omlish.os.atomics import AtomicPathSwapping
14
11
 
15
- from .paths import DeployPath
16
- from .paths import DeployPathOwner
17
12
  from .specs import DeployVenvSpec
18
- from .types import DeployAppTag
19
- from .types import DeployHome
20
13
 
21
14
 
22
- class DeployVenvManager(DeployPathOwner):
15
+ class DeployVenvManager:
23
16
  def __init__(
24
17
  self,
25
18
  *,
26
- deploy_home: ta.Optional[DeployHome] = None,
27
19
  atomics: AtomicPathSwapping,
28
20
  ) -> None:
29
21
  super().__init__()
30
22
 
31
- self._deploy_home = deploy_home
32
23
  self._atomics = atomics
33
24
 
34
- @cached_nullary
35
- def _dir(self) -> str:
36
- return os.path.join(check.non_empty_str(self._deploy_home), 'venvs')
37
-
38
- def get_owned_deploy_paths(self) -> ta.AbstractSet[DeployPath]:
39
- return {
40
- DeployPath.parse('venvs/@app/@tag/'),
41
- }
42
-
43
25
  async def setup_venv(
44
26
  self,
45
- app_dir: str,
46
- venv_dir: str,
47
27
  spec: DeployVenvSpec,
28
+ git_dir: str,
29
+ venv_dir: str,
48
30
  ) -> None:
49
31
  sys_exe = 'python3'
50
32
 
@@ -58,7 +40,7 @@ class DeployVenvManager(DeployPathOwner):
58
40
 
59
41
  #
60
42
 
61
- reqs_txt = os.path.join(app_dir, 'requirements.txt')
43
+ reqs_txt = os.path.join(git_dir, 'requirements.txt')
62
44
 
63
45
  if os.path.isfile(reqs_txt):
64
46
  if spec.use_uv:
@@ -68,14 +50,3 @@ class DeployVenvManager(DeployPathOwner):
68
50
  pip_cmd = ['-m', 'pip']
69
51
 
70
52
  await asyncio_subprocesses.check_call(venv_exe, *pip_cmd,'install', '-r', reqs_txt)
71
-
72
- async def setup_app_venv(
73
- self,
74
- app_tag: DeployAppTag,
75
- spec: DeployVenvSpec,
76
- ) -> None:
77
- await self.setup_venv(
78
- os.path.join(check.non_empty_str(self._deploy_home), 'apps', app_tag.app, app_tag.tag),
79
- os.path.join(self._dir(), app_tag.app, app_tag.tag),
80
- spec,
81
- )
@@ -1527,6 +1527,15 @@ def snake_case(name: str) -> str:
1527
1527
  ##
1528
1528
 
1529
1529
 
1530
+ def strip_with_newline(s: str) -> str:
1531
+ if not s:
1532
+ return ''
1533
+ return s.strip() + '\n'
1534
+
1535
+
1536
+ ##
1537
+
1538
+
1530
1539
  def is_dunder(name: str) -> bool:
1531
1540
  return (
1532
1541
  name[:2] == name[-2:] == '__' and
ominfra/scripts/manage.py CHANGED
@@ -2732,6 +2732,15 @@ def snake_case(name: str) -> str:
2732
2732
  ##
2733
2733
 
2734
2734
 
2735
+ def strip_with_newline(s: str) -> str:
2736
+ if not s:
2737
+ return ''
2738
+ return s.strip() + '\n'
2739
+
2740
+
2741
+ ##
2742
+
2743
+
2735
2744
  def is_dunder(name: str) -> bool:
2736
2745
  return (
2737
2746
  name[:2] == name[-2:] == '__' and
@@ -3392,6 +3401,38 @@ class LinuxOsRelease:
3392
3401
  return dct
3393
3402
 
3394
3403
 
3404
+ ########################################
3405
+ # ../../../omlish/os/paths.py
3406
+
3407
+
3408
+ def abs_real_path(p: str) -> str:
3409
+ return os.path.abspath(os.path.realpath(p))
3410
+
3411
+
3412
+ def is_path_in_dir(base_dir: str, target_path: str) -> bool:
3413
+ base_dir = abs_real_path(base_dir)
3414
+ target_path = abs_real_path(target_path)
3415
+
3416
+ return target_path.startswith(base_dir + os.path.sep)
3417
+
3418
+
3419
+ def relative_symlink(
3420
+ src: str,
3421
+ dst: str,
3422
+ *,
3423
+ target_is_directory: bool = False,
3424
+ dir_fd: ta.Optional[int] = None,
3425
+ **kwargs: ta.Any,
3426
+ ) -> None:
3427
+ os.symlink(
3428
+ os.path.relpath(src, os.path.dirname(dst)),
3429
+ dst,
3430
+ target_is_directory=target_is_directory,
3431
+ dir_fd=dir_fd,
3432
+ **kwargs,
3433
+ )
3434
+
3435
+
3395
3436
  ########################################
3396
3437
  # ../../../omdev/packaging/specifiers.py
3397
3438
  # Copyright (c) Donald Stufft and individual contributors.
@@ -4068,11 +4109,12 @@ def build_command_name_map(crs: CommandRegistrations) -> CommandNameMap:
4068
4109
  # ../deploy/paths.py
4069
4110
  """
4070
4111
  TODO:
4071
- - run/pidfile
4112
+ - run/{.pid,.sock}
4072
4113
  - logs/...
4073
4114
  - current symlink
4074
4115
  - conf/{nginx,supervisor}
4075
4116
  - env/?
4117
+ - apps/<app>/shared
4076
4118
  """
4077
4119
 
4078
4120
 
@@ -4262,6 +4304,9 @@ class DeployPathOwner(abc.ABC):
4262
4304
  raise NotImplementedError
4263
4305
 
4264
4306
 
4307
+ DeployPathOwners = ta.NewType('DeployPathOwners', ta.Sequence[DeployPathOwner])
4308
+
4309
+
4265
4310
  class SingleDirDeployPathOwner(DeployPathOwner, abc.ABC):
4266
4311
  def __init__(
4267
4312
  self,
@@ -4300,6 +4345,17 @@ class SingleDirDeployPathOwner(DeployPathOwner, abc.ABC):
4300
4345
  ##
4301
4346
 
4302
4347
 
4348
+ def check_valid_deploy_spec_path(s: str) -> str:
4349
+ check.non_empty_str(s)
4350
+ for c in ['..', '//']:
4351
+ check.not_in(c, s)
4352
+ check.arg(not s.startswith('/'))
4353
+ return s
4354
+
4355
+
4356
+ ##
4357
+
4358
+
4303
4359
  @dc.dataclass(frozen=True)
4304
4360
  class DeployGitRepo:
4305
4361
  host: ta.Optional[str] = None
@@ -4312,7 +4368,7 @@ class DeployGitRepo:
4312
4368
 
4313
4369
 
4314
4370
  @dc.dataclass(frozen=True)
4315
- class DeployGitCheckout:
4371
+ class DeployGitSpec:
4316
4372
  repo: DeployGitRepo
4317
4373
  rev: DeployRev
4318
4374
 
@@ -4338,8 +4394,62 @@ class DeployVenvSpec:
4338
4394
 
4339
4395
  use_uv: bool = False
4340
4396
 
4397
+
4398
+ ##
4399
+
4400
+
4401
+ @dc.dataclass(frozen=True)
4402
+ class DeployConfFile:
4403
+ path: str
4404
+ body: str
4405
+
4341
4406
  def __post_init__(self) -> None:
4342
- hash(self)
4407
+ check_valid_deploy_spec_path(self.path)
4408
+
4409
+
4410
+ #
4411
+
4412
+
4413
+ @dc.dataclass(frozen=True)
4414
+ class DeployConfLink(abc.ABC): # noqa
4415
+ """
4416
+ May be either:
4417
+ - @conf(.ext)* - links a single file in root of app conf dir to conf/@conf/@dst(.ext)*
4418
+ - @conf/file - links a single file in a single subdir to conf/@conf/@dst-file
4419
+ - @conf/ - links a directory in root of app conf dir to conf/@conf/@dst/
4420
+ """
4421
+
4422
+ src: str
4423
+
4424
+ def __post_init__(self) -> None:
4425
+ check_valid_deploy_spec_path(self.src)
4426
+ if '/' in self.src:
4427
+ check.equal(self.src.count('/'), 1)
4428
+
4429
+
4430
+ class AppDeployConfLink(DeployConfLink):
4431
+ pass
4432
+
4433
+
4434
+ class TagDeployConfLink(DeployConfLink):
4435
+ pass
4436
+
4437
+
4438
+ #
4439
+
4440
+
4441
+ @dc.dataclass(frozen=True)
4442
+ class DeployConfSpec:
4443
+ files: ta.Optional[ta.Sequence[DeployConfFile]] = None
4444
+
4445
+ links: ta.Optional[ta.Sequence[DeployConfLink]] = None
4446
+
4447
+ def __post_init__(self) -> None:
4448
+ if self.files:
4449
+ seen: ta.Set[str] = set()
4450
+ for f in self.files:
4451
+ check.not_in(f.path, seen)
4452
+ seen.add(f.path)
4343
4453
 
4344
4454
 
4345
4455
  ##
@@ -4348,12 +4458,12 @@ class DeployVenvSpec:
4348
4458
  @dc.dataclass(frozen=True)
4349
4459
  class DeploySpec:
4350
4460
  app: DeployApp
4351
- checkout: DeployGitCheckout
4461
+
4462
+ git: DeployGitSpec
4352
4463
 
4353
4464
  venv: ta.Optional[DeployVenvSpec] = None
4354
4465
 
4355
- def __post_init__(self) -> None:
4356
- hash(self)
4466
+ conf: ta.Optional[DeployConfSpec] = None
4357
4467
 
4358
4468
  @cached_nullary
4359
4469
  def key(self) -> DeployKey:
@@ -6435,7 +6545,7 @@ class AtomicPathSwapping(abc.ABC):
6435
6545
  ##
6436
6546
 
6437
6547
 
6438
- class OsRenameAtomicPathSwap(AtomicPathSwap):
6548
+ class OsReplaceAtomicPathSwap(AtomicPathSwap):
6439
6549
  def __init__(
6440
6550
  self,
6441
6551
  kind: AtomicPathSwapKind,
@@ -6463,7 +6573,7 @@ class OsRenameAtomicPathSwap(AtomicPathSwap):
6463
6573
  return self._tmp_path
6464
6574
 
6465
6575
  def _commit(self) -> None:
6466
- os.rename(self._tmp_path, self._dst_path)
6576
+ os.replace(self._tmp_path, self._dst_path)
6467
6577
 
6468
6578
  def _abort(self) -> None:
6469
6579
  shutil.rmtree(self._tmp_path, ignore_errors=True)
@@ -6510,7 +6620,7 @@ class TempDirAtomicPathSwapping(AtomicPathSwapping):
6510
6620
  else:
6511
6621
  raise TypeError(kind)
6512
6622
 
6513
- return OsRenameAtomicPathSwap(
6623
+ return OsReplaceAtomicPathSwap(
6514
6624
  kind,
6515
6625
  dst_path,
6516
6626
  tmp_path,
@@ -6748,6 +6858,137 @@ class PingCommandExecutor(CommandExecutor[PingCommand, PingCommand.Output]):
6748
6858
  CommandExecutorMap = ta.NewType('CommandExecutorMap', ta.Mapping[ta.Type[Command], CommandExecutor])
6749
6859
 
6750
6860
 
6861
+ ########################################
6862
+ # ../deploy/conf.py
6863
+ """
6864
+ TODO:
6865
+ - @conf DeployPathPlaceholder? :|
6866
+ - post-deploy: remove any dir_links not present in new spec
6867
+ - * only if succeeded * - otherwise, remove any dir_links present in new spec but not previously present?
6868
+ - no such thing as 'previously present'.. build a 'deploy state' and pass it back?
6869
+ - ** whole thing can be atomic **
6870
+ - 1) new atomic temp dir
6871
+ - 2) for each subdir not needing modification, hardlink into temp dir
6872
+ - 3) for each subdir needing modification, new subdir, hardlink all files not needing modification
6873
+ - 4) write (or if deleting, omit) new files
6874
+ - 5) swap top level
6875
+ - ** whole deploy can be atomic(-ish) - do this for everything **
6876
+ - just a '/deploy/current' dir
6877
+ - some things (venvs) cannot be moved, thus the /deploy/venvs dir
6878
+ - ** ensure (enforce) equivalent relpath nesting
6879
+ """
6880
+
6881
+
6882
+ class DeployConfManager(SingleDirDeployPathOwner):
6883
+ def __init__(
6884
+ self,
6885
+ *,
6886
+ deploy_home: ta.Optional[DeployHome] = None,
6887
+ ) -> None:
6888
+ super().__init__(
6889
+ owned_dir='conf',
6890
+ deploy_home=deploy_home,
6891
+ )
6892
+
6893
+ async def _write_conf_file(
6894
+ self,
6895
+ cf: DeployConfFile,
6896
+ conf_dir: str,
6897
+ ) -> None:
6898
+ conf_file = os.path.join(conf_dir, cf.path)
6899
+ check.arg(is_path_in_dir(conf_dir, conf_file))
6900
+
6901
+ os.makedirs(os.path.dirname(conf_file), exist_ok=True)
6902
+
6903
+ with open(conf_file, 'w') as f: # noqa
6904
+ f.write(cf.body)
6905
+
6906
+ async def _make_conf_link(
6907
+ self,
6908
+ link: DeployConfLink,
6909
+ conf_dir: str,
6910
+ app_tag: DeployAppTag,
6911
+ link_dir: str,
6912
+ ) -> None:
6913
+ link_src = os.path.join(conf_dir, link.src)
6914
+ check.arg(is_path_in_dir(conf_dir, link_src))
6915
+
6916
+ is_link_dir = link.src.endswith('/')
6917
+ if is_link_dir:
6918
+ check.arg(link.src.count('/') == 1)
6919
+ check.arg(os.path.isdir(link_src))
6920
+ link_dst_pfx = link.src
6921
+ link_dst_sfx = ''
6922
+
6923
+ elif '/' in link.src:
6924
+ check.arg(os.path.isfile(link_src))
6925
+ d, f = os.path.split(link.src)
6926
+ # TODO: check filename :|
6927
+ link_dst_pfx = d + '/'
6928
+ link_dst_sfx = '-' + f
6929
+
6930
+ else:
6931
+ check.arg(os.path.isfile(link_src))
6932
+ if '.' in link.src:
6933
+ l, _, r = link.src.partition('.')
6934
+ link_dst_pfx = l + '/'
6935
+ link_dst_sfx = '.' + r
6936
+ else:
6937
+ link_dst_pfx = link.src + '/'
6938
+ link_dst_sfx = ''
6939
+
6940
+ if isinstance(link, AppDeployConfLink):
6941
+ link_dst_mid = str(app_tag.app)
6942
+ sym_root = link_dir
6943
+ elif isinstance(link, TagDeployConfLink):
6944
+ link_dst_mid = '-'.join([app_tag.app, app_tag.tag])
6945
+ sym_root = conf_dir
6946
+ else:
6947
+ raise TypeError(link)
6948
+
6949
+ link_dst = ''.join([
6950
+ link_dst_pfx,
6951
+ link_dst_mid,
6952
+ link_dst_sfx,
6953
+ ])
6954
+
6955
+ root_conf_dir = self._make_dir()
6956
+ sym_src = os.path.join(sym_root, link.src)
6957
+ sym_dst = os.path.join(root_conf_dir, link_dst)
6958
+ check.arg(is_path_in_dir(root_conf_dir, sym_dst))
6959
+
6960
+ os.makedirs(os.path.dirname(sym_dst), exist_ok=True)
6961
+ relative_symlink(sym_src, sym_dst, target_is_directory=is_link_dir)
6962
+
6963
+ async def write_conf(
6964
+ self,
6965
+ spec: DeployConfSpec,
6966
+ conf_dir: str,
6967
+ app_tag: DeployAppTag,
6968
+ link_dir: str,
6969
+ ) -> None:
6970
+ conf_dir = os.path.abspath(conf_dir)
6971
+ os.makedirs(conf_dir)
6972
+
6973
+ #
6974
+
6975
+ for cf in spec.files or []:
6976
+ await self._write_conf_file(
6977
+ cf,
6978
+ conf_dir,
6979
+ )
6980
+
6981
+ #
6982
+
6983
+ for link in spec.links or []:
6984
+ await self._make_conf_link(
6985
+ link,
6986
+ conf_dir,
6987
+ app_tag,
6988
+ link_dir,
6989
+ )
6990
+
6991
+
6751
6992
  ########################################
6752
6993
  # ../deploy/tmp.py
6753
6994
 
@@ -8240,7 +8481,7 @@ class DeployGitManager(SingleDirDeployPathOwner):
8240
8481
 
8241
8482
  #
8242
8483
 
8243
- async def checkout(self, checkout: DeployGitCheckout, dst_dir: str) -> None:
8484
+ async def checkout(self, spec: DeployGitSpec, dst_dir: str) -> None:
8244
8485
  check.state(not os.path.exists(dst_dir))
8245
8486
  with self._git._atomics.begin_atomic_path_swap( # noqa
8246
8487
  'dir',
@@ -8248,14 +8489,14 @@ class DeployGitManager(SingleDirDeployPathOwner):
8248
8489
  auto_commit=True,
8249
8490
  make_dirs=True,
8250
8491
  ) as dst_swap:
8251
- await self.fetch(checkout.rev)
8492
+ await self.fetch(spec.rev)
8252
8493
 
8253
8494
  dst_call = functools.partial(asyncio_subprocesses.check_call, cwd=dst_swap.tmp_path)
8254
8495
  await dst_call('git', 'init')
8255
8496
 
8256
8497
  await dst_call('git', 'remote', 'add', 'local', self._dir)
8257
- await dst_call('git', 'fetch', '--depth=1', 'local', checkout.rev)
8258
- await dst_call('git', 'checkout', checkout.rev, *(checkout.subtrees or []))
8498
+ await dst_call('git', 'fetch', '--depth=1', 'local', spec.rev)
8499
+ await dst_call('git', 'checkout', spec.rev, *(spec.subtrees or []))
8259
8500
 
8260
8501
  def get_repo_dir(self, repo: DeployGitRepo) -> RepoDir:
8261
8502
  try:
@@ -8264,8 +8505,12 @@ class DeployGitManager(SingleDirDeployPathOwner):
8264
8505
  repo_dir = self._repo_dirs[repo] = DeployGitManager.RepoDir(self, repo)
8265
8506
  return repo_dir
8266
8507
 
8267
- async def checkout(self, checkout: DeployGitCheckout, dst_dir: str) -> None:
8268
- await self.get_repo_dir(checkout.repo).checkout(checkout, dst_dir)
8508
+ async def checkout(
8509
+ self,
8510
+ spec: DeployGitSpec,
8511
+ dst_dir: str,
8512
+ ) -> None:
8513
+ await self.get_repo_dir(spec.repo).checkout(spec, dst_dir)
8269
8514
 
8270
8515
 
8271
8516
  ########################################
@@ -8277,32 +8522,21 @@ TODO:
8277
8522
  """
8278
8523
 
8279
8524
 
8280
- class DeployVenvManager(DeployPathOwner):
8525
+ class DeployVenvManager:
8281
8526
  def __init__(
8282
8527
  self,
8283
8528
  *,
8284
- deploy_home: ta.Optional[DeployHome] = None,
8285
8529
  atomics: AtomicPathSwapping,
8286
8530
  ) -> None:
8287
8531
  super().__init__()
8288
8532
 
8289
- self._deploy_home = deploy_home
8290
8533
  self._atomics = atomics
8291
8534
 
8292
- @cached_nullary
8293
- def _dir(self) -> str:
8294
- return os.path.join(check.non_empty_str(self._deploy_home), 'venvs')
8295
-
8296
- def get_owned_deploy_paths(self) -> ta.AbstractSet[DeployPath]:
8297
- return {
8298
- DeployPath.parse('venvs/@app/@tag/'),
8299
- }
8300
-
8301
8535
  async def setup_venv(
8302
8536
  self,
8303
- app_dir: str,
8304
- venv_dir: str,
8305
8537
  spec: DeployVenvSpec,
8538
+ git_dir: str,
8539
+ venv_dir: str,
8306
8540
  ) -> None:
8307
8541
  sys_exe = 'python3'
8308
8542
 
@@ -8316,7 +8550,7 @@ class DeployVenvManager(DeployPathOwner):
8316
8550
 
8317
8551
  #
8318
8552
 
8319
- reqs_txt = os.path.join(app_dir, 'requirements.txt')
8553
+ reqs_txt = os.path.join(git_dir, 'requirements.txt')
8320
8554
 
8321
8555
  if os.path.isfile(reqs_txt):
8322
8556
  if spec.use_uv:
@@ -8327,17 +8561,6 @@ class DeployVenvManager(DeployPathOwner):
8327
8561
 
8328
8562
  await asyncio_subprocesses.check_call(venv_exe, *pip_cmd,'install', '-r', reqs_txt)
8329
8563
 
8330
- async def setup_app_venv(
8331
- self,
8332
- app_tag: DeployAppTag,
8333
- spec: DeployVenvSpec,
8334
- ) -> None:
8335
- await self.setup_venv(
8336
- os.path.join(check.non_empty_str(self._deploy_home), 'apps', app_tag.app, app_tag.tag),
8337
- os.path.join(self._dir(), app_tag.app, app_tag.tag),
8338
- spec,
8339
- )
8340
-
8341
8564
 
8342
8565
  ########################################
8343
8566
  # ../remote/_main.py
@@ -8886,12 +9109,16 @@ class DeployAppManager(DeployPathOwner):
8886
9109
  self,
8887
9110
  *,
8888
9111
  deploy_home: ta.Optional[DeployHome] = None,
9112
+
9113
+ conf: DeployConfManager,
8889
9114
  git: DeployGitManager,
8890
9115
  venvs: DeployVenvManager,
8891
9116
  ) -> None:
8892
9117
  super().__init__()
8893
9118
 
8894
9119
  self._deploy_home = deploy_home
9120
+
9121
+ self._conf = conf
8895
9122
  self._git = git
8896
9123
  self._venvs = venvs
8897
9124
 
@@ -8901,27 +9128,72 @@ class DeployAppManager(DeployPathOwner):
8901
9128
 
8902
9129
  def get_owned_deploy_paths(self) -> ta.AbstractSet[DeployPath]:
8903
9130
  return {
8904
- DeployPath.parse('apps/@app/@tag/'),
9131
+ DeployPath.parse('apps/@app/current'),
9132
+ DeployPath.parse('apps/@app/deploying'),
9133
+
9134
+ DeployPath.parse('apps/@app/tags/@tag/conf/'),
9135
+ DeployPath.parse('apps/@app/tags/@tag/git/'),
9136
+ DeployPath.parse('apps/@app/tags/@tag/venv/'),
8905
9137
  }
8906
9138
 
8907
9139
  async def prepare_app(
8908
9140
  self,
8909
9141
  spec: DeploySpec,
8910
9142
  ) -> None:
8911
- app_tag = DeployAppTag(spec.app, make_deploy_tag(spec.checkout.rev, spec.key()))
8912
- app_dir = os.path.join(self._dir(), spec.app, app_tag.tag)
9143
+ app_tag = DeployAppTag(spec.app, make_deploy_tag(spec.git.rev, spec.key()))
9144
+
9145
+ #
9146
+
9147
+ app_dir = os.path.join(self._dir(), spec.app)
9148
+ os.makedirs(app_dir, exist_ok=True)
8913
9149
 
8914
9150
  #
8915
9151
 
9152
+ tag_dir = os.path.join(app_dir, 'tags', app_tag.tag)
9153
+ os.makedirs(tag_dir)
9154
+
9155
+ #
9156
+
9157
+ deploying_file = os.path.join(app_dir, 'deploying')
9158
+ current_file = os.path.join(app_dir, 'current')
9159
+
9160
+ if os.path.exists(deploying_file):
9161
+ os.unlink(deploying_file)
9162
+ relative_symlink(tag_dir, deploying_file, target_is_directory=True)
9163
+
9164
+ #
9165
+
9166
+ git_dir = os.path.join(tag_dir, 'git')
8916
9167
  await self._git.checkout(
8917
- spec.checkout,
8918
- app_dir,
9168
+ spec.git,
9169
+ git_dir,
8919
9170
  )
8920
9171
 
8921
9172
  #
8922
9173
 
8923
9174
  if spec.venv is not None:
8924
- await self._venvs.setup_app_venv(app_tag, spec.venv)
9175
+ venv_dir = os.path.join(tag_dir, 'venv')
9176
+ await self._venvs.setup_venv(
9177
+ spec.venv,
9178
+ git_dir,
9179
+ venv_dir,
9180
+ )
9181
+
9182
+ #
9183
+
9184
+ if spec.conf is not None:
9185
+ conf_dir = os.path.join(tag_dir, 'conf')
9186
+ conf_link_dir = os.path.join(current_file, 'conf')
9187
+ await self._conf.write_conf(
9188
+ spec.conf,
9189
+ conf_dir,
9190
+ app_tag,
9191
+ conf_link_dir,
9192
+ )
9193
+
9194
+ #
9195
+
9196
+ os.replace(deploying_file, current_file)
8925
9197
 
8926
9198
 
8927
9199
  ########################################
@@ -10043,23 +10315,45 @@ def bind_deploy(
10043
10315
  ) -> InjectorBindings:
10044
10316
  lst: ta.List[InjectorBindingOrBindings] = [
10045
10317
  inj.bind(deploy_config),
10318
+ ]
10046
10319
 
10047
- #
10320
+ #
10048
10321
 
10049
- inj.bind(DeployAppManager, singleton=True),
10322
+ def bind_manager(cls: type) -> InjectorBindings:
10323
+ return inj.as_bindings(
10324
+ inj.bind(cls, singleton=True),
10050
10325
 
10051
- inj.bind(DeployGitManager, singleton=True),
10326
+ *([inj.bind(DeployPathOwner, to_key=cls, array=True)] if issubclass(cls, DeployPathOwner) else []),
10327
+ )
10328
+
10329
+ lst.extend([
10330
+ inj.bind_array(DeployPathOwner),
10331
+ inj.bind_array_type(DeployPathOwner, DeployPathOwners),
10332
+ ])
10333
+
10334
+ #
10335
+
10336
+ lst.extend([
10337
+ bind_manager(DeployAppManager),
10052
10338
 
10053
- inj.bind(DeployTmpManager, singleton=True),
10339
+ bind_manager(DeployConfManager),
10340
+
10341
+ bind_manager(DeployGitManager),
10342
+
10343
+ bind_manager(DeployTmpManager),
10054
10344
  inj.bind(AtomicPathSwapping, to_key=DeployTmpManager),
10055
10345
 
10056
- inj.bind(DeployVenvManager, singleton=True),
10346
+ bind_manager(DeployVenvManager),
10347
+ ])
10057
10348
 
10058
- #
10349
+ #
10059
10350
 
10351
+ lst.extend([
10060
10352
  bind_command(DeployCommand, DeployCommandExecutor),
10061
10353
  bind_command(InterpCommand, InterpCommandExecutor),
10062
- ]
10354
+ ])
10355
+
10356
+ #
10063
10357
 
10064
10358
  if (dh := deploy_config.deploy_home) is not None:
10065
10359
  dh = os.path.abspath(os.path.expanduser(dh))
@@ -2406,6 +2406,15 @@ def snake_case(name: str) -> str:
2406
2406
  ##
2407
2407
 
2408
2408
 
2409
+ def strip_with_newline(s: str) -> str:
2410
+ if not s:
2411
+ return ''
2412
+ return s.strip() + '\n'
2413
+
2414
+
2415
+ ##
2416
+
2417
+
2409
2418
  def is_dunder(name: str) -> bool:
2410
2419
  return (
2411
2420
  name[:2] == name[-2:] == '__' and
@@ -6110,6 +6119,7 @@ class ServerConfig:
6110
6119
 
6111
6120
  groups: ta.Optional[ta.Sequence[ProcessGroupConfig]] = None
6112
6121
 
6122
+ # TODO: implement - make sure to accept broken symlinks
6113
6123
  group_config_dirs: ta.Optional[ta.Sequence[str]] = None
6114
6124
 
6115
6125
  @classmethod
@@ -245,6 +245,7 @@ class ServerConfig:
245
245
 
246
246
  groups: ta.Optional[ta.Sequence[ProcessGroupConfig]] = None
247
247
 
248
+ # TODO: implement - make sure to accept broken symlinks
248
249
  group_config_dirs: ta.Optional[ta.Sequence[str]] = None
249
250
 
250
251
  @classmethod
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: ominfra
3
- Version: 0.0.0.dev166
3
+ Version: 0.0.0.dev167
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.dev166
16
- Requires-Dist: omlish==0.0.0.dev166
15
+ Requires-Dist: omdev==0.0.0.dev167
16
+ Requires-Dist: omlish==0.0.0.dev167
17
17
  Provides-Extra: all
18
18
  Requires-Dist: paramiko~=3.5; extra == "all"
19
19
  Requires-Dist: asyncssh~=2.18; extra == "all"
@@ -44,17 +44,18 @@ ominfra/manage/commands/ping.py,sha256=DVZFzL1Z_f-Bq53vxMrL3xOi0iK_nMonJE4KvQf9w
44
44
  ominfra/manage/commands/subprocess.py,sha256=yHGMbAI-xKe_9BUs5IZ3Yav8qRE-I9aGnBtTwW15Pnw,2440
45
45
  ominfra/manage/commands/types.py,sha256=XFZPeqeIBAaIIQF3pdPbGxLlb-LCrz6WtlDWO2q_vz0,210
46
46
  ominfra/manage/deploy/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
47
- ominfra/manage/deploy/apps.py,sha256=Nv3eXVPxEbSz4wyTI8chJ-C-EEC721vFssvvg0YDRXo,1937
47
+ ominfra/manage/deploy/apps.py,sha256=jLvam6sVcWfRmyyLeJyTxz0IErWTn98rf1-8lIEEYlM,3292
48
48
  ominfra/manage/deploy/commands.py,sha256=N9qVntnRgJ_IneI7rEQB2Za0oU7gouPfm-sl2MCwW1E,764
49
+ ominfra/manage/deploy/conf.py,sha256=HxKLWIT95StavJ0uHhMhuvWio1gTYBlDJv6N1_XdxEo,4417
49
50
  ominfra/manage/deploy/config.py,sha256=aR6ubMEWqkTI55XtcG1Cczn6YhCVN6eSL8DT5EHQJN0,166
50
- ominfra/manage/deploy/git.py,sha256=6CGLvGH8uYkuT8gyZHybJb7sgUPTtFgy7grj1YHkI9g,3747
51
- ominfra/manage/deploy/inject.py,sha256=8wuIgdzkDCHbc69nD1meLjwfCOMdWOIfDT5yijL-du8,1441
51
+ ominfra/manage/deploy/git.py,sha256=BGht0UmY1d76kSGHESWuG8qxNKLuOGrwey5V9M4pfCY,3746
52
+ ominfra/manage/deploy/inject.py,sha256=o-bgWvziUuE5XzsE9_rcPSdWqPO9AAf2SbXamEpir2k,1978
52
53
  ominfra/manage/deploy/interp.py,sha256=OKkenH8YKEW_mEDR6X7_ZLxK9a1Ox6KHSwFPTHT6OzA,1029
53
- ominfra/manage/deploy/paths.py,sha256=tK8zZFWOHDRdTN5AlTe-3MpgZqovhWrljGosQmeEYvo,6839
54
- ominfra/manage/deploy/specs.py,sha256=qQSMA2ns0dalbizYwOp14i154mCOkjyvvh4_cN0gT3k,1574
54
+ ominfra/manage/deploy/paths.py,sha256=Rq9OsK3MpporH6tfuumzLO51e4Yz_wi6RiXEg82iyLY,6947
55
+ ominfra/manage/deploy/specs.py,sha256=Xv1OQ3yQxOTR39Gj5ACgAn7vpgL-zVuGk1CHk_xmt5o,2883
55
56
  ominfra/manage/deploy/tmp.py,sha256=L0pIfQuxQ7_6gC_AAv7eubI37_IPzCVR29hkn1MHL2Q,1230
56
57
  ominfra/manage/deploy/types.py,sha256=o95wqvTGNRq8Cxx7VpqeX-9x1tI8k8BpqPFvJZkJYBA,305
57
- ominfra/manage/deploy/venvs.py,sha256=u8ds1TVyipFVSfm7sgEHGPw4JPG_hXY2j1LnTCLjxqw,2337
58
+ ominfra/manage/deploy/venvs.py,sha256=1co8l3eSxl2WccrF-XOTqeltoMccRH63o69NblV3yok,1366
58
59
  ominfra/manage/remote/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
59
60
  ominfra/manage/remote/_main.py,sha256=p5KoiS2WMw6QAqlDl_Zun-JybmCsy8awIfpBMLBjGMY,4356
60
61
  ominfra/manage/remote/channel.py,sha256=36xR9Ti9ZA8TUBtxmY0u7_3Lv7E6wzQTxlZl7gLR5GE,2224
@@ -75,13 +76,13 @@ ominfra/manage/targets/connection.py,sha256=5e8h9Miej2DKJxZfLyxpGe8y-Y0V_b_AuUW1
75
76
  ominfra/manage/targets/inject.py,sha256=P4597xWM-V3I_gCt2O71OLhYQkkXtuJvkYRsIbhhMcE,1561
76
77
  ominfra/manage/targets/targets.py,sha256=CFl8Uirgn3gfowO1Fn-LBK-6qYqEMFJ9snPUl0gCRuM,1753
77
78
  ominfra/scripts/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
78
- ominfra/scripts/journald2aws.py,sha256=EC8tSKW3hztBV_Kr_ykK72AmcvnWivUxcz6Sfg3M_hI,155085
79
- ominfra/scripts/manage.py,sha256=M8k4XmOrq2R3rjUNU3SAHeqdjvWZptpXvFAl4A0RmGM,294060
80
- ominfra/scripts/supervisor.py,sha256=uPcw4o8gt8xvQ97jXK-WBAaZj3D81Lq9tqDoKxGvLCU,273791
79
+ ominfra/scripts/journald2aws.py,sha256=Cfxnq9N2PaRLBrGSPq4qi39SOGldKZXT_Ve2_uPWcpY,155191
80
+ ominfra/scripts/manage.py,sha256=Yb_Ok9rB8zMycxED0fjzHQVNmVQ2qc14pASuzpves68,301283
81
+ ominfra/scripts/supervisor.py,sha256=y9o5NyKKb7ugwk3ZEywzlqVFd7QDhsSO2_C8oSqO_28,273957
81
82
  ominfra/supervisor/LICENSE.txt,sha256=yvqaMNsDhWxziHa9ien6qCW1SkZv-DQlAg96XjfSee8,1746
82
83
  ominfra/supervisor/__init__.py,sha256=Y3l4WY4JRi2uLG6kgbGp93fuGfkxkKwZDvhsa0Rwgtk,15
83
84
  ominfra/supervisor/__main__.py,sha256=I0yFw-C08OOiZ3BF6lF1Oiv789EQXu-_j6whDhQUTEA,66
84
- ominfra/supervisor/configs.py,sha256=InaLW0T93dbAuzKyc6H8wGIFR2oM1GROSiQX9hZY_ko,13711
85
+ ominfra/supervisor/configs.py,sha256=UxcZ1yaw7CiwiSb7A1AegwcWKFXqtswaKaHlIN8Bt_Q,13771
85
86
  ominfra/supervisor/dispatchers.py,sha256=zXLwQS4Vc6dWw5o9QOL04UMDt7w6CKu9wf19CjUiS2Q,1005
86
87
  ominfra/supervisor/dispatchersimpl.py,sha256=q3dEyOHWTPKm28nmAGisjgIW1BX6O3-SzbYa7nWuTEs,11349
87
88
  ominfra/supervisor/events.py,sha256=XGrtzHr1xm0dwjz329fn9eR0_Ap-LQL6Sk8LJ8eVDEo,6692
@@ -119,9 +120,9 @@ ominfra/tailscale/api.py,sha256=C5-t_b6jZXUWcy5k8bXm7CFnk73pSdrlMOgGDeGVrpw,1370
119
120
  ominfra/tailscale/cli.py,sha256=h6akQJMl0KuWLHS7Ur6WcBZ2JwF0DJQhsPTnFBdGyNk,3571
120
121
  ominfra/tools/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
121
122
  ominfra/tools/listresources.py,sha256=4qVg5txsb10EHhvqXXeM6gJ2jx9LbroEnPydDv1uXs0,6176
122
- ominfra-0.0.0.dev166.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
123
- ominfra-0.0.0.dev166.dist-info/METADATA,sha256=K1Yk7mm_LiNl0FtGC69NQqG23xD68-dpAwbNiBgfn90,731
124
- ominfra-0.0.0.dev166.dist-info/WHEEL,sha256=PZUExdf71Ui_so67QXpySuHtCi3-J3wvF4ORK6k_S8U,91
125
- ominfra-0.0.0.dev166.dist-info/entry_points.txt,sha256=kgecQ2MgGrM9qK744BoKS3tMesaC3yjLnl9pa5CRczg,37
126
- ominfra-0.0.0.dev166.dist-info/top_level.txt,sha256=E-b2OHkk_AOBLXHYZQ2EOFKl-_6uOGd8EjeG-Zy6h_w,8
127
- ominfra-0.0.0.dev166.dist-info/RECORD,,
123
+ ominfra-0.0.0.dev167.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
124
+ ominfra-0.0.0.dev167.dist-info/METADATA,sha256=JFB_UaK98rzNZ6vQWBEVI9f7zjJ4DeSf00RpqMQrczs,731
125
+ ominfra-0.0.0.dev167.dist-info/WHEEL,sha256=PZUExdf71Ui_so67QXpySuHtCi3-J3wvF4ORK6k_S8U,91
126
+ ominfra-0.0.0.dev167.dist-info/entry_points.txt,sha256=kgecQ2MgGrM9qK744BoKS3tMesaC3yjLnl9pa5CRczg,37
127
+ ominfra-0.0.0.dev167.dist-info/top_level.txt,sha256=E-b2OHkk_AOBLXHYZQ2EOFKl-_6uOGd8EjeG-Zy6h_w,8
128
+ ominfra-0.0.0.dev167.dist-info/RECORD,,