2
2
mirror of https://github.com/octoleo/restic.git synced 2024-12-04 10:58:26 +00:00
restic/cmd/restic/cleanup.go

90 lines
2.0 KiB
Go
Raw Normal View History

2015-07-19 15:50:55 +00:00
package main
import (
"os"
"os/signal"
"sync"
"syscall"
2017-07-23 12:21:03 +00:00
"github.com/restic/restic/internal/debug"
2015-07-19 15:50:55 +00:00
)
var cleanupHandlers struct {
sync.Mutex
list []func(code int) (int, error)
done bool
ch chan os.Signal
2015-07-19 15:50:55 +00:00
}
func init() {
cleanupHandlers.ch = make(chan os.Signal, 1)
go CleanupHandler(cleanupHandlers.ch)
signal.Notify(cleanupHandlers.ch, syscall.SIGINT, syscall.SIGTERM)
2017-09-23 09:12:44 +00:00
}
2015-07-19 15:50:55 +00:00
// AddCleanupHandler adds the function f to the list of cleanup handlers so
// that it is executed when all the cleanup handlers are run, e.g. when SIGINT
// is received.
func AddCleanupHandler(f func(code int) (int, error)) {
2015-07-19 15:50:55 +00:00
cleanupHandlers.Lock()
defer cleanupHandlers.Unlock()
2016-09-15 19:17:20 +00:00
// reset the done flag for integration tests
cleanupHandlers.done = false
2015-07-19 15:50:55 +00:00
cleanupHandlers.list = append(cleanupHandlers.list, f)
}
// RunCleanupHandlers runs all registered cleanup handlers
func RunCleanupHandlers(code int) int {
2015-07-19 15:50:55 +00:00
cleanupHandlers.Lock()
defer cleanupHandlers.Unlock()
2015-07-19 15:57:18 +00:00
if cleanupHandlers.done {
return code
2015-07-19 15:57:18 +00:00
}
cleanupHandlers.done = true
2015-07-19 15:50:55 +00:00
for _, f := range cleanupHandlers.list {
var err error
code, err = f(code)
2015-07-19 15:50:55 +00:00
if err != nil {
Warnf("error in cleanup handler: %v\n", err)
2015-07-19 15:50:55 +00:00
}
}
2016-09-15 19:17:20 +00:00
cleanupHandlers.list = nil
return code
2015-07-19 15:50:55 +00:00
}
2024-02-22 21:00:42 +00:00
// CleanupHandler handles the SIGINT and SIGTERM signals.
2015-07-19 15:50:55 +00:00
func CleanupHandler(c <-chan os.Signal) {
for s := range c {
2016-09-27 20:35:08 +00:00
debug.Log("signal %v received, cleaning up", s)
Warnf("%ssignal %v received, cleaning up\n", clearLine(0), s)
if val, _ := os.LookupEnv("RESTIC_DEBUG_STACKTRACE_SIGINT"); val != "" {
_, _ = os.Stderr.WriteString("\n--- STACKTRACE START ---\n\n")
_, _ = os.Stderr.WriteString(debug.DumpStacktrace())
_, _ = os.Stderr.WriteString("\n--- STACKTRACE END ---\n")
}
code := 0
if s == syscall.SIGINT || s == syscall.SIGTERM {
code = 130
} else {
code = 1
}
Exit(code)
2015-07-19 15:50:55 +00:00
}
}
// Exit runs the cleanup handlers and then terminates the process with the
// given exit code.
func Exit(code int) {
code = RunCleanupHandlers(code)
debug.Log("exiting with status code %d", code)
os.Exit(code)
}