pulumi/pkg/engine/lifecycletest/analyzer_test.go

486 lines
15 KiB
Go

// Copyright 2022-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 lifecycletest
import (
"context"
"encoding/json"
"errors"
"testing"
"github.com/blang/semver"
. "github.com/pulumi/pulumi/pkg/v3/engine" //nolint:revive
"github.com/pulumi/pulumi/pkg/v3/resource/deploy"
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/deploytest"
"github.com/pulumi/pulumi/sdk/v3/go/common/apitype"
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
"github.com/stretchr/testify/assert"
)
type testRequiredPolicy struct {
name string
version string
config map[string]*json.RawMessage
}
func (p *testRequiredPolicy) Name() string {
return p.name
}
func (p *testRequiredPolicy) Version() string {
return p.version
}
func (p *testRequiredPolicy) Install(_ context.Context) (string, error) {
return "", nil
}
func (p *testRequiredPolicy) Config() map[string]*json.RawMessage {
return p.config
}
func NewRequiredPolicy(name, version string, config map[string]*json.RawMessage) RequiredPolicy {
return &testRequiredPolicy{
name: name,
version: version,
config: config,
}
}
func TestSimpleAnalyzer(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.NoError(t, err)
return nil
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: hostF,
},
}
project := p.GetProject()
_, err := TestOp(Update).Run(project, p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil)
assert.NoError(t, err)
}
func TestSimpleAnalyzeResourceFailure(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{
AnalyzeF: func(r plugin.AnalyzerResource) ([]plugin.AnalyzeDiagnostic, error) {
return []plugin.AnalyzeDiagnostic{{
PolicyName: "always-fails",
PolicyPackName: "analyzerA",
Description: "a policy that always fails",
Message: "a policy failed",
EnforcementLevel: apitype.Mandatory,
URN: r.URN,
}}, nil
},
}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.Error(t, err)
return nil
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: hostF,
},
}
project := p.GetProject()
_, err := TestOp(Update).Run(project, p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil)
assert.Error(t, err)
}
func TestSimpleAnalyzeStackFailure(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{
AnalyzeStackF: func(rs []plugin.AnalyzerStackResource) ([]plugin.AnalyzeDiagnostic, error) {
return []plugin.AnalyzeDiagnostic{{
PolicyName: "always-fails",
PolicyPackName: "analyzerA",
Description: "a policy that always fails",
Message: "a policy failed",
EnforcementLevel: apitype.Mandatory,
URN: rs[0].URN,
}}, nil
},
}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.NoError(t, err)
return nil
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
SkipDisplayTests: true, // TODO: this seems flaky, could use some more investigation.
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: hostF,
},
}
project := p.GetProject()
_, err := TestOp(Update).Run(project, p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil)
assert.Error(t, err)
}
// TestResourceRemediation tests a very simple sequence of remediations. We register two, to ensure that
// the remediations are applied in the order specified, an important part of the design.
func TestResourceRemediation(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{
RemediateF: func(r plugin.AnalyzerResource) ([]plugin.Remediation, error) {
// Run two remediations to ensure they are applied in order.
return []plugin.Remediation{
{
PolicyName: "ignored",
PolicyPackName: "analyzerA",
PolicyPackVersion: "1.0.0",
Description: "a remediation that gets ignored because it runs first",
Properties: resource.PropertyMap{
"a": resource.NewStringProperty("nope"),
"ggg": resource.NewBoolProperty(true),
},
},
{
PolicyName: "real-deal",
PolicyPackName: "analyzerA",
PolicyPackVersion: "1.0.0",
Description: "a remediation that actually gets applied because it runs last",
Properties: resource.PropertyMap{
"a": resource.NewStringProperty("foo"),
"fff": resource.NewBoolProperty(true),
"z": resource.NewStringProperty("bar"),
},
},
}, nil
},
}, nil
}),
}
program := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.NoError(t, err)
return nil
})
host := deploytest.NewPluginHostF(nil, nil, program, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: host,
},
}
project := p.GetProject()
snap, err := TestOp(Update).Run(project, p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil)
// Expect no error, valid snapshot, two resources:
assert.Nil(t, err)
assert.NotNil(t, snap)
assert.Equal(t, 2, len(snap.Resources)) // stack plus pkA:m:typA
// Ensure the rewritten properties have been applied to the inputs:
r := snap.Resources[1]
assert.Equal(t, "pkgA:m:typA", string(r.Type))
assert.Equal(t, 3, len(r.Inputs))
assert.Equal(t, "foo", r.Inputs["a"].StringValue())
assert.Equal(t, true, r.Inputs["fff"].BoolValue())
assert.Equal(t, "bar", r.Inputs["z"].StringValue())
}
// TestRemediationDiagnostic tests the case where a remediation issues a diagnostic rather than transforming
// state. In this case, the deployment should still succeed, even though no transforms took place.
func TestRemediationDiagnostic(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{
RemediateF: func(r plugin.AnalyzerResource) ([]plugin.Remediation, error) {
return []plugin.Remediation{{
PolicyName: "warning",
PolicyPackName: "analyzerA",
PolicyPackVersion: "1.0.0",
Description: "a remediation with a diagnostic",
Diagnostic: "warning - could not run due to unknowns",
}}, nil
},
}, nil
}),
}
program := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.NoError(t, err)
return nil
})
host := deploytest.NewPluginHostF(nil, nil, program, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: host,
},
}
project := p.GetProject()
snap, err := TestOp(Update).Run(project, p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil)
// Expect no error, valid snapshot, two resources:
assert.NoError(t, err)
assert.NotNil(t, snap)
assert.Equal(t, 2, len(snap.Resources)) // stack plus pkA:m:typA
}
// TestRemediateFailure tests the case where a remediation fails to execute. In this case, the whole
// deployment itself should also fail.
func TestRemediateFailure(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{
RemediateF: func(r plugin.AnalyzerResource) ([]plugin.Remediation, error) {
return nil, errors.New("this remediation failed")
},
}, nil
}),
}
program := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.NoError(t, err)
return nil
})
host := deploytest.NewPluginHostF(nil, nil, program, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: host,
},
}
project := p.GetProject()
snap, res := TestOp(Update).Run(project, p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil)
assert.NotNil(t, res)
assert.NotNil(t, snap)
assert.Equal(t, 0, len(snap.Resources))
}
func TestSimpleAnalyzeResourceFailureRemediateDowngradedToMandatory(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{
AnalyzeF: func(r plugin.AnalyzerResource) ([]plugin.AnalyzeDiagnostic, error) {
return []plugin.AnalyzeDiagnostic{{
PolicyName: "always-fails",
PolicyPackName: "analyzerA",
Description: "a policy that always fails",
Message: "a policy failed",
EnforcementLevel: apitype.Remediate,
URN: r.URN,
}}, nil
},
}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.Error(t, err)
return nil
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: hostF,
},
Steps: []TestStep{
{
Op: Update,
SkipPreview: true,
ExpectFailure: true,
Validate: func(project workspace.Project, target deploy.Target, entries JournalEntries,
events []Event, err error,
) error {
violationEvents := []Event{}
for _, e := range events {
if e.Type == PolicyViolationEvent {
violationEvents = append(violationEvents, e)
}
}
assert.Len(t, violationEvents, 1)
assert.Equal(t, apitype.Mandatory,
violationEvents[0].Payload().(PolicyViolationEventPayload).EnforcementLevel)
return err
},
},
},
}
p.Run(t, nil)
}
func TestSimpleAnalyzeStackFailureRemediateDowngradedToMandatory(t *testing.T) {
t.Parallel()
loaders := []*deploytest.PluginLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{}, nil
}),
deploytest.NewAnalyzerLoader("analyzerA", func(_ *plugin.PolicyAnalyzerOptions) (plugin.Analyzer, error) {
return &deploytest.Analyzer{
AnalyzeStackF: func(rs []plugin.AnalyzerStackResource) ([]plugin.AnalyzeDiagnostic, error) {
return []plugin.AnalyzeDiagnostic{{
PolicyName: "always-fails",
PolicyPackName: "analyzerA",
Description: "a policy that always fails",
Message: "a policy failed",
EnforcementLevel: apitype.Remediate,
URN: rs[0].URN,
}}, nil
},
}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
_, err := monitor.RegisterResource("pkgA:m:typA", "resA", true)
assert.NoError(t, err)
return nil
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{
T: t,
SkipDisplayTests: true, // TODO: this seems flaky, could use some more investigation.
UpdateOptions: UpdateOptions{
RequiredPolicies: []RequiredPolicy{NewRequiredPolicy("analyzerA", "", nil)},
},
HostF: hostF,
},
Steps: []TestStep{
{
Op: Update,
SkipPreview: true,
ExpectFailure: true,
Validate: func(project workspace.Project, target deploy.Target, entries JournalEntries,
events []Event, err error,
) error {
violationEvents := []Event{}
for _, e := range events {
if e.Type == PolicyViolationEvent {
violationEvents = append(violationEvents, e)
}
}
assert.Len(t, violationEvents, 1)
assert.Equal(t, apitype.Mandatory,
violationEvents[0].Payload().(PolicyViolationEventPayload).EnforcementLevel)
return err
},
},
},
}
p.Run(t, nil)
}