Handle the SIGINT and SIGTERM signals to support stopping a test cleanly (#84)

This commit is contained in:
Arturo Filastò 2019-12-27 11:32:08 +01:00 committed by Simone Basso
parent 265177a8a6
commit 7bbbab8774
3 changed files with 48 additions and 5 deletions

View File

@ -2,6 +2,9 @@ package run
import ( import (
"errors" "errors"
"os"
"os/signal"
"syscall"
"github.com/alecthomas/kingpin" "github.com/alecthomas/kingpin"
"github.com/apex/log" "github.com/apex/log"
@ -13,6 +16,22 @@ import (
"github.com/ooni/probe-cli/nettests" "github.com/ooni/probe-cli/nettests"
) )
// listenForSignals will listen for SIGINT and SIGTERM. When it receives those
// signals it will set isTerminatedAtomicInt to non-zero, which will cleanly
// shutdown the test logic.
// TODO refactor this to use a cancellable context.Context instead of a bool
// flag, probably as part of: https://github.com/ooni/probe-cli/issues/45
func listenForSignals(ctx *ooni.Context) {
s := make(chan os.Signal, 1)
signal.Notify(s, os.Interrupt, syscall.SIGTERM)
go func() {
<-s
log.Info("caught a stop signal, shutting down cleanly")
ctx.Terminate()
}()
}
func runNettestGroup(tg string, ctx *ooni.Context, network *database.Network) error { func runNettestGroup(tg string, ctx *ooni.Context, network *database.Network) error {
group, ok := nettests.NettestGroups[tg] group, ok := nettests.NettestGroups[tg]
if !ok { if !ok {
@ -27,13 +46,17 @@ func runNettestGroup(tg string, ctx *ooni.Context, network *database.Network) er
return err return err
} }
listenForSignals(ctx)
for i, nt := range group.Nettests { for i, nt := range group.Nettests {
if ctx.IsTerminated() == true {
log.Debugf("context is terminated, breaking")
break
}
log.Debugf("Running test %T", nt) log.Debugf("Running test %T", nt)
ctl := nettests.NewController(nt, ctx, result) ctl := nettests.NewController(nt, ctx, result)
ctl.SetNettestIndex(i, len(group.Nettests)) ctl.SetNettestIndex(i, len(group.Nettests))
if err = nt.Run(ctl); err != nil { if err = nt.Run(ctl); err != nil {
log.WithError(err).Errorf("Failed to run %s", group.Label) log.WithError(err).Errorf("Failed to run %s", group.Label)
return err
} }
} }

View File

@ -77,7 +77,6 @@ func (c *Controller) SetNettestIndex(i, n int) {
// This function will continue to run in most cases but will // This function will continue to run in most cases but will
// immediately halt if something's wrong with the file system. // immediately halt if something's wrong with the file system.
func (c *Controller) Run(builder *engine.ExperimentBuilder, inputs []string) error { func (c *Controller) Run(builder *engine.ExperimentBuilder, inputs []string) error {
// This will configure the controller as handler for the callbacks // This will configure the controller as handler for the callbacks
// called by ooni/probe-engine/experiment.Experiment. // called by ooni/probe-engine/experiment.Experiment.
builder.SetCallbacks(engine.Callbacks(c)) builder.SetCallbacks(engine.Callbacks(c))
@ -108,6 +107,10 @@ func (c *Controller) Run(builder *engine.ExperimentBuilder, inputs []string) err
c.ntStartTime = time.Now() c.ntStartTime = time.Now()
for idx, input := range inputs { for idx, input := range inputs {
if c.Ctx.IsTerminated() == true {
log.Debug("isTerminated == true, breaking the input loop")
break
}
c.curInputIdx = idx // allow for precise progress c.curInputIdx = idx // allow for precise progress
idx64 := int64(idx) idx64 := int64(idx)
log.Debug(color.RedString("status.measurement_start")) log.Debug(color.RedString("status.measurement_start"))

23
ooni.go
View File

@ -3,6 +3,7 @@ package ooni
import ( import (
"io/ioutil" "io/ioutil"
"os" "os"
"sync/atomic"
"github.com/apex/log" "github.com/apex/log"
"github.com/ooni/probe-cli/config" "github.com/ooni/probe-cli/config"
@ -29,6 +30,10 @@ type Context struct {
dbPath string dbPath string
configPath string configPath string
// We need to use a int64 in order to use the atomic.AddInt64/LoadInt64
// operations to ensure consistent reads of the variables.
isTerminatedAtomicInt int64
} }
// MaybeLocationLookup will lookup the location of the user unless it's already cached // MaybeLocationLookup will lookup the location of the user unless it's already cached
@ -36,6 +41,17 @@ func (c *Context) MaybeLocationLookup() error {
return c.Session.MaybeLookupLocation() return c.Session.MaybeLookupLocation()
} }
// IsTerminated checks to see if the isTerminatedAtomicInt is set to a non zero
// value and therefore we have received the signal to shutdown the running test
func (c *Context) IsTerminated() bool {
i := atomic.LoadInt64(&c.isTerminatedAtomicInt)
return i != 0
}
func (c *Context) Terminate() {
atomic.AddInt64(&c.isTerminatedAtomicInt, 1)
}
// Init the OONI manager // Init the OONI manager
func (c *Context) Init() error { func (c *Context) Init() error {
var err error var err error
@ -94,9 +110,10 @@ func (c *Context) Init() error {
// NewContext creates a new context instance. // NewContext creates a new context instance.
func NewContext(configPath string, homePath string) *Context { func NewContext(configPath string, homePath string) *Context {
return &Context{ return &Context{
Home: homePath, Home: homePath,
Config: &config.Config{}, Config: &config.Config{},
configPath: configPath, configPath: configPath,
isTerminatedAtomicInt: 0,
} }
} }