|
1 | 1 | package extsignals |
2 | 2 |
|
3 | 3 | import ( |
4 | | - "fmt" |
5 | | - "github.com/rs/zerolog/log" |
6 | | - "golang.org/x/sys/unix" |
| 4 | + "context" |
7 | 5 | "os" |
8 | 6 | "os/signal" |
9 | 7 | "sort" |
10 | 8 | "sync" |
11 | 9 | "syscall" |
| 10 | + |
| 11 | + "github.com/rs/zerolog/log" |
12 | 12 | ) |
13 | 13 |
|
14 | 14 | var ( |
@@ -40,44 +40,64 @@ func AddSignalHandler(signalHandler SignalHandler) { |
40 | 40 | handlers.Store(signalHandler.Name, signalHandler) |
41 | 41 | } |
42 | 42 |
|
| 43 | +func ClearSignalHandlers() { |
| 44 | + handlers.Range(func(key, value interface{}) bool { |
| 45 | + handlers.Delete(key) |
| 46 | + return true |
| 47 | + }) |
| 48 | +} |
| 49 | + |
43 | 50 | // RemoveSignalHandlersByName removes signal handlers by name. This is mainly used for testing. |
44 | 51 | func RemoveSignalHandlersByName(names ...string) { |
45 | 52 | for _, name := range names { |
46 | 53 | handlers.Delete(name) |
47 | 54 | } |
48 | 55 | } |
49 | 56 |
|
| 57 | +func createSignalChannel(context context.Context) { |
| 58 | + signalChannel := make(chan os.Signal, 1) |
| 59 | + Notify(signalChannel) |
| 60 | + go func(signals <-chan os.Signal) { |
| 61 | + for { |
| 62 | + select { |
| 63 | + case <-context.Done(): |
| 64 | + signal.Stop(signalChannel) |
| 65 | + return |
| 66 | + case s := <-signals: |
| 67 | + handlerList := make([]SignalHandler, 0) |
| 68 | + handlers.Range(func(key, value interface{}) bool { |
| 69 | + handlerList = append(handlerList, value.(SignalHandler)) |
| 70 | + return true |
| 71 | + }) |
| 72 | + sort.Sort(ByOrder(handlerList)) |
| 73 | + signalName := GetSignalName(s.(syscall.Signal)) |
| 74 | + for _, handler := range handlerList { |
| 75 | + log.Debug().Str("signal", signalName).Str("handler", handler.Name).Int("order", handler.Order).Msg("received signal - call handler") |
| 76 | + handler.Handler(s) |
| 77 | + } |
| 78 | + } |
| 79 | + } |
| 80 | + }(signalChannel) |
| 81 | +} |
| 82 | + |
50 | 83 | func ActivateSignalHandlers() { |
| 84 | + ActivateSignalHandlerWithContext(context.Background()) |
| 85 | +} |
| 86 | + |
| 87 | +func ActivateSignalHandlerWithContext(context context.Context) { |
51 | 88 | AddSignalHandler(SignalHandler{ |
52 | 89 | Handler: func(signal os.Signal) { |
53 | 90 | switch signal { |
54 | 91 | case syscall.SIGINT: |
55 | 92 | os.Exit(128 + int(signal.(syscall.Signal))) |
56 | 93 |
|
57 | 94 | case syscall.SIGTERM: |
58 | | - fmt.Printf("Terminated: %d\n", int(signal.(syscall.Signal))) |
59 | 95 | os.Exit(128 + int(signal.(syscall.Signal))) |
60 | 96 | } |
61 | 97 | }, |
62 | 98 | Order: OrderTermination, |
63 | 99 | Name: "Termination", |
64 | 100 | }) |
65 | 101 |
|
66 | | - signalChannel := make(chan os.Signal, 1) |
67 | | - signal.Notify(signalChannel, syscall.SIGINT, syscall.SIGTERM, syscall.SIGUSR1) |
68 | | - go func(signals <-chan os.Signal) { |
69 | | - for s := range signals { |
70 | | - handlerList := make([]SignalHandler, 0) |
71 | | - handlers.Range(func(key, value interface{}) bool { |
72 | | - handlerList = append(handlerList, value.(SignalHandler)) |
73 | | - return true |
74 | | - }) |
75 | | - sort.Sort(ByOrder(handlerList)) |
76 | | - signalName := unix.SignalName(s.(syscall.Signal)) |
77 | | - for _, handler := range handlerList { |
78 | | - log.Debug().Str("signal", signalName).Str("handler", handler.Name).Int("order", handler.Order).Msg("received signal - call handler") |
79 | | - handler.Handler(s) |
80 | | - } |
81 | | - } |
82 | | - }(signalChannel) |
| 102 | + createSignalChannel(context) |
83 | 103 | } |
0 commit comments