File: HINTS.md

package info (click to toggle)
docker-systemctl-replacement 1.4.4181-1.1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, bullseye, forky, sid, trixie
  • size: 1,628 kB
  • sloc: python: 19,142; makefile: 208
file content (191 lines) | stat: -rw-r--r-- 8,434 bytes parent folder | download
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
# Tricks and Hints

## The --now internals

The option "--now" is only used for a very low number
of systemctl commands. In the systemctl.py variant
it is (ab)used to run an alternative method.

## systemctl.py list-unit-files --now

Shows the list of files found on the disk. Those are
not parsed and they are not check for an enabled-link.
It is just what is there as a first step after scanning
for systemd-relevant unit information.

## systemctl.py list-dependencies --now

Shows the start-order of services. Use -v debugging to 
see how the start-rank is computed from the Before/After 
clauses in the unit files.

Note that upon a "<stop> operation" the start order is
run through in reverse order.

The current 'default' target does not filter the list.
Instead it only uses Before/After on the units in the
'multi-user.target' but it does not process any of
its required units.

# Implementation helpers

## docker images --filter

https://github.com/moby/moby/blob/10c0af083544460a2ddc2218f37dc24a077f7d90/docs/reference/commandline/images.md#filtering

    docker images --filter dangling=true || dangling=false
    docker images --filter label=<key> || label=<key>=<value>

whereas filter by name is just an argument "docker images <name>"

## docker images --format

    docker images --format "{{.ID}}: {{.Repository}}\t{{.Size}}"
    and "{{.Tag}} {{.Digest}} {{.CreatedSince}} {{.CreatedAt}}"

## IPv4 -vs- IPv6

For java / tomcat applications one can add a java-option to use ipv4

    -Djava.net.preferIPv4Stack=true

For docker-compose 2.1+ one can add a sysctl to disable ipv6.

    sysctls:
       - net.ipv6.conf.eth0.disable_ipv6=1
       - net.ipv6.conf.all.disable_ipv6=1

And there is an option to disable '::1 localhost' with systemctl.py

    systemctl.py --ipv4 daemon-reload

## oneshoot services

https://www.freedesktop.org/software/systemd/man/systemd.service.html

There is a note that after "start unit" the unit is considered to be
active. A second "start unit" shall NOT run the execs again.

Indirectly it hints to the detail that a "start unit" does not run
any execs when "is-active unit" true. See issue #6.

## system specifies

https://www.freedesktop.org/software/systemd/man/systemd.unit.html

* "%n" Full unit name
* "%N" Unescaped full unit name = Same as "%n", but with escaping undone
* "%p" Prefix name = For instantiated units, this refers to the string before the "@" character of the unit name. For non-instantiated units, this refers to the name of the unit with the type suffix removed.
* "%P" Unescaped prefix name = Same as "%p", but with escaping undone
* "%i" Instance name = For instantiated units: this is the string between the "@" character and the suffix of the unit name.
* "%I" Unescaped instance name = Same as "%i", but with escaping undone
* "%f" Unescaped filename = This is either the unescaped instance name (if applicable) with / prepended (if applicable), or the unescaped prefix name prepended with /.
* "%t" Runtime directory = This is either /run (for the system manager) or the path "$XDG_RUNTIME_DIR" resolves to (for user managers).
* "%u" User name = This is the name of the user running the service manager instance. In case of the system manager this resolves to "root".
* "%U" User UID  = This is the numeric UID of the user running the service manager instance. In case of the system manager this resolves to "0".
* "%h" User home directory  = This is the home directory of the user running the service manager instance. In case of the system manager this resolves to "/root".
* "%s" User shell = This is the shell of the user running the service manager instance. In case of the system manager this resolves to "/bin/sh".
* "%m" Machine ID = The machine ID of the running system, formatted as string. See machine-id(5) for more information.
* "%b" Boot ID = The boot ID of the running system, formatted as string. See random(4) for more information.
* "%H" Host name = The hostname of the running system at the point in time the unit configuration is loaded.
* "%v" Kernel release = Identical to uname -r output
* "%%" Single percent sign = Use "%%" in place of "%" to specify a single percent sign.

## KillMode with SendSIGKILL and SendSIGHUP

When grepping through existing *.service files one can see a number
of references to those two options
* SendSIGKILL=no / postgresql + plymouth-start
* SendSIGHUP=yes / getty + console + journal-service

For a docker container one can safely exclude the latter but the
usage for postgresql makes it an importation option being in use.
In both cases the option changes the behaviour of "systemctl kill".

https://www.freedesktop.org/software/systemd/man/systemd.kill.html

As one can already guess, the default for SendSIGHUP is "no" and it
means that after the "KillSignal=" it should immediately send an
addtional SIGHUP - which is defined to also be forwarded to the
children of a program. 

One can also guess the default for SendSIGKILL to be "yes". Here it
is defined to work only after a timeout - as it shall send a KILL
to all remaining processes(!) of a service that are still around.

The third signal in use is a SIGTERM which is the default that is
sometimes overridden as "KillSignal=SIGHUP". Obviously it does
not make sense to mix Kill-SIGHUP and SendSIGHUP and there is no
such thing to exist in real service systemd files.

Last not least, there is also a "KillMode=control-group" default.
Which effectivly means that the specified sequences of Signals
(SIGTERM + SIGHUP & SIGKILL) should be sent to all processes.

As it is implemented for systemctl.py 1.0 there are some differences
in the handling - a "systemctl.py stop" will only execute ExecStop
but no implicit "systemctl kill" after that (KillMode=none). And
the SIGTERM & SIGKILL will only go the $MAINPID of a service as far
as we know about it (KillMode=process). If there is NO ExecStop
then there is a fallback from "signal stop" to "signal kill".

## Start Exec-Mode

It may be a little known detail from the systemd specs but in
reality one should NOT have multiple ExecStart lines in a spec
file. (quoting from specs: "Unless Type= is oneshot, exactly 
one command must be given.")

Instead the systemd specification has a number of prefixes for
the ExecStart line that modify the behaviour that are somewhat
related to the call of fork+execve.

https://www.freedesktop.org/software/systemd/man/systemd.service.html#ExecStart=

* a "-" prefix is the most obvious: ignore the return value
* a "@" prefix allows to have argv[0] to be not the called program
* a "+" prefix makes for a priviledged services process which
  basically means that User/Group changes should NOT be made
  (User/Group would affect ExecStart/ExecStop otherwise)
* and some unintelligible description of "!" and "!!" prefixes
  the seem to be alterations of a "+"

Moreover, "-" and "@" and "+" may be combined and the prefix
characters may occur in any order. Phew!

And just for another thing - one might expect that along with
ExecStart / ExecStop / ExecReload there also Exec-statements
for other frontend commands - including ExecXyPre/ExecXyPost
statements for each of them. But this is not the case - 
specifically there exists NO ExecRestart statement in the
systemd specs. Only these are valid:

* ExecStartPre
* ExecStart
* ExecStartPost
* ExecReload
* ExecStop
* ExecStopPost

On top of that, if any of the ExecStopPre, ExecStart, ExecStopPost
commands fails then you will also see that the original systemd 
daemon will execute the ExecStopPost lines to cleanup the system.
(and it will NOT run the ExecStop lines in that case). In order
for ExecStopPost to know the reason why it has been called there
are additional environment variables around

* $SERVICE_RESULT = "success", "timeout", "exit-code", "signal"..
* $EXIT_CODE = "exited", "killed" (the most relevant styles)
* $EXIT_STATUS - the exit-code of $MAINPID .. and for the
  combination of "success" + "exited" this is always 0.

That's a bit tricky to follow and "systemctl.py" does not do it.
One can assume that there is hardly any programmer to have taken
advantage of that - it sounds a bit like wizardry around.

As for the hint of "only one ExecStart" - while the script after
ExecStart is NOT run in a shell you can still use multiple
commands by having them seperated with "{space}{semicolon}{space}".
And a stray semicolon may instead be escaped with "\;". Phew!