From 6a940d9ea8a92c1acbc84e600878dc72792b5e3a Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Mon, 29 Jul 2013 12:04:48 -0700 Subject: [PATCH] packer: use locks/conds to avoid races on RemoteCmd.Exited [GH-42] --- packer/communicator.go | 37 +++++++++++++++++++++++++++++++++++-- packer/communicator_test.go | 5 ++--- 2 files changed, 37 insertions(+), 5 deletions(-) diff --git a/packer/communicator.go b/packer/communicator.go index ff992743f..4c8c5cce2 100644 --- a/packer/communicator.go +++ b/packer/communicator.go @@ -4,7 +4,7 @@ import ( "github.com/mitchellh/iochan" "io" "strings" - "time" + "sync" ) // RemoteCmd represents a remote command being prepared or run. @@ -32,6 +32,10 @@ type RemoteCmd struct { // Once Exited is true, this will contain the exit code of the process. ExitStatus int + + // Internal locks and such used for safely setting some shared variables + l sync.Mutex + exitCond *sync.Cond } // A Communicator is the interface used to communicate with the machine @@ -125,9 +129,38 @@ OutputLoop: return nil } +// SetExited is a helper for setting that this process is exited. This +// should be called by communicators who are running a remote command in +// order to set that the command is done. +func (r *RemoteCmd) SetExited(status int) { + r.l.Lock() + if r.exitCond == nil { + r.exitCond = sync.NewCond(new(sync.Mutex)) + } + r.l.Unlock() + + r.exitCond.L.Lock() + defer r.exitCond.L.Unlock() + + r.Exited = true + r.ExitStatus = status + r.exitCond.Broadcast() +} + // Wait waits for the remote command to complete. func (r *RemoteCmd) Wait() { + // Make sure our condition variable is initialized. + r.l.Lock() + if r.exitCond == nil { + r.exitCond = sync.NewCond(new(sync.Mutex)) + } + r.l.Unlock() + + // Wait on the condition variable to notify we exited + r.exitCond.L.Lock() + defer r.exitCond.L.Unlock() + for !r.Exited { - time.Sleep(50 * time.Millisecond) + r.exitCond.Wait() } } diff --git a/packer/communicator_test.go b/packer/communicator_test.go index 96e823a26..d9f31c2bd 100644 --- a/packer/communicator_test.go +++ b/packer/communicator_test.go @@ -59,7 +59,7 @@ func TestRemoteCmd_StartWithUi(t *testing.T) { go func() { time.Sleep(100 * time.Millisecond) - rc.Exited = true + rc.SetExited(0) }() err := rc.StartWithUi(testComm, testUi) @@ -85,8 +85,7 @@ func TestRemoteCmd_Wait(t *testing.T) { result <- true }() - cmd.ExitStatus = 42 - cmd.Exited = true + cmd.SetExited(42) select { case <-result: