mirror of
https://github.com/rclone/rclone.git
synced 2024-11-22 15:30:06 +08:00
14629c66f9
This prevents an `rclone rcd` server from prematurely going into the 'deactivating' state, which was causing systemd to kill it with a SIGABRT after the stop timeout. Fixes #7540
44 lines
1.2 KiB
Go
44 lines
1.2 KiB
Go
package systemd
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"sync"
|
|
|
|
"github.com/coreos/go-systemd/v22/daemon"
|
|
"github.com/rclone/rclone/lib/atexit"
|
|
)
|
|
|
|
// Notify systemd that the service is ready. This returns a
|
|
// function which should be called to notify that the service is
|
|
// stopping. This function will be called on exit if the service exits
|
|
// on a signal.
|
|
// NOTE: this function should only be called once, and so it
|
|
// should generally only be used directly in a command's Run handler.
|
|
// It should not be called as a result of rc commands. See #7540.
|
|
func Notify() func() {
|
|
if _, err := daemon.SdNotify(false, daemon.SdNotifyReady); err != nil {
|
|
log.Printf("failed to notify ready to systemd: %v", err)
|
|
}
|
|
var finaliseOnce sync.Once
|
|
finalise := func() {
|
|
finaliseOnce.Do(func() {
|
|
if _, err := daemon.SdNotify(false, daemon.SdNotifyStopping); err != nil {
|
|
log.Printf("failed to notify stopping to systemd: %v", err)
|
|
}
|
|
})
|
|
}
|
|
finaliseHandle := atexit.Register(finalise)
|
|
return func() {
|
|
atexit.Unregister(finaliseHandle)
|
|
finalise()
|
|
}
|
|
}
|
|
|
|
// UpdateStatus updates the systemd status
|
|
func UpdateStatus(status string) error {
|
|
systemdStatus := fmt.Sprintf("STATUS=%s", status)
|
|
_, err := daemon.SdNotify(false, systemdStatus)
|
|
return err
|
|
}
|