Merge pull request #1088 from apernet/wip-shutdown

feat: graceful client shutdown
This commit is contained in:
Toby 2024-05-17 19:50:14 -07:00 committed by GitHub
commit 15e58468a7
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -9,10 +9,12 @@ import (
"net" "net"
"net/netip" "net/netip"
"os" "os"
"os/signal"
"runtime" "runtime"
"slices" "slices"
"strconv" "strconv"
"strings" "strings"
"syscall"
"time" "time"
"github.com/spf13/cobra" "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 { type clientModeRunner struct {
ModeMap map[string]func() error ModeMap map[string]func() error
} }
type clientModeRunnerResult struct {
OK bool
Msg string
Err error
}
func (r *clientModeRunner) Add(name string, f func() error) { func (r *clientModeRunner) Add(name string, f func() error) {
if r.ModeMap == nil { if r.ModeMap == nil {
r.ModeMap = make(map[string]func() error) r.ModeMap = make(map[string]func() error)
@ -526,9 +557,9 @@ func (r *clientModeRunner) Add(name string, f func() error) {
r.ModeMap[name] = f r.ModeMap[name] = f
} }
func (r *clientModeRunner) Run() { func (r *clientModeRunner) Run() clientModeRunnerResult {
if len(r.ModeMap) == 0 { if len(r.ModeMap) == 0 {
logger.Fatal("no mode specified") return clientModeRunnerResult{OK: false, Msg: "no mode specified"}
} }
type modeError struct { type modeError struct {
@ -546,9 +577,13 @@ func (r *clientModeRunner) Run() {
for i := 0; i < len(r.ModeMap); i++ { for i := 0; i < len(r.ModeMap); i++ {
e := <-errChan e := <-errChan
if e.Err != nil { 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 { func clientSOCKS5(config socks5Config, c client.Client) error {