mirror of https://github.com/pulumi/pulumi.git
642 lines
17 KiB
Go
642 lines
17 KiB
Go
// Copyright 2024, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package fuzzing
|
|
|
|
import (
|
|
"fmt"
|
|
"slices"
|
|
"strings"
|
|
"time"
|
|
|
|
lt "github.com/pulumi/pulumi/pkg/v3/engine/lifecycletest/framework"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/providers"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/stretchr/testify/require"
|
|
"golang.org/x/exp/maps"
|
|
)
|
|
|
|
// GenerateReproTest generates a string containing Go code for a lifecycle test that reproduces the scenario captured by
|
|
// the given *Specs.
|
|
func GenerateReproTest(
|
|
t lt.TB,
|
|
sso StackSpecOptions,
|
|
snapSpec *SnapshotSpec,
|
|
progSpec *ProgramSpec,
|
|
provSpec *ProviderSpec,
|
|
planSpec *PlanSpec,
|
|
) string {
|
|
var b strings.Builder
|
|
|
|
writeHeader(&b)
|
|
writePackageImports(&b)
|
|
|
|
g := &generator{b: &b}
|
|
writeTestFunction(t, g, sso, snapSpec, progSpec, provSpec, planSpec)
|
|
|
|
return b.String()
|
|
}
|
|
|
|
// writeHeader writes the standard Pulumi license header to the given strings.Builder.
|
|
func writeHeader(b *strings.Builder) {
|
|
year := time.Now().Year()
|
|
|
|
b.WriteString(fmt.Sprintf(`// Copyright %d, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
`, year))
|
|
}
|
|
|
|
// writePackageImports writes a superset of the imports that we'll need for a generated lifecycle test.
|
|
func writePackageImports(b *strings.Builder) {
|
|
b.WriteString(`package lifecycletest
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/blang/semver"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/engine"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/deploytest"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/providers"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
|
|
|
|
lt "github.com/pulumi/pulumi/pkg/v3/engine/lifecycletest/framework"
|
|
)
|
|
`)
|
|
}
|
|
|
|
// generator encapsulates writing indented Go code to a strings.Builder.
|
|
type generator struct {
|
|
// The underlying strings.Builder to write to.
|
|
b *strings.Builder
|
|
// The current indentation string (which will typically be a possibly empty series of tabs).
|
|
prefix string
|
|
}
|
|
|
|
// indent increases the indentation level of the generator.
|
|
func (g *generator) indent() {
|
|
g.prefix += "\t"
|
|
}
|
|
|
|
// dedent decreases the indentation level of the generator.
|
|
func (g *generator) dedent() {
|
|
g.prefix = g.prefix[:len(g.prefix)-1]
|
|
}
|
|
|
|
// writeLine writes a newline-prefixed line of Go code to the generator's strings.Builder, prefixed by the current
|
|
// indentation level.
|
|
func (g *generator) writeLine(s string) {
|
|
g.b.WriteString("\n" + g.prefix + s)
|
|
}
|
|
|
|
// writeBlock writes a newline-prefixed block of Go code to the generator's strings.Builder. The block's prefix and
|
|
// suffix will be indented at the current level, while the block's contents will be indented one level deeper. For
|
|
// example, the call:
|
|
//
|
|
// writeBlock(
|
|
// "func foo() {",
|
|
// func(g *generator) {
|
|
// g.writeLine("bar := 42")
|
|
// },
|
|
// "}",
|
|
// )
|
|
//
|
|
// will yield:
|
|
//
|
|
// func foo() {
|
|
// bar := 42
|
|
// }
|
|
func (g *generator) writeBlock(
|
|
prefix string,
|
|
block func(g *generator),
|
|
suffix string,
|
|
) {
|
|
g.writeLine(prefix)
|
|
g.indent()
|
|
block(g)
|
|
g.dedent()
|
|
g.writeLine(suffix)
|
|
}
|
|
|
|
// writeTestFunction writes a Go test function that reproduces the scenario captured by the given *Specs.
|
|
func writeTestFunction(
|
|
t require.TestingT,
|
|
g *generator,
|
|
sso StackSpecOptions,
|
|
snapSpec *SnapshotSpec,
|
|
progSpec *ProgramSpec,
|
|
provSpec *ProviderSpec,
|
|
planSpec *PlanSpec,
|
|
) {
|
|
g.writeBlock(
|
|
"func TestRepro(t *testing.T) {",
|
|
func(g *generator) {
|
|
g.writeLine("t.Parallel()")
|
|
|
|
g.writeLine("")
|
|
g.writeBlock(
|
|
"p := <.TestPlan{",
|
|
func(g *generator) {
|
|
g.writeLine(fmt.Sprintf(`Project: "%s",`, sso.Project))
|
|
g.writeLine(fmt.Sprintf(`Stack: "%s",`, sso.Stack))
|
|
},
|
|
"}",
|
|
)
|
|
g.writeLine("project := p.GetProject()")
|
|
|
|
g.writeLine("")
|
|
g.writeLine("// Set up the initial snapshot.")
|
|
g.writeBlock(
|
|
"setupLoaders := []*deploytest.ProviderLoader{",
|
|
writeSetupLoaderElements(provSpec),
|
|
"}",
|
|
)
|
|
|
|
g.writeLine("")
|
|
g.writeBlock(
|
|
"setupProgramF := deploytest.NewLanguageRuntimeF("+
|
|
"func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {",
|
|
writeResourceRegistrationStatements(t, snapSpec.Resources),
|
|
"})",
|
|
)
|
|
|
|
g.writeLine("")
|
|
g.writeLine("setupHostF := deploytest.NewPluginHostF(nil, nil, setupProgramF, setupLoaders...)")
|
|
g.writeBlock(
|
|
"setupOpts := lt.TestUpdateOptions{",
|
|
func(g *generator) {
|
|
g.writeLine("T: t,")
|
|
g.writeLine("HostF: setupHostF,")
|
|
},
|
|
"}",
|
|
)
|
|
|
|
g.writeLine("setupSnap, err := lt.TestOp(engine.Update)." +
|
|
"RunStep(project, p.GetTarget(t, nil), setupOpts, false, p.BackendClient, nil, \"0\")")
|
|
g.writeLine("require.NoError(t, err)")
|
|
|
|
g.writeLine("")
|
|
g.writeLine("// Set up the reproduction providers and program.")
|
|
g.writeBlock(
|
|
"createF := func(ctx context.Context, req plugin.CreateRequest) (plugin.CreateResponse, error) {",
|
|
writeCreateFStatements(provSpec),
|
|
"}",
|
|
)
|
|
g.writeBlock(
|
|
"deleteF := func(ctx context.Context, req plugin.DeleteRequest) (plugin.DeleteResponse, error) {",
|
|
writeDeleteFStatements(provSpec),
|
|
"}",
|
|
)
|
|
g.writeBlock(
|
|
"diffF := func(ctx context.Context, req plugin.DiffRequest) (plugin.DiffResponse, error) {",
|
|
writeDiffFStatements(provSpec),
|
|
"}",
|
|
)
|
|
g.writeBlock(
|
|
"readF := func(ctx context.Context, req plugin.ReadRequest) (plugin.ReadResponse, error) {",
|
|
writeReadFStatements(provSpec),
|
|
"}",
|
|
)
|
|
g.writeBlock(
|
|
"updateF := func(ctx context.Context, req plugin.UpdateRequest) (plugin.UpdateResponse, error) {",
|
|
writeUpdateFStatements(provSpec),
|
|
"}",
|
|
)
|
|
|
|
g.writeLine("")
|
|
g.writeBlock(
|
|
"reproLoaders := []*deploytest.ProviderLoader{",
|
|
writeReproLoaderElements(provSpec),
|
|
"}",
|
|
)
|
|
|
|
g.writeLine("")
|
|
g.writeBlock(
|
|
"reproProgramF := deploytest.NewLanguageRuntimeF("+
|
|
"func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {",
|
|
writeResourceRegistrationStatements(t, progSpec.ResourceRegistrations),
|
|
"})",
|
|
)
|
|
|
|
g.writeLine("")
|
|
g.writeLine("reproHostF := deploytest.NewPluginHostF(nil, nil, reproProgramF, reproLoaders...)")
|
|
g.writeBlock(
|
|
"reproOpts := lt.TestUpdateOptions{",
|
|
func(g *generator) {
|
|
g.writeLine("T: t,")
|
|
g.writeLine("HostF: reproHostF,")
|
|
g.writeBlock(
|
|
"UpdateOptions: engine.UpdateOptions{",
|
|
func(g *generator) {
|
|
if len(planSpec.TargetURNs) > 0 {
|
|
g.writeBlock(
|
|
"Targets: deploy.NewUrnTargets([]string{",
|
|
func(g *generator) {
|
|
for _, urn := range planSpec.TargetURNs {
|
|
g.writeLine(fmt.Sprintf(`"%s",`, urn))
|
|
}
|
|
},
|
|
"}),",
|
|
)
|
|
}
|
|
},
|
|
"},",
|
|
)
|
|
},
|
|
"}",
|
|
)
|
|
|
|
var operation string
|
|
switch planSpec.Operation {
|
|
case PlanOperationUpdate:
|
|
operation = "engine.Update"
|
|
case PlanOperationRefresh:
|
|
operation = "engine.Refresh"
|
|
case PlanOperationDestroy:
|
|
operation = "engine.Destroy"
|
|
}
|
|
|
|
g.writeLine("")
|
|
g.writeLine("// Trigger the reproduction.")
|
|
g.writeLine(fmt.Sprintf(
|
|
"reproSnap, err := "+
|
|
"lt.TestOp(%s).RunStep(project, p.GetTarget(t, setupSnap), reproOpts, false, p.BackendClient, nil, \"1\")",
|
|
operation,
|
|
))
|
|
g.writeLine("require.NoError(t, err)")
|
|
},
|
|
"}",
|
|
)
|
|
}
|
|
|
|
func writeSetupLoaderElements(provSpec *ProviderSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
pkgs := maps.Keys(provSpec.Packages)
|
|
slices.Sort(pkgs)
|
|
|
|
for _, pkg := range pkgs {
|
|
g.writeBlock(
|
|
fmt.Sprintf(
|
|
"deploytest.NewProviderLoader(\"%s\", semver.MustParse(\"1.0.0\"), func() (plugin.Provider, error) {",
|
|
pkg,
|
|
),
|
|
func(g *generator) {
|
|
g.writeLine("return &deploytest.Provider{}, nil")
|
|
},
|
|
"}),",
|
|
)
|
|
}
|
|
}
|
|
}
|
|
|
|
func writeResourceRegistrationStatements(t require.TestingT, rs []*ResourceSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
indicesByURN := map[resource.URN]int{}
|
|
varFor := func(urn resource.URN) string {
|
|
if i, has := indicesByURN[urn]; has {
|
|
return fmt.Sprintf("res%d", i)
|
|
}
|
|
|
|
return "resUnknown"
|
|
}
|
|
|
|
for i, r := range rs {
|
|
indicesByURN[r.URN()] = i
|
|
|
|
if r.Provider != "" {
|
|
ref, err := providers.ParseReference(r.Provider)
|
|
require.NoError(t, err)
|
|
|
|
g.writeLine(fmt.Sprintf(
|
|
"res%dProvRef, err := providers.NewReference(%[2]s.URN, %[2]s.ID)",
|
|
i, varFor(ref.URN()),
|
|
))
|
|
g.writeLine("require.NoError(t, err)")
|
|
}
|
|
|
|
g.writeBlock(
|
|
fmt.Sprintf(
|
|
"res%d, err := monitor.RegisterResource(\"%s\", \"%s\", %v, deploytest.ResourceOptions{",
|
|
i, r.Type, r.Name, r.Custom,
|
|
),
|
|
func(g *generator) {
|
|
if r.Delete {
|
|
g.writeLine("// You'll need to set up a means for Delete: true to be set on this resource")
|
|
g.writeLine("// Delete: true,")
|
|
}
|
|
if r.PendingReplacement {
|
|
g.writeLine("// You'll need to set up a means for PendingReplacement: true to be set on this resource")
|
|
g.writeLine("// PendingReplacement: true,")
|
|
}
|
|
|
|
if r.Protect {
|
|
g.writeLine("Protect: true,")
|
|
}
|
|
if r.RetainOnDelete {
|
|
g.writeLine("RetainOnDelete: true,")
|
|
}
|
|
|
|
if r.Provider != "" {
|
|
g.writeLine(fmt.Sprintf("Provider: res%dProvRef.String(),", i))
|
|
}
|
|
|
|
if r.Parent != "" {
|
|
g.writeLine(fmt.Sprintf("Parent: %s.URN,", varFor(r.Parent)))
|
|
}
|
|
|
|
if len(r.Dependencies) > 0 {
|
|
g.writeBlock(
|
|
"Dependencies: []resource.URN{",
|
|
func(g *generator) {
|
|
for _, dep := range r.Dependencies {
|
|
g.writeLine(varFor(dep) + ".URN,")
|
|
}
|
|
},
|
|
"},",
|
|
)
|
|
}
|
|
|
|
if len(r.PropertyDependencies) > 0 {
|
|
g.writeBlock(
|
|
"PropertyDeps: map[resource.PropertyKey][]resource.URN{",
|
|
func(g *generator) {
|
|
for k, deps := range r.PropertyDependencies {
|
|
g.writeBlock(
|
|
fmt.Sprintf("\"%s\": {", k),
|
|
func(g *generator) {
|
|
for _, dep := range deps {
|
|
g.writeLine(varFor(dep) + ".URN,")
|
|
}
|
|
},
|
|
"},",
|
|
)
|
|
}
|
|
},
|
|
"},",
|
|
)
|
|
}
|
|
|
|
if r.DeletedWith != "" {
|
|
g.writeLine(fmt.Sprintf("DeletedWith: %s.URN,", varFor(r.DeletedWith)))
|
|
}
|
|
|
|
if len(r.Aliases) > 0 {
|
|
g.writeBlock(
|
|
"AliasURNs: []resource.URN{",
|
|
func(g *generator) {
|
|
for _, alias := range r.Aliases {
|
|
g.writeLine(fmt.Sprintf("\"%s\",", alias))
|
|
}
|
|
},
|
|
"},",
|
|
)
|
|
}
|
|
},
|
|
"})",
|
|
)
|
|
g.writeLine("require.NoError(t, err)")
|
|
g.writeLine("")
|
|
}
|
|
|
|
g.writeLine("return nil")
|
|
}
|
|
}
|
|
|
|
func writeCreateFStatements(provSpec *ProviderSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
if len(provSpec.Create) > 0 {
|
|
g.writeLine("switch req.URN {")
|
|
for urn := range provSpec.Create {
|
|
g.writeBlock(
|
|
fmt.Sprintf("case \"%s\":", urn),
|
|
func(g *generator) {
|
|
g.writeBlock(
|
|
"return plugin.CreateResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Status: resource.StatusUnknown,")
|
|
},
|
|
"}, fmt.Errorf(\"create failure for %s\", req.URN)",
|
|
)
|
|
},
|
|
"",
|
|
)
|
|
}
|
|
g.writeLine("}")
|
|
}
|
|
|
|
g.writeBlock(
|
|
"return plugin.CreateResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Properties: req.Properties,")
|
|
g.writeLine("Status: resource.StatusOK,")
|
|
},
|
|
"}, nil",
|
|
)
|
|
}
|
|
}
|
|
|
|
func writeDeleteFStatements(provSpec *ProviderSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
if len(provSpec.Delete) > 0 {
|
|
g.writeLine("switch req.URN {")
|
|
for urn := range provSpec.Delete {
|
|
g.writeBlock(
|
|
fmt.Sprintf("case \"%s\":", urn),
|
|
func(g *generator) {
|
|
g.writeBlock(
|
|
"return plugin.DeleteResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Status: resource.StatusUnknown,")
|
|
},
|
|
"}, fmt.Errorf(\"delete failure for %s\", req.URN)",
|
|
)
|
|
},
|
|
"",
|
|
)
|
|
}
|
|
g.writeLine("}")
|
|
}
|
|
|
|
g.writeBlock(
|
|
"return plugin.DeleteResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Status: resource.StatusOK,")
|
|
},
|
|
"}, nil",
|
|
)
|
|
}
|
|
}
|
|
|
|
func writeDiffFStatements(provSpec *ProviderSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
if len(provSpec.Diff) > 0 {
|
|
g.writeLine("switch req.URN {")
|
|
for urn, action := range provSpec.Diff {
|
|
g.writeBlock(
|
|
fmt.Sprintf("case \"%s\":", urn),
|
|
func(g *generator) {
|
|
switch action {
|
|
case ProviderDiffDeleteBeforeReplace:
|
|
g.writeBlock(
|
|
"return plugin.DiffResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Changes: plugin.DiffSome,")
|
|
g.writeLine("ReplaceKeys: []resource.PropertyKey{\"__replace\"},")
|
|
g.writeLine("DeleteBeforeReplace: true,")
|
|
},
|
|
"}, nil",
|
|
)
|
|
case ProviderDiffDeleteAfterReplace:
|
|
g.writeBlock(
|
|
"return plugin.DiffResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Changes: plugin.DiffSome,")
|
|
g.writeLine("ReplaceKeys: []resource.PropertyKey{\"__replace\"},")
|
|
g.writeLine("DeleteBeforeReplace: false,")
|
|
},
|
|
"}, nil",
|
|
)
|
|
case ProviderDiffChange:
|
|
g.writeLine("return plugin.DiffResponse{Changes: plugin.DiffSome}, nil")
|
|
case ProviderDiffFailure:
|
|
g.writeLine("return plugin.DiffResponse{}, fmt.Errorf(\"diff failure for %s\", req.URN)")
|
|
}
|
|
},
|
|
"",
|
|
)
|
|
}
|
|
g.writeLine("}")
|
|
}
|
|
|
|
g.writeLine("return plugin.DiffResponse{}, nil")
|
|
}
|
|
}
|
|
|
|
func writeReadFStatements(provSpec *ProviderSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
if len(provSpec.Read) > 0 {
|
|
g.writeLine("switch req.URN {")
|
|
for urn, action := range provSpec.Read {
|
|
g.writeBlock(
|
|
fmt.Sprintf("case \"%s\":", urn),
|
|
func(g *generator) {
|
|
switch action {
|
|
case ProviderReadDeleted:
|
|
g.writeLine("return plugin.ReadResponse{}, nil")
|
|
case ProviderReadFailure:
|
|
g.writeBlock(
|
|
"return plugin.ReadResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Status: resource.StatusUnknown,")
|
|
},
|
|
"}, fmt.Errorf(\"read failure for %s\", req.URN)",
|
|
)
|
|
}
|
|
},
|
|
"",
|
|
)
|
|
}
|
|
g.writeLine("}")
|
|
}
|
|
|
|
g.writeBlock(
|
|
"return plugin.ReadResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("ReadResult: plugin.ReadResult{Outputs: resource.PropertyMap{}},")
|
|
g.writeLine("Status: resource.StatusOK,")
|
|
},
|
|
"}, nil",
|
|
)
|
|
}
|
|
}
|
|
|
|
func writeUpdateFStatements(provSpec *ProviderSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
if len(provSpec.Update) > 0 {
|
|
g.writeLine("switch req.URN {")
|
|
for urn := range provSpec.Update {
|
|
g.writeBlock(
|
|
fmt.Sprintf("case \"%s\":", urn),
|
|
func(g *generator) {
|
|
g.writeBlock(
|
|
"return plugin.UpdateResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Status: resource.StatusUnknown,")
|
|
},
|
|
"}, fmt.Errorf(\"update failure for %s\", req.URN)",
|
|
)
|
|
},
|
|
"",
|
|
)
|
|
}
|
|
g.writeLine("}")
|
|
}
|
|
|
|
g.writeBlock(
|
|
"return plugin.UpdateResponse{",
|
|
func(g *generator) {
|
|
g.writeLine("Properties: req.NewInputs,")
|
|
g.writeLine("Status: resource.StatusOK,")
|
|
},
|
|
"}, nil",
|
|
)
|
|
}
|
|
}
|
|
|
|
func writeReproLoaderElements(provSpec *ProviderSpec) func(g *generator) {
|
|
return func(g *generator) {
|
|
pkgs := maps.Keys(provSpec.Packages)
|
|
slices.Sort(pkgs)
|
|
|
|
for _, pkg := range pkgs {
|
|
g.writeBlock(
|
|
fmt.Sprintf(
|
|
"deploytest.NewProviderLoader(\"%s\", semver.MustParse(\"1.0.0\"), func() (plugin.Provider, error) {",
|
|
pkg,
|
|
),
|
|
func(g *generator) {
|
|
g.writeBlock(
|
|
"return &deploytest.Provider{",
|
|
func(g *generator) {
|
|
g.writeLine("CreateF: createF,")
|
|
g.writeLine("DeleteF: deleteF,")
|
|
g.writeLine("DiffF: diffF,")
|
|
g.writeLine("ReadF: readF,")
|
|
g.writeLine("UpdateF: updateF,")
|
|
},
|
|
"}, nil",
|
|
)
|
|
},
|
|
"}),",
|
|
)
|
|
}
|
|
}
|
|
}
|