Skip to content

Commit af18206

Browse files
corherethaJeztah
andcommitted
daemon/command: add support for sd_notify "reload" notifications
commit f74b856 added an [ExecReload] to the systemd unit, which allows users to signal the daemon to reload its config through systemd (`systemctl reload docker.service`). While reloading works, systemd expects the `ExecReload` command to be synchronous, so that it knows when the reload completes, and can account for this when managing dependent services. > Note however that reloading a daemon by enqueuing a signal (...) is usually > not a good choice, because this is an asynchronous operation and hence not > suitable when ordering reloads of multiple services against each other. Systemd 253 introduced a new Type (Type=notify-reload, see [systemd#25916]), which allows setting a "ReloadSignal" instead, if the service supports it by including a [MONOTONIC_USEC] value in its "RELOADING=1" notifications. This patch: - adds reload notifications to the daemon to notify when the daemon got signaled to reload its configuration see [sd_notify(3)]. - appends a [MONOTONIC_USEC] value to the reload notification to support systemd 253's notify-reload protocol. This is backwards-compatible with older systemd releases as the systemd service manager ignores unknown assignments in notifications. - adds a `notifyReady` callback to notify when the reload finished, which we currently send regardless if the reload was successful or failed (which could be due to an invalid config). [ExecReload]: https://www.freedesktop.org/software/systemd/man/latest/systemd.service.html#ExecReload= [systemd#25916]: systemd/systemd#25916 [MONOTONIC_USEC]: https://www.freedesktop.org/software/systemd/man/latest/sd_notify.html#MONOTONIC_USEC=… [sd_notify(3)]: https://www.freedesktop.org/software/systemd/man/latest/sd_notify.html#RELOADING=1 Signed-off-by: Cory Snider <[email protected]> Co-authored-by: Sebastiaan van Stijn <[email protected]> Signed-off-by: Sebastiaan van Stijn <[email protected]>
1 parent 0e52362 commit af18206

4 files changed

Lines changed: 34 additions & 0 deletions

File tree

daemon/command/daemon.go

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -521,6 +521,13 @@ func (cli *daemonCLI) reloadConfig() {
521521
}
522522
}
523523

524+
// On Linux, we use sd_notify to indicate we're reloading config. We send
525+
// this signal as early as possible to let systemd know we're reloading,
526+
// but must signal "ready" after this completes (even on failure), which
527+
// is done by the reload function defined above.
528+
done := notifyReloading()
529+
defer done()
530+
524531
if err := config.Reload(*cli.configFile, cli.flags, reload); err != nil {
525532
log.G(ctx).WithError(err).Error("Error reloading configuration")
526533
return

daemon/command/daemon_freebsd.go

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,10 @@ func preNotifyReady() error {
1111
func notifyReady() {
1212
}
1313

14+
// notifyReloading sends a message to the host when the server got signaled to
15+
// reloading its configuration. It is a no-op on FreeBSD.
16+
func notifyReloading() func() { return func() {} }
17+
1418
// notifyStopping sends a message to the host when the server is shutting down
1519
func notifyStopping() {
1620
}

daemon/command/daemon_linux.go

Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,25 @@ func notifyStopping() {
5858
_, _ = systemdDaemon.SdNotify(false, systemdDaemon.SdNotifyStopping)
5959
}
6060

61+
// notifyReloading sends a message to the host when the server got signaled to
62+
// reloading its configuration, see [sd_notify(3)]. The server should be running
63+
// as a systemd unit with "Type=notify" or "Type=notify-reload" (see
64+
// [systemd.service(5)]).
65+
//
66+
// notifyReloading returns a callback that must be called after reloading completes
67+
// (either successfully or unsuccessfully) to send [notifyReady].
68+
//
69+
// [sd_notify(3)]: https://www.freedesktop.org/software/systemd/man/latest/sd_notify.html#RELOADING=1
70+
// [systemd.service(5)]: https://www.freedesktop.org/software/systemd/man/latest/systemd.service.html#Type=
71+
func notifyReloading() (done func()) {
72+
sent, _ := systemdDaemon.SdNotify(false, systemdDaemon.SdNotifyReloading+"\n"+systemdDaemon.SdNotifyMonotonicUsec())
73+
if !sent {
74+
// Nothing to do if no reloading event was sent.
75+
return func() {}
76+
}
77+
return notifyReady
78+
}
79+
6180
func validateCPURealtimeOptions(cfg *config.Config) error {
6281
if cfg.CPURealtimePeriod == 0 && cfg.CPURealtimeRuntime == 0 {
6382
return nil

daemon/command/daemon_windows.go

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,10 @@ func preNotifyReady() error {
5656
func notifyReady() {
5757
}
5858

59+
// notifyReloading sends a message to the host when the server got signaled to
60+
// reloading its configuration. It is a no-op on Windows.
61+
func notifyReloading() func() { return func() {} }
62+
5963
// notifyStopping sends a message to the host when the server is shutting down
6064
func notifyStopping() {
6165
}

0 commit comments

Comments
 (0)