package packer import ( "context" "errors" "fmt" "testing" "time" packersdk "github.com/hashicorp/packer/packer-plugin-sdk/packer" ) func TestProvisionHook_Impl(t *testing.T) { var raw interface{} raw = &ProvisionHook{} if _, ok := raw.(packersdk.Hook); !ok { t.Fatalf("must be a Hook") } } func TestProvisionHook(t *testing.T) { pA := &packersdk.MockProvisioner{} pB := &packersdk.MockProvisioner{} ui := testUi() var comm packersdk.Communicator = new(packersdk.MockCommunicator) var data interface{} = nil hook := &ProvisionHook{ Provisioners: []*HookedProvisioner{ {pA, nil, ""}, {pB, nil, ""}, }, } hook.Run(context.Background(), "foo", ui, comm, data) if !pA.ProvCalled { t.Error("provision should be called on pA") } if !pB.ProvCalled { t.Error("provision should be called on pB") } } func TestProvisionHook_nilComm(t *testing.T) { pA := &packersdk.MockProvisioner{} pB := &packersdk.MockProvisioner{} ui := testUi() var comm packersdk.Communicator = nil var data interface{} = nil hook := &ProvisionHook{ Provisioners: []*HookedProvisioner{ {pA, nil, ""}, {pB, nil, ""}, }, } err := hook.Run(context.Background(), "foo", ui, comm, data) if err == nil { t.Fatal("should error") } } func TestProvisionHook_cancel(t *testing.T) { topCtx, topCtxCancel := context.WithCancel(context.Background()) p := &packersdk.MockProvisioner{ ProvFunc: func(ctx context.Context) error { topCtxCancel() <-ctx.Done() return ctx.Err() }, } hook := &ProvisionHook{ Provisioners: []*HookedProvisioner{ {p, nil, ""}, }, } err := hook.Run(topCtx, "foo", nil, new(packersdk.MockCommunicator), nil) if err == nil { t.Fatal("should have err") } } // TODO(mitchellh): Test that they're run in the proper order func TestPausedProvisioner_impl(t *testing.T) { var _ packersdk.Provisioner = new(PausedProvisioner) } func TestPausedProvisionerPrepare(t *testing.T) { mock := new(packersdk.MockProvisioner) prov := &PausedProvisioner{ Provisioner: mock, } prov.Prepare(42) if !mock.PrepCalled { t.Fatal("prepare should be called") } if mock.PrepConfigs[0] != 42 { t.Fatal("should have proper configs") } } func TestPausedProvisionerProvision(t *testing.T) { mock := new(packersdk.MockProvisioner) prov := &PausedProvisioner{ Provisioner: mock, } ui := testUi() comm := new(packersdk.MockCommunicator) prov.Provision(context.Background(), ui, comm, make(map[string]interface{})) if !mock.ProvCalled { t.Fatal("prov should be called") } if mock.ProvUi != ui { t.Fatal("should have proper ui") } if mock.ProvCommunicator != comm { t.Fatal("should have proper comm") } } func TestPausedProvisionerProvision_waits(t *testing.T) { startTime := time.Now() waitTime := 50 * time.Millisecond prov := &PausedProvisioner{ PauseBefore: waitTime, Provisioner: &packersdk.MockProvisioner{ ProvFunc: func(context.Context) error { timeSinceStartTime := time.Since(startTime) if timeSinceStartTime < waitTime { return fmt.Errorf("Spent not enough time waiting: %s", timeSinceStartTime) } return nil }, }, } err := prov.Provision(context.Background(), testUi(), new(packersdk.MockCommunicator), make(map[string]interface{})) if err != nil { t.Fatalf("prov failed: %v", err) } } func TestPausedProvisionerCancel(t *testing.T) { topCtx, cancelTopCtx := context.WithCancel(context.Background()) mock := new(packersdk.MockProvisioner) prov := &PausedProvisioner{ Provisioner: mock, } mock.ProvFunc = func(ctx context.Context) error { cancelTopCtx() <-ctx.Done() return ctx.Err() } err := prov.Provision(topCtx, testUi(), new(packersdk.MockCommunicator), make(map[string]interface{})) if err == nil { t.Fatal("should have err") } } func TestDebuggedProvisioner_impl(t *testing.T) { var _ packersdk.Provisioner = new(DebuggedProvisioner) } func TestDebuggedProvisionerPrepare(t *testing.T) { mock := new(packersdk.MockProvisioner) prov := &DebuggedProvisioner{ Provisioner: mock, } prov.Prepare(42) if !mock.PrepCalled { t.Fatal("prepare should be called") } if mock.PrepConfigs[0] != 42 { t.Fatal("should have proper configs") } } func TestDebuggedProvisionerProvision(t *testing.T) { mock := new(packersdk.MockProvisioner) prov := &DebuggedProvisioner{ Provisioner: mock, } ui := testUi() comm := new(packersdk.MockCommunicator) writeReader(ui, "\n") prov.Provision(context.Background(), ui, comm, make(map[string]interface{})) if !mock.ProvCalled { t.Fatal("prov should be called") } if mock.ProvUi != ui { t.Fatal("should have proper ui") } if mock.ProvCommunicator != comm { t.Fatal("should have proper comm") } } func TestDebuggedProvisionerCancel(t *testing.T) { topCtx, topCtxCancel := context.WithCancel(context.Background()) mock := new(packersdk.MockProvisioner) prov := &DebuggedProvisioner{ Provisioner: mock, } mock.ProvFunc = func(ctx context.Context) error { topCtxCancel() <-ctx.Done() return ctx.Err() } err := prov.Provision(topCtx, testUi(), new(packersdk.MockCommunicator), make(map[string]interface{})) if err == nil { t.Fatal("should have error") } } func TestRetriedProvisioner_impl(t *testing.T) { var _ packersdk.Provisioner = new(RetriedProvisioner) } func TestRetriedProvisionerPrepare(t *testing.T) { mock := new(packersdk.MockProvisioner) prov := &RetriedProvisioner{ Provisioner: mock, } err := prov.Prepare(42) if err != nil { t.Fatal("should not have errored") } if !mock.PrepCalled { t.Fatal("prepare should be called") } if mock.PrepConfigs[0] != 42 { t.Fatal("should have proper configs") } } func TestRetriedProvisionerProvision(t *testing.T) { mock := &packersdk.MockProvisioner{ ProvFunc: func(ctx context.Context) error { return errors.New("failed") }, } prov := &RetriedProvisioner{ MaxRetries: 2, Provisioner: mock, } ui := testUi() comm := new(packersdk.MockCommunicator) err := prov.Provision(context.Background(), ui, comm, make(map[string]interface{})) if err != nil { t.Fatal("should not have errored") } if !mock.ProvCalled { t.Fatal("prov should be called") } if !mock.ProvRetried { t.Fatal("prov should be retried") } if mock.ProvUi != ui { t.Fatal("should have proper ui") } if mock.ProvCommunicator != comm { t.Fatal("should have proper comm") } } func TestRetriedProvisionerCancelledProvision(t *testing.T) { // Don't retry if context is cancelled ctx, topCtxCancel := context.WithCancel(context.Background()) mock := &packersdk.MockProvisioner{ ProvFunc: func(ctx context.Context) error { topCtxCancel() <-ctx.Done() return ctx.Err() }, } prov := &RetriedProvisioner{ MaxRetries: 2, Provisioner: mock, } ui := testUi() comm := new(packersdk.MockCommunicator) err := prov.Provision(ctx, ui, comm, make(map[string]interface{})) if err == nil { t.Fatal("should have errored") } if !mock.ProvCalled { t.Fatal("prov should be called") } if mock.ProvRetried { t.Fatal("prov should NOT be retried") } if mock.ProvUi != ui { t.Fatal("should have proper ui") } if mock.ProvCommunicator != comm { t.Fatal("should have proper comm") } } func TestRetriedProvisionerCancel(t *testing.T) { topCtx, cancelTopCtx := context.WithCancel(context.Background()) mock := new(packersdk.MockProvisioner) prov := &RetriedProvisioner{ Provisioner: mock, } mock.ProvFunc = func(ctx context.Context) error { cancelTopCtx() <-ctx.Done() return ctx.Err() } err := prov.Provision(topCtx, testUi(), new(packersdk.MockCommunicator), make(map[string]interface{})) if err == nil { t.Fatal("should have err") } }