From b216c4f128e4714b4c237cf532497efe1f9afcf6 Mon Sep 17 00:00:00 2001 From: Toby Date: Fri, 17 May 2024 18:02:58 -0700 Subject: [PATCH] feat: graceful client shutdown --- app/cmd/client.go | 43 +++++++++++++++++++++++++++++++++++++++---- 1 file changed, 39 insertions(+), 4 deletions(-) diff --git a/app/cmd/client.go b/app/cmd/client.go index b6745abf4d..355523ac4b 100644 --- a/app/cmd/client.go +++ b/app/cmd/client.go @@ -9,10 +9,12 @@ import ( "net" "net/netip" "os" + "os/signal" "runtime" "slices" "strconv" "strings" + "syscall" "time" "github.com/spf13/cobra" @@ -512,13 +514,42 @@ func runClient(cmd *cobra.Command, args []string) { }) } - runner.Run() + signalChan := make(chan os.Signal, 1) + signal.Notify(signalChan, os.Interrupt, syscall.SIGTERM) + defer signal.Stop(signalChan) + + runnerChan := make(chan clientModeRunnerResult, 1) + go func() { + runnerChan <- runner.Run() + }() + + select { + case <-signalChan: + logger.Info("received signal, shutting down gracefully") + case r := <-runnerChan: + if r.OK { + logger.Info(r.Msg) + } else { + _ = c.Close() // Close the client here as Fatal will exit the program without running defer + if r.Err != nil { + logger.Fatal(r.Msg, zap.Error(r.Err)) + } else { + logger.Fatal(r.Msg) + } + } + } } type clientModeRunner struct { ModeMap map[string]func() error } +type clientModeRunnerResult struct { + OK bool + Msg string + Err error +} + func (r *clientModeRunner) Add(name string, f func() error) { if r.ModeMap == nil { r.ModeMap = make(map[string]func() error) @@ -526,9 +557,9 @@ func (r *clientModeRunner) Add(name string, f func() error) { r.ModeMap[name] = f } -func (r *clientModeRunner) Run() { +func (r *clientModeRunner) Run() clientModeRunnerResult { if len(r.ModeMap) == 0 { - logger.Fatal("no mode specified") + return clientModeRunnerResult{OK: false, Msg: "no mode specified"} } type modeError struct { @@ -546,9 +577,13 @@ func (r *clientModeRunner) Run() { for i := 0; i < len(r.ModeMap); i++ { e := <-errChan if e.Err != nil { - logger.Fatal("failed to run "+e.Name, zap.Error(e.Err)) + return clientModeRunnerResult{OK: false, Msg: "failed to run " + e.Name, Err: e.Err} } } + + // We don't really have any such cases, as currently none of our modes would stop on themselves without error. + // But we leave the possibility here for future expansion. + return clientModeRunnerResult{OK: true, Msg: "finished without error"} } func clientSOCKS5(config socks5Config, c client.Client) error {