lntest: Improve HarnessNode stop logic and remove restart().

This commit is contained in:
Jim Posen 2017-11-03 15:21:35 -07:00 committed by Olaoluwa Osuntokun
parent 19ed1fb8db
commit 02177a3f1c
2 changed files with 32 additions and 44 deletions

@ -338,7 +338,17 @@ func (n *NetworkHarness) DisconnectNodes(ctx context.Context, a, b *HarnessNode)
// and invalidated prior state, or persistent state recovery, simulating node // and invalidated prior state, or persistent state recovery, simulating node
// crashes, etc. // crashes, etc.
func (n *NetworkHarness) RestartNode(node *HarnessNode, callback func() error) error { func (n *NetworkHarness) RestartNode(node *HarnessNode, callback func() error) error {
return node.restart(n.lndErrorChan, callback) if err := node.stop(); err != nil {
return err
}
if callback != nil {
if err := callback(); err != nil {
return err
}
}
return node.start(n.lndErrorChan)
} }
// ShutdownNode stops an active lnd process and returns when the process has // ShutdownNode stops an active lnd process and returns when the process has

@ -204,8 +204,6 @@ func newNode(cfg nodeConfig) (*HarnessNode, error) {
cfg: &cfg, cfg: &cfg,
NodeID: nodeNum, NodeID: nodeNum,
chanWatchRequests: make(chan *chanWatchRequest), chanWatchRequests: make(chan *chanWatchRequest),
processExit: make(chan struct{}),
quit: make(chan struct{}),
}, nil }, nil
} }
@ -217,7 +215,12 @@ func (hn *HarnessNode) DBPath() string {
// Start launches a new process running lnd. Additionally, the PID of the // Start launches a new process running lnd. Additionally, the PID of the
// launched process is saved in order to possibly kill the process forcibly // launched process is saved in order to possibly kill the process forcibly
// later. // later.
//
// This may not clean up properly if an error is returned, so the caller should
// call shutdown() regardless of the return value.
func (hn *HarnessNode) start(lndError chan<- error) error { func (hn *HarnessNode) start(lndError chan<- error) error {
hn.quit = make(chan struct{})
args := hn.cfg.genArgs() args := hn.cfg.genArgs()
hn.cmd = exec.Command("lnd", args...) hn.cmd = exec.Command("lnd", args...)
@ -251,6 +254,7 @@ func (hn *HarnessNode) start(lndError chan<- error) error {
// Launch a new goroutine which that bubbles up any potential fatal // Launch a new goroutine which that bubbles up any potential fatal
// process errors to the goroutine running the tests. // process errors to the goroutine running the tests.
hn.processExit = make(chan struct{})
go func() { go func() {
err := hn.cmd.Wait() err := hn.cmd.Wait()
@ -363,56 +367,30 @@ func (hn *HarnessNode) cleanup() error {
// Stop attempts to stop the active lnd process. // Stop attempts to stop the active lnd process.
func (hn *HarnessNode) stop() error { func (hn *HarnessNode) stop() error {
// Do nothing if the process never started successfully. // Do nothing if the process is not running.
if hn.LightningClient == nil { if hn.processExit == nil {
return nil return nil
} }
// Do nothing if the process already finished. // If start() failed before creating a client, we will just wait for the
select { // child process to die.
case <-hn.quit: if hn.LightningClient != nil {
return nil // Don't watch for error because sometimes the RPC connection gets
case <-hn.processExit: // closed before a response is returned.
return nil req := lnrpc.StopRequest{}
default: ctx := context.Background()
hn.LightningClient.StopDaemon(ctx, &req)
} }
// Don't watch for error because sometimes the RPC connection gets // Wait for lnd process and other goroutines to exit.
// closed before a response is returned. <-hn.processExit
req := lnrpc.StopRequest{}
ctx := context.Background()
hn.LightningClient.StopDaemon(ctx, &req)
close(hn.quit) close(hn.quit)
hn.wg.Wait() hn.wg.Wait()
return nil
}
// Restart attempts to restart a lightning node by shutting it down cleanly,
// then restarting the process. This function is fully blocking. Upon restart,
// the RPC connection to the node will be re-attempted, continuing iff the
// connection attempt is successful. Additionally, if a callback is passed, the
// closure will be executed after the node has been shutdown, but before the
// process has been started up again.
func (hn *HarnessNode) restart(errChan chan error, callback func() error) error {
if err := hn.stop(); err != nil {
return err
}
<-hn.processExit
hn.quit = nil
hn.processExit = nil
hn.LightningClient = nil hn.LightningClient = nil
hn.processExit = make(chan struct{}) return nil
hn.quit = make(chan struct{})
hn.wg = sync.WaitGroup{}
if callback != nil {
if err := callback(); err != nil {
return err
}
}
return hn.start(errChan)
} }
// shutdown stops the active lnd process and cleans up any temporary directories // shutdown stops the active lnd process and cleans up any temporary directories