mirror of
https://github.com/opencloud-eu/opencloud.git
synced 2026-01-04 12:10:21 -05:00
Bumps [github.com/oklog/run](https://github.com/oklog/run) from 1.1.0 to 1.2.0. - [Release notes](https://github.com/oklog/run/releases) - [Commits](https://github.com/oklog/run/compare/v1.1.0...v1.2.0) --- updated-dependencies: - dependency-name: github.com/oklog/run dependency-version: 1.2.0 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com>
97 lines
2.8 KiB
Go
97 lines
2.8 KiB
Go
package run
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"os/signal"
|
|
)
|
|
|
|
// ContextHandler returns an actor, i.e. an execute and interrupt func, that
|
|
// terminates when the provided context is canceled.
|
|
func ContextHandler(ctx context.Context) (execute func() error, interrupt func(error)) {
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
return func() error {
|
|
<-ctx.Done()
|
|
return ctx.Err()
|
|
}, func(error) {
|
|
cancel()
|
|
}
|
|
}
|
|
|
|
// SignalHandler returns an actor, i.e. an execute and interrupt func, that
|
|
// terminates with ErrSignal when the process receives one of the provided
|
|
// signals, or with ctx.Error() when the parent context is canceled. If no
|
|
// signals are provided, the actor will terminate on any signal, per
|
|
// [signal.Notify].
|
|
func SignalHandler(ctx context.Context, signals ...os.Signal) (execute func() error, interrupt func(error)) {
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
return func() error {
|
|
testc := getTestSigChan(ctx)
|
|
sigc := make(chan os.Signal, 1)
|
|
signal.Notify(sigc, signals...)
|
|
defer signal.Stop(sigc)
|
|
select {
|
|
case sig := <-testc:
|
|
return &SignalError{Signal: sig}
|
|
case sig := <-sigc:
|
|
return &SignalError{Signal: sig}
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
}
|
|
}, func(error) {
|
|
cancel()
|
|
}
|
|
}
|
|
|
|
type testSigChanKey struct{}
|
|
|
|
func getTestSigChan(ctx context.Context) <-chan os.Signal {
|
|
c, _ := ctx.Value(testSigChanKey{}).(<-chan os.Signal) // can be nil
|
|
return c
|
|
}
|
|
|
|
func putTestSigChan(ctx context.Context, c <-chan os.Signal) context.Context {
|
|
return context.WithValue(ctx, testSigChanKey{}, c)
|
|
}
|
|
|
|
// SignalError is returned by the signal handler's execute function when it
|
|
// terminates due to a received signal.
|
|
//
|
|
// SignalError has a design error that impacts comparison with errors.As.
|
|
// Callers should prefer using errors.Is(err, ErrSignal) to check for signal
|
|
// errors, and should only use errors.As in the rare case that they need to
|
|
// program against the specific os.Signal value.
|
|
type SignalError struct {
|
|
Signal os.Signal
|
|
}
|
|
|
|
// Error implements the error interface.
|
|
//
|
|
// It was a design error to define this method on a value receiver rather than a
|
|
// pointer receiver. For compatibility reasons it won't be changed.
|
|
func (e SignalError) Error() string {
|
|
return fmt.Sprintf("received signal %s", e.Signal)
|
|
}
|
|
|
|
// Is addresses a design error in the SignalError type, so that errors.Is with
|
|
// ErrSignal will return true.
|
|
func (e SignalError) Is(err error) bool {
|
|
return errors.Is(err, ErrSignal)
|
|
}
|
|
|
|
// As fixes a design error in the SignalError type, so that errors.As with the
|
|
// literal `&SignalError{}` will return true.
|
|
func (e SignalError) As(target interface{}) bool {
|
|
switch target.(type) {
|
|
case *SignalError, SignalError:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
// ErrSignal is returned by SignalHandler when a signal triggers termination.
|
|
var ErrSignal = errors.New("signal error")
|