mirror of
https://github.com/apernet/hysteria.git
synced 2025-04-03 20:47:38 +03:00
Merge pull request #1088 from apernet/wip-shutdown
feat: graceful client shutdown
This commit is contained in:
commit
15e58468a7
1 changed files with 39 additions and 4 deletions
|
@ -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 {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue