packer/rpc: use packer.MockBuilder for tests

This commit is contained in:
Mitchell Hashimoto 2013-11-02 22:47:23 -05:00
parent 623706676b
commit a4077c9d8a
3 changed files with 76 additions and 81 deletions

View File

@ -1,11 +1,17 @@
package packer package packer
import (
"errors"
)
// MockBuilder is an implementation of Builder that can be used for tests. // MockBuilder is an implementation of Builder that can be used for tests.
// You can set some fake return values and you can keep track of what // You can set some fake return values and you can keep track of what
// methods were called on the builder. It is fairly basic. // methods were called on the builder. It is fairly basic.
type MockBuilder struct { type MockBuilder struct {
ArtifactId string ArtifactId string
PrepareWarnings []string PrepareWarnings []string
RunErrResult bool
RunNilResult bool
PrepareCalled bool PrepareCalled bool
PrepareConfig []interface{} PrepareConfig []interface{}
@ -27,6 +33,15 @@ func (tb *MockBuilder) Run(ui Ui, h Hook, c Cache) (Artifact, error) {
tb.RunHook = h tb.RunHook = h
tb.RunUi = ui tb.RunUi = ui
tb.RunCache = c tb.RunCache = c
if tb.RunErrResult {
return nil, errors.New("foo")
}
if tb.RunNilResult {
return nil, nil
}
return &MockArtifact{ return &MockArtifact{
IdValue: tb.ArtifactId, IdValue: tb.ArtifactId,
}, nil }, nil

View File

@ -1,7 +1,6 @@
package rpc package rpc
import ( import (
"errors"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
"net/rpc" "net/rpc"
"reflect" "reflect"
@ -10,47 +9,8 @@ import (
var testBuilderArtifact = &testArtifact{} var testBuilderArtifact = &testArtifact{}
type testBuilder struct { func builderRPCClient(t *testing.T) (*packer.MockBuilder, packer.Builder) {
prepareCalled bool b := new(packer.MockBuilder)
prepareConfig []interface{}
runCalled bool
runCache packer.Cache
runHook packer.Hook
runUi packer.Ui
cancelCalled bool
errRunResult bool
nilRunResult bool
}
func (b *testBuilder) Prepare(config ...interface{}) ([]string, error) {
b.prepareCalled = true
b.prepareConfig = config
return nil, nil
}
func (b *testBuilder) Run(ui packer.Ui, hook packer.Hook, cache packer.Cache) (packer.Artifact, error) {
b.runCache = cache
b.runCalled = true
b.runHook = hook
b.runUi = ui
if b.errRunResult {
return nil, errors.New("foo")
} else if b.nilRunResult {
return nil, nil
} else {
return testBuilderArtifact, nil
}
}
func (b *testBuilder) Cancel() {
b.cancelCalled = true
}
func TestBuilderRPC(t *testing.T) {
// Create the interface to test
b := new(testBuilder)
// Start the server // Start the server
server := rpc.NewServer() server := rpc.NewServer()
@ -62,18 +22,26 @@ func TestBuilderRPC(t *testing.T) {
if err != nil { if err != nil {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
return b, Builder(client)
}
func TestBuilderPrepare(t *testing.T) {
b, bClient := builderRPCClient(t)
// Test Prepare // Test Prepare
config := 42 config := 42
bClient := Builder(client)
bClient.Prepare(config) bClient.Prepare(config)
if !b.prepareCalled { if !b.PrepareCalled {
t.Fatal("should be called") t.Fatal("should be called")
} }
if !reflect.DeepEqual(b.prepareConfig, []interface{}{42}) { if !reflect.DeepEqual(b.PrepareConfig, []interface{}{42}) {
t.Fatalf("bad: %#v", b.prepareConfig) t.Fatalf("bad: %#v", b.PrepareConfig)
} }
}
func TestBuilderRun(t *testing.T) {
b, bClient := builderRPCClient(t)
// Test Run // Test Run
cache := new(testCache) cache := new(testCache)
@ -84,22 +52,21 @@ func TestBuilderRPC(t *testing.T) {
t.Fatalf("err: %s", err) t.Fatalf("err: %s", err)
} }
if !b.runCalled { if !b.RunCalled {
t.Fatal("run should be called") t.Fatal("run should be called")
} }
if b.runCalled { b.RunCache.Lock("foo")
b.runCache.Lock("foo")
if !cache.lockCalled { if !cache.lockCalled {
t.Fatal("should be called") t.Fatal("should be called")
} }
b.runHook.Run("foo", nil, nil, nil) b.RunHook.Run("foo", nil, nil, nil)
if !hook.RunCalled { if !hook.RunCalled {
t.Fatal("should be called") t.Fatal("should be called")
} }
b.runUi.Say("format") b.RunUi.Say("format")
if !ui.sayCalled { if !ui.sayCalled {
t.Fatal("say should be called") t.Fatal("say should be called")
} }
@ -113,30 +80,43 @@ func TestBuilderRPC(t *testing.T) {
} }
} }
// Test run with nil result func TestBuilderRun_nilResult(t *testing.T) {
b.nilRunResult = true b, bClient := builderRPCClient(t)
artifact, err = bClient.Run(ui, hook, cache) b.RunNilResult = true
cache := new(testCache)
hook := &packer.MockHook{}
ui := &testUi{}
artifact, err := bClient.Run(ui, hook, cache)
if artifact != nil { if artifact != nil {
t.Fatalf("bad: %#v", artifact) t.Fatalf("bad: %#v", artifact)
} }
if err != nil { if err != nil {
t.Fatalf("bad: %#v", err) t.Fatalf("bad: %#v", err)
} }
}
// Test with an error func TestBuilderRun_ErrResult(t *testing.T) {
b.errRunResult = true b, bClient := builderRPCClient(t)
b.nilRunResult = false b.RunErrResult = true
artifact, err = bClient.Run(ui, hook, cache)
cache := new(testCache)
hook := &packer.MockHook{}
ui := &testUi{}
artifact, err := bClient.Run(ui, hook, cache)
if artifact != nil { if artifact != nil {
t.Fatalf("bad: %#v", artifact) t.Fatalf("bad: %#v", artifact)
} }
if err == nil { if err == nil {
t.Fatal("should have error") t.Fatal("should have error")
} }
}
func TestBuilderCancel(t *testing.T) {
b, bClient := builderRPCClient(t)
// Test Cancel
bClient.Cancel() bClient.Cancel()
if !b.cancelCalled { if !b.CancelCalled {
t.Fatal("cancel should be called") t.Fatal("cancel should be called")
} }
} }

View File

@ -7,7 +7,7 @@ import (
"testing" "testing"
) )
var testEnvBuilder = &testBuilder{} var testEnvBuilder = &packer.MockBuilder{}
var testEnvCache = &testCache{} var testEnvCache = &testCache{}
var testEnvUi = &testUi{} var testEnvUi = &testUi{}
@ -90,7 +90,7 @@ func TestEnvironmentRPC(t *testing.T) {
} }
builder.Prepare(nil) builder.Prepare(nil)
if !testEnvBuilder.prepareCalled { if !testEnvBuilder.PrepareCalled {
t.Fatal("should be called") t.Fatal("should be called")
} }