conduit icon indicating copy to clipboard operation
conduit copied to clipboard

Graceful Pipeline Exit

Open tzaffi opened this issue 2 years ago • 1 comments

Problem

Currently, when the main conduit process receives a shutdown signal, it abruptly tears down all its goroutines even though the pipeline shares a common context.Context object that is supposed to make it possible to exit more gracefully and in a deterministic fashion. In particular, it is currently not possible to coordinate a new requirement such as having each plugin finish the current round before exit.

Solution

This story posits the following goals (though after further discussion we may want to trim the goals or break them out into separate issues):

  1. Handle an interrupt signal in a way that won't cause duplicate calls to plugin Stop(). As a starting point refer to #97 but better crafting of where to place the signal handling is required. See also #99
  2. Consider adding a new conduit stop command that looks for the PID file and sends an interrupt to that process
  3. Consider the possibility of letting each plugin go to the end. Even better make this configurable via the type of interrupt signal received. EG:
    • conduit stop now would send an os.Interrupt (or a kill -TERM $(PROCNUM) ) and kills all plugin processes immediately
    • conduit stop end-of-round produces something like kill -USR1 $(PROCNUM) and kills plugin process in the following order at the end of its current round: ImporterProcessorsExporter

Some questions

  1. Should we add WhyStopped() to the Pipeline interface, and send a telemetry observation with this context cancellation cause?
  2. Do we still need Stop() in the Pipeline's interface?
  3. Do we need sentinel error to act as sentinel cancelation causes and to distinguish between different legitimate reasons?

Some possible code in the client making use of the stop cause:

defer func() {
    pline.Stop()
    if !errors.Is(pline.WhyStopped(), pipeline.BecauseStopMethod) {
        logger.Error("unexpected conduit pipeline exit: %v", pline.WhyStopped())
    }
}()

Dependencies

None

Urgency

Low - I haven't heard of other developers in the community complain of this issue.

tzaffi avatar Jun 12 '23 20:06 tzaffi

Copying similar description from #99

Subject of the issue

The pipeline object is managed using a context with a cancel function and a wait group. This setup is strange.

Rather than managing the process internal to the pipeline, Start() should be renamed to Run(), and Run() should be blocking. If the context is cancelled Run() can call Stop() automatically.

The caller would then use a waitgroup and run the pipeline in a go-routine if it wants to. Run() could also return the error directly rather than implementing setError and Error.

winder avatar Aug 10 '23 16:08 winder