mirror of https://github.com/pulumi/pulumi.git
99 lines
3.8 KiB
Go
99 lines
3.8 KiB
Go
// Copyright 2016-2018, Pulumi Corporation. All rights reserved.
|
|
//go:build (nodejs || all) && !xplatform_acceptance
|
|
|
|
package ints
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/providers"
|
|
"github.com/pulumi/pulumi/pkg/v3/testing/integration"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
// TestPartialState tests that the engine persists partial state of a resource if a provider
|
|
// provides partial state alongside a resource creation or update error.
|
|
//
|
|
// The setup of this test uses a dynamic provider that will partially fail if a resource's state
|
|
// value is the number 4.
|
|
//
|
|
//nolint:paralleltest // ProgramTest calls t.Parallel()
|
|
func TestPartialState(t *testing.T) {
|
|
integration.ProgramTest(t, &integration.ProgramTestOptions{
|
|
Dir: "step1",
|
|
Dependencies: []string{"@pulumi/pulumi"},
|
|
Quick: true,
|
|
ExpectFailure: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stackInfo integration.RuntimeValidationStackInfo) {
|
|
// The first update tries to create a resource with state 4. This fails partially.
|
|
assert.NotNil(t, stackInfo.Deployment)
|
|
assert.Equal(t, 3, len(stackInfo.Deployment.Resources))
|
|
stackRes := stackInfo.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
providerRes := stackInfo.Deployment.Resources[1]
|
|
assert.True(t, providers.IsProviderType(providerRes.URN.Type()))
|
|
|
|
a := stackInfo.Deployment.Resources[2]
|
|
|
|
// We should still have persisted the resource and its outputs to the snapshot
|
|
assert.Equal(t, "doomed", a.URN.Name())
|
|
assert.Equal(t, 4.0, a.Outputs["state"].(float64))
|
|
assert.Equal(t, []string{"state can't be 4"}, a.InitErrors)
|
|
},
|
|
EditDirs: []integration.EditDir{
|
|
{
|
|
Dir: "step2",
|
|
Additive: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stackInfo integration.RuntimeValidationStackInfo) {
|
|
// The next update deletes the resource. We should successfully delete it.
|
|
assert.NotNil(t, stackInfo.Deployment)
|
|
assert.Equal(t, 1, len(stackInfo.Deployment.Resources))
|
|
stackRes := stackInfo.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
},
|
|
},
|
|
{
|
|
Dir: "step3",
|
|
Additive: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stackInfo integration.RuntimeValidationStackInfo) {
|
|
// Step 3 creates a resource with state 5, which succeeds.
|
|
assert.NotNil(t, stackInfo.Deployment)
|
|
assert.Equal(t, 3, len(stackInfo.Deployment.Resources))
|
|
stackRes := stackInfo.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
providerRes := stackInfo.Deployment.Resources[1]
|
|
assert.True(t, providers.IsProviderType(providerRes.URN.Type()))
|
|
|
|
a := stackInfo.Deployment.Resources[2]
|
|
assert.Equal(t, "not-doomed", a.URN.Name())
|
|
assert.Equal(t, 5.0, a.Outputs["state"].(float64))
|
|
assert.Nil(t, nil)
|
|
},
|
|
},
|
|
{
|
|
Dir: "step4",
|
|
Additive: true,
|
|
ExpectFailure: true,
|
|
ExtraRuntimeValidation: func(t *testing.T, stackInfo integration.RuntimeValidationStackInfo) {
|
|
// Step 4 updates the resource to have state 4, which fails partially.
|
|
assert.NotNil(t, stackInfo.Deployment)
|
|
assert.Equal(t, 3, len(stackInfo.Deployment.Resources))
|
|
stackRes := stackInfo.Deployment.Resources[0]
|
|
assert.Equal(t, resource.RootStackType, stackRes.URN.Type())
|
|
providerRes := stackInfo.Deployment.Resources[1]
|
|
assert.True(t, providers.IsProviderType(providerRes.URN.Type()))
|
|
|
|
a := stackInfo.Deployment.Resources[2]
|
|
|
|
// We should have persisted the updated resource's new outputs
|
|
// to the snapshot.
|
|
assert.Equal(t, "not-doomed", a.URN.Name())
|
|
assert.Equal(t, 4.0, a.Outputs["state"].(float64))
|
|
assert.Equal(t, []string{"state can't be 4"}, a.InitErrors)
|
|
},
|
|
},
|
|
},
|
|
})
|
|
}
|