提交 e4afcb71 作者: Steven Allen

interrupt: fix send on closed

If we get a signal while shutting down, we could end up sending on a closed
channel.

License: MIT
Signed-off-by: 's avatarSteven Allen <steven@stebalien.com>
上级 1682880d
...@@ -15,38 +15,43 @@ import ( ...@@ -15,38 +15,43 @@ import (
// IntrHandler helps set up an interrupt handler that can // IntrHandler helps set up an interrupt handler that can
// be cleanly shut down through the io.Closer interface. // be cleanly shut down through the io.Closer interface.
type IntrHandler struct { type IntrHandler struct {
sig chan os.Signal closing chan struct{}
wg sync.WaitGroup wg sync.WaitGroup
} }
func NewIntrHandler() *IntrHandler { func NewIntrHandler() *IntrHandler {
ih := &IntrHandler{} return &IntrHandler{closing: make(chan struct{})}
ih.sig = make(chan os.Signal, 1)
return ih
} }
func (ih *IntrHandler) Close() error { func (ih *IntrHandler) Close() error {
close(ih.sig) close(ih.closing)
ih.wg.Wait() ih.wg.Wait()
return nil return nil
} }
// Handle starts handling the given signals, and will call the handler // Handle starts handling the given signals, and will call the handler
// callback function each time a signal is catched. The function is passed // callback function each time a signal is caught. The function is passed
// the number of times the handler has been triggered in total, as // the number of times the handler has been triggered in total, as
// well as the handler itself, so that the handling logic can use the // well as the handler itself, so that the handling logic can use the
// handler's wait group to ensure clean shutdown when Close() is called. // handler's wait group to ensure clean shutdown when Close() is called.
func (ih *IntrHandler) Handle(handler func(count int, ih *IntrHandler), sigs ...os.Signal) { func (ih *IntrHandler) Handle(handler func(count int, ih *IntrHandler), sigs ...os.Signal) {
signal.Notify(ih.sig, sigs...) notify := make(chan os.Signal, 1)
signal.Notify(notify, sigs...)
ih.wg.Add(1) ih.wg.Add(1)
go func() { go func() {
defer ih.wg.Done() defer ih.wg.Done()
defer signal.Stop(notify)
count := 0 count := 0
for range ih.sig { for {
count++ select {
handler(count, ih) case <-ih.closing:
return
case <-notify:
count++
handler(count, ih)
}
} }
signal.Stop(ih.sig)
}() }()
} }
......
Markdown 格式
0%
您添加了 0 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论