1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
|
import contextlib
import json
import pathlib
import shlex
from typing import Any, Dict, Union, List, IO, TextIO, Optional, cast
from .container_engines import Podman
from .container_types import CephContainer, InitContainer
from .context import CephadmContext
from .context_getters import fetch_meta
from .daemon_identity import DaemonIdentity
from .file_utils import write_new
from .net_utils import EndPoint
# Ideally, all ContainerCommands would be converted to init containers. Until
# that is done one can wrap a CephContainer in a ContainerCommand object and
# pass that as a pre- or post- command to run arbitrary container based
# commands in the script.
class ContainerCommand:
def __init__(
self,
container: CephContainer,
comment: str = '',
background: bool = False,
):
self.container = container
self.comment = comment
self.background = background
Command = Union[List[str], str, ContainerCommand]
def write_service_scripts(
ctx: CephadmContext,
ident: DaemonIdentity,
*,
container: CephContainer,
init_containers: Optional[List[InitContainer]] = None,
endpoints: Optional[List[EndPoint]] = None,
pre_start_commands: Optional[List[Command]] = None,
post_stop_commands: Optional[List[Command]] = None,
timeout: Optional[int] = None,
) -> None:
"""Write the scripts that systemd services will call in order to
start/stop/etc components of a cephadm managed daemon. Also writes some
metadata about the service getting deployed.
"""
data_dir = pathlib.Path(ident.data_dir(ctx.data_dir))
run_file_path = data_dir / 'unit.run'
meta_file_path = data_dir / 'unit.meta'
post_stop_file_path = data_dir / 'unit.poststop'
stop_file_path = data_dir / 'unit.stop'
image_file_path = data_dir / 'unit.image'
# use an ExitStack to make writing the files an all-or-nothing affair. If
# any file fails to write then the write_new'd file will not get renamed
# into place
with contextlib.ExitStack() as estack:
# write out the main file to run (start) a service
runf = estack.enter_context(write_new(run_file_path))
runf.write('set -e\n')
for command in pre_start_commands or []:
_write_command(ctx, runf, command)
init_containers = init_containers or []
if init_containers:
_write_init_container_cmds_clean(ctx, runf, init_containers[0])
for idx, ic in enumerate(init_containers):
_write_init_container_cmds(ctx, runf, idx, ic)
_write_container_cmd_to_bash(ctx, runf, container, ident.daemon_name)
# some metadata about the deploy
metaf = estack.enter_context(write_new(meta_file_path))
meta: Dict[str, Any] = fetch_meta(ctx)
meta.update(
{
'memory_request': int(ctx.memory_request)
if ctx.memory_request
else None,
'memory_limit': int(ctx.memory_limit)
if ctx.memory_limit
else None,
}
)
if not meta.get('ports'):
if endpoints:
meta['ports'] = [e.port for e in endpoints]
else:
meta['ports'] = []
metaf.write(json.dumps(meta, indent=4) + '\n')
# post-stop command(s)
pstopf = estack.enter_context(write_new(post_stop_file_path))
# this is a fallback to eventually stop any underlying container that
# was not stopped properly by unit.stop, this could happen in very slow
# setups as described in the issue
# https://tracker.ceph.com/issues/58242.
_write_stop_actions(ctx, cast(TextIO, pstopf), container, timeout)
for command in post_stop_commands or []:
_write_command(ctx, pstopf, command)
# stop command(s)
stopf = estack.enter_context(write_new(stop_file_path))
_write_stop_actions(ctx, cast(TextIO, stopf), container, timeout)
if container:
imgf = estack.enter_context(write_new(image_file_path))
imgf.write(container.image + '\n')
def _write_container_cmd_to_bash(
ctx: CephadmContext,
file_obj: IO[str],
container: 'CephContainer',
comment: Optional[str] = None,
background: Optional[bool] = False,
) -> None:
if comment:
# Sometimes adding a comment, especially if there are multiple containers in one
# unit file, makes it easier to read and grok.
assert '\n' not in comment
file_obj.write(f'# {comment}\n')
# Sometimes, adding `--rm` to a run_cmd doesn't work. Let's remove the container manually
_bash_cmd(
file_obj, container.rm_cmd(old_cname=True), check=False, stderr=False
)
_bash_cmd(file_obj, container.rm_cmd(), check=False, stderr=False)
# Sometimes, `podman rm` doesn't find the container. Then you'll have to add `--storage`
if isinstance(ctx.container_engine, Podman):
_bash_cmd(
file_obj,
container.rm_cmd(storage=True),
check=False,
stderr=False,
)
_bash_cmd(
file_obj,
container.rm_cmd(old_cname=True, storage=True),
check=False,
stderr=False,
)
# container run command
_bash_cmd(file_obj, container.run_cmd(), background=bool(background))
def _write_init_container_cmds(
ctx: CephadmContext,
file_obj: IO[str],
index: int,
init_container: 'InitContainer',
) -> None:
file_obj.write(f'# init container {index}: {init_container.cname}\n')
_bash_cmd(file_obj, init_container.run_cmd())
_write_init_container_cmds_clean(
ctx, file_obj, init_container, comment=''
)
def _write_init_container_cmds_clean(
ctx: CephadmContext,
file_obj: IO[str],
init_container: 'InitContainer',
comment: str = 'init container cleanup',
) -> None:
if comment:
assert '\n' not in comment
file_obj.write(f'# {comment}\n')
_bash_cmd(
file_obj,
init_container.rm_cmd(),
check=False,
stderr=False,
)
# Sometimes, `podman rm` doesn't find the container. Then you'll have to add `--storage`
if isinstance(ctx.container_engine, Podman):
_bash_cmd(
file_obj,
init_container.rm_cmd(storage=True),
check=False,
stderr=False,
)
def _write_stop_actions(
ctx: CephadmContext,
f: TextIO,
container: 'CephContainer',
timeout: Optional[int],
) -> None:
# following generated script basically checks if the container exists
# before stopping it. Exit code will be success either if it doesn't
# exist or if it exists and is stopped successfully.
container_exists = f'{ctx.container_engine.path} inspect %s &>/dev/null'
f.write(
f'! {container_exists % container.old_cname} || {" ".join(container.stop_cmd(old_cname=True, timeout=timeout))} \n'
)
f.write(
f'! {container_exists % container.cname} || {" ".join(container.stop_cmd(timeout=timeout))} \n'
)
def _bash_cmd(
fh: IO[str],
cmd: List[str],
check: bool = True,
background: bool = False,
stderr: bool = True,
) -> None:
line = ' '.join(shlex.quote(arg) for arg in cmd)
if not check:
line = f'! {line}'
if not stderr:
line = f'{line} 2> /dev/null'
if background:
line = f'{line} &'
fh.write(line)
fh.write('\n')
def _write_command(
ctx: CephadmContext,
fh: IO[str],
cmd: Command,
) -> None:
"""Wrapper func for turning a command list or string into something suitable
for appending to a run script.
"""
if isinstance(cmd, list):
_bash_cmd(fh, cmd)
elif isinstance(cmd, ContainerCommand):
_write_container_cmd_to_bash(
ctx,
fh,
cmd.container,
comment=cmd.comment,
background=cmd.background,
)
else:
fh.write(cmd)
if not cmd.endswith('\n'):
fh.write('\n')
|