pulumi/pkg/engine/lifecycletest/parameterized_test.go

502 lines
17 KiB
Go

// Copyright 2024-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"
"fmt"
"testing"
"github.com/blang/semver"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
. "github.com/pulumi/pulumi/pkg/v3/engine" //nolint:revive
"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/promise"
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/config"
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
pulumirpc "github.com/pulumi/pulumi/sdk/v3/proto/go"
)
// TestPackageRef tests we can request a package ref from the engine and then use that, instead of Version,
// PackageDownloadURL etc.
func TestPackageRef(t *testing.T) {
t.Parallel()
loaders := []*deploytest.ProviderLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{
CreateF: func(_ context.Context, req plugin.CreateRequest) (plugin.CreateResponse, error) {
return plugin.CreateResponse{
ID: "0",
Properties: req.Properties,
Status: resource.StatusOK,
}, nil
},
}, nil
}),
deploytest.NewProviderLoader("pkgA", semver.MustParse("2.0.0"), func() (plugin.Provider, error) {
return &deploytest.Provider{
CreateF: func(_ context.Context, req plugin.CreateRequest) (plugin.CreateResponse, error) {
return plugin.CreateResponse{
ID: "1",
Properties: req.Properties,
Status: resource.StatusOK,
}, nil
},
}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
pkg1Ref, err := monitor.RegisterPackage("pkgA", "1.0.0", "", nil, nil)
require.NoError(t, err)
pkg2Ref, err := monitor.RegisterPackage("pkgA", "2.0.0", "", nil, nil)
require.NoError(t, err)
// If we register the "same" provider in parallel, we should get the same ref.
promises := []*promise.Promise[string]{}
for i := 0; i < 100; i++ {
var pcs promise.CompletionSource[string]
promises = append(promises, pcs.Promise())
go func() {
ref, err := monitor.RegisterPackage("pkgB", "1.0.0", "downloadUrl", nil, nil)
require.NoError(t, err)
pcs.MustFulfill(ref)
}()
}
ctx := context.Background()
expected, err := promises[0].Result(ctx)
require.NoError(t, err)
for i := 1; i < 100; i++ {
got, err := promises[i].Result(ctx)
require.NoError(t, err)
assert.Equal(t, expected, got)
}
// Now register some resources using the UUID for the provider, instead of a normal provider ref.
resp, err := monitor.RegisterResource("pkgA:m:typA", "resA", true, deploytest.ResourceOptions{
PackageRef: pkg1Ref,
})
require.NoError(t, err)
assert.Equal(t, resource.ID("0"), resp.ID)
resp, err = monitor.RegisterResource("pkgA:m:typA", "resB", true, deploytest.ResourceOptions{
PackageRef: pkg2Ref,
})
require.NoError(t, err)
assert.Equal(t, resource.ID("1"), resp.ID)
return err
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{T: t, HostF: hostF},
}
snap, err := TestOp(Update).RunStep(p.GetProject(), p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil, "0")
assert.NoError(t, err)
assert.NotNil(t, snap)
assert.Len(t, snap.Resources, 4)
assert.Equal(t, string(snap.Resources[0].URN)+"::"+string(snap.Resources[0].ID), snap.Resources[1].Provider)
assert.Equal(t, string(snap.Resources[2].URN)+"::"+string(snap.Resources[2].ID), snap.Resources[3].Provider)
}
// TestReplacementParameterizedProvider tests that we can register a parameterized provider that replaces a base
// provider.
func TestReplacementParameterizedProvider(t *testing.T) {
t.Parallel()
loadCount := 0
loaders := []*deploytest.ProviderLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
loadCount++
var param string
return &deploytest.Provider{
ParameterizeF: func(
ctx context.Context, req plugin.ParameterizeRequest,
) (plugin.ParameterizeResponse, error) {
value := req.Parameters.(*plugin.ParameterizeValue)
param = string(value.Value)
return plugin.ParameterizeResponse{
Name: value.Name,
Version: value.Version,
}, nil
},
CreateF: func(_ context.Context, req plugin.CreateRequest) (plugin.CreateResponse, error) {
if req.URN.Type() == "pkgExt:m:typA" {
assert.Equal(t, "replacement", param)
}
return plugin.CreateResponse{
ID: "id",
Properties: req.Properties,
Status: resource.StatusOK,
}, nil
},
InvokeF: func(_ context.Context, req plugin.InvokeRequest) (plugin.InvokeResponse, error) {
assert.Equal(t, "pkgExt:index:func", req.Tok.String())
assert.Equal(t, resource.NewStringProperty("in"), req.Args["input"])
return plugin.InvokeResponse{
Properties: resource.PropertyMap{
"output": resource.NewStringProperty("in " + param),
},
}, nil
},
ReadF: func(_ context.Context, req plugin.ReadRequest) (plugin.ReadResponse, error) {
if param == "" {
assert.Equal(t, tokens.Type("pkgA:m:typA"), req.URN.Type())
} else {
assert.Equal(t, tokens.Type("pkgExt:m:typA"), req.URN.Type())
}
return plugin.ReadResponse{
ReadResult: plugin.ReadResult{
ID: req.ID,
Inputs: req.Inputs,
Outputs: req.State,
},
Status: resource.StatusOK,
}, nil
},
CallF: func(_ context.Context, req plugin.CallRequest, _ *deploytest.ResourceMonitor) (plugin.CallResponse, error) {
assert.Equal(t, "pkgExt:index:call", req.Tok.String())
assert.Equal(t, resource.NewStringProperty("in"), req.Args["input"])
assert.Equal(t, map[resource.PropertyKey][]resource.URN{
"input": {"urn:pulumi:stack::m::typA::resB"},
}, req.Options.ArgDependencies)
return plugin.CallResponse{
Return: resource.PropertyMap{
"output": resource.NewStringProperty("output"),
},
ReturnDependencies: map[resource.PropertyKey][]resource.URN{
"output": {"urn:pulumi:stack::m::typA::resB"},
},
Failures: nil,
}, nil
},
ConstructF: func(
_ context.Context,
req plugin.ConstructRequest,
_ *deploytest.ResourceMonitor,
) (plugin.ConstructResponse, error) {
if param == "" {
assert.Equal(t, tokens.Type("pkgA:m:typA"), req.Type)
assert.Equal(t, "mlcA", req.Name)
} else {
assert.Equal(t, tokens.Type("pkgExt:m:typA"), req.Type)
assert.Equal(t, "mlcB", req.Name)
}
return plugin.ConstructResponse{
URN: resource.NewURN("", "", "", req.Type, req.Name),
Outputs: resource.PropertyMap{
"output": resource.NewStringProperty("output"),
},
OutputDependencies: map[resource.PropertyKey][]resource.URN{
"output": {"urn:pulumi:stack::m::typA::resB"},
},
}, nil
},
}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
pkgRef, err := monitor.RegisterPackage("pkgA", "1.0.0", "", nil, nil)
require.NoError(t, err)
// Register a resource using that base provider
_, err = monitor.RegisterResource("pkgA:m:typA", "resA", true, deploytest.ResourceOptions{
PackageRef: pkgRef,
})
require.NoError(t, err)
// Register a multi-language component with the base provider
mlcA, err := monitor.RegisterResource("pkgA:m:typA", "mlcA", true, deploytest.ResourceOptions{
PackageRef: pkgRef,
Remote: true,
})
assert.NoError(t, err)
assert.Equal(t, "mlcA", mlcA.URN.Name())
assert.Equal(t, resource.PropertyMap{
"output": resource.NewStringProperty("output"),
}, mlcA.Outputs)
assert.Equal(t, map[resource.PropertyKey][]resource.URN{
"output": {"urn:pulumi:stack::m::typA::resB"},
}, mlcA.Dependencies)
// Now register a replacement provider
extRef, err := monitor.RegisterPackage("pkgA", "1.0.0", "", nil, &pulumirpc.Parameterization{
Name: "pkgExt",
Version: "0.5.0",
Value: []byte("replacement"),
})
require.NoError(t, err)
// Test registering a resource with the replacement provider
_, err = monitor.RegisterResource("pkgExt:m:typA", "resB", true, deploytest.ResourceOptions{
PackageRef: extRef,
})
require.NoError(t, err)
// Register a multi-language component with the replacement provider
mlcB, err := monitor.RegisterResource("pkgExt:m:typA", "mlcB", true, deploytest.ResourceOptions{
PackageRef: extRef,
Remote: true,
})
assert.NoError(t, err)
assert.Equal(t, "mlcB", mlcB.URN.Name())
assert.Equal(t, resource.PropertyMap{
"output": resource.NewStringProperty("output"),
}, mlcB.Outputs)
assert.Equal(t, map[resource.PropertyKey][]resource.URN{
"output": {"urn:pulumi:stack::m::typA::resB"},
}, mlcB.Dependencies)
// Test invoking a function on the replacement provider
result, _, err := monitor.Invoke("pkgExt:index:func", resource.PropertyMap{
"input": resource.NewStringProperty("in"),
}, "", "", extRef)
require.NoError(t, err)
assert.Equal(t, resource.PropertyMap{
"output": resource.NewStringProperty("in replacement"),
}, result)
// Test reading a resource on the replacement provider
_, _, err = monitor.ReadResource("pkgExt:m:typA", "resC", "id", "", resource.PropertyMap{}, "", "", "", extRef)
require.NoError(t, err)
// Test calling a function on the replacement provider
callOuts, callDeps, callFailures, err := monitor.Call(
"pkgExt:index:call",
resource.PropertyMap{
"input": resource.NewStringProperty("in"),
},
map[resource.PropertyKey][]resource.URN{
"input": {"urn:pulumi:stack::m::typA::resB"},
},
"", /*provider*/
"", /*version*/
extRef,
)
assert.NoError(t, err)
assert.Equal(t, resource.PropertyMap{
"output": resource.NewStringProperty("output"),
}, callOuts)
assert.Equal(t, map[resource.PropertyKey][]resource.URN{
"output": {"urn:pulumi:stack::m::typA::resB"},
}, callDeps)
assert.Nil(t, callFailures)
// Test that we can create an explicit replacement provider and can use it
prov, err := monitor.RegisterResource("pulumi:providers:pkgExt", "provider", true, deploytest.ResourceOptions{
PackageRef: extRef,
})
assert.NoError(t, err)
provID := prov.ID
if provID == "" {
provID = providers.UnknownID
}
provRef, err := providers.NewReference(prov.URN, provID)
assert.NoError(t, err)
_, err = monitor.RegisterResource("pkgExt:m:typA", "resD", true, deploytest.ResourceOptions{
Provider: provRef.String(),
})
require.NoError(t, err)
return err
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{T: t, HostF: hostF},
}
snap, err := TestOp(Update).RunStep(
p.GetProject(), p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil, "up")
require.NoError(t, err)
assert.NotNil(t, snap)
assert.Len(t, snap.Resources, 7)
// Check that we loaded the provider thrice
assert.Equal(t, 3, loadCount)
// Check the state of the parameterized provider is what we expect
prov := snap.Resources[2]
assert.Equal(t, tokens.Type("pulumi:providers:pkgExt"), prov.Type)
assert.Equal(t, "default_0_5_0", prov.URN.Name())
assert.Equal(t, resource.NewPropertyMapFromMap(map[string]any{
"version": "0.5.0",
"__internal": map[string]any{
"name": "pkgA",
"version": "1.0.0",
"parameterization": "cmVwbGFjZW1lbnQ=",
},
}), prov.Inputs)
snap, err = TestOp(Refresh).RunStep(
p.GetProject(), p.GetTarget(t, snap), p.Options, false, p.BackendClient, nil, "refresh")
require.NoError(t, err)
assert.NotNil(t, snap)
assert.Len(t, snap.Resources, 7)
snap, err = TestOp(Destroy).RunStep(
p.GetProject(), p.GetTarget(t, snap), p.Options, false, p.BackendClient, nil, "destroy")
require.NoError(t, err)
assert.NotNil(t, snap)
assert.Len(t, snap.Resources, 0)
}
// TestReplacementParameterizedProviderConfig tests that we can register a parameterized provider that uses config keys
// like "name" without clashing against the internal state the engine tracks for parameterization. c.f.
// https://github.com/pulumi/pulumi/issues/16757.
func TestReplacementParameterizedProviderConfig(t *testing.T) {
t.Parallel()
loaders := []*deploytest.ProviderLoader{
deploytest.NewProviderLoader("pkgA", semver.MustParse("1.0.0"), func() (plugin.Provider, error) {
var param string
return &deploytest.Provider{
ConfigureF: func(_ context.Context, req plugin.ConfigureRequest) (plugin.ConfigureResponse, error) {
// Ensure that the provider configuration is what we expect.
var expected resource.PropertyMap
if param == "replacement" {
expected = resource.NewPropertyMapFromMap(map[string]any{
"version": "0.5.0",
"name": "testingExt",
})
} else {
expected = resource.NewPropertyMapFromMap(map[string]any{
"version": "1.0.0",
"name": "testingBase",
})
}
if !req.Inputs.DeepEquals(expected) {
return plugin.ConfigureResponse{},
fmt.Errorf("expected provider configuration to be %v, got %v", expected, req.Inputs)
}
return plugin.ConfigureResponse{}, nil
},
ParameterizeF: func(
ctx context.Context, req plugin.ParameterizeRequest,
) (plugin.ParameterizeResponse, error) {
value := req.Parameters.(*plugin.ParameterizeValue)
param = string(value.Value)
return plugin.ParameterizeResponse{
Name: value.Name,
Version: value.Version,
}, nil
},
CreateF: func(_ context.Context, req plugin.CreateRequest) (plugin.CreateResponse, error) {
if req.URN.Type() == "pkgExt:m:typA" {
assert.Equal(t, "replacement", param)
}
return plugin.CreateResponse{
ID: "id",
Properties: req.Properties,
Status: resource.StatusOK,
}, nil
},
}, nil
}),
}
programF := deploytest.NewLanguageRuntimeF(func(_ plugin.RunInfo, monitor *deploytest.ResourceMonitor) error {
pkgRef, err := monitor.RegisterPackage("pkgA", "1.0.0", "http://example.com", nil, nil)
require.NoError(t, err)
// Register a resource using that base provider
_, err = monitor.RegisterResource("pkgA:m:typA", "resA", true, deploytest.ResourceOptions{
PackageRef: pkgRef,
})
require.NoError(t, err)
// Now register a replacement provider
extRef, err := monitor.RegisterPackage("pkgA", "1.0.0", "", nil, &pulumirpc.Parameterization{
Name: "pkgExt",
Version: "0.5.0",
Value: []byte("replacement"),
})
require.NoError(t, err)
_, err = monitor.RegisterResource("pkgExt:m:typA", "resB", true, deploytest.ResourceOptions{
PackageRef: extRef,
})
require.NoError(t, err)
return err
})
hostF := deploytest.NewPluginHostF(nil, nil, programF, loaders...)
p := &TestPlan{
Options: TestUpdateOptions{T: t, HostF: hostF},
Config: config.Map{
config.MustParseKey("pkgA:name"): config.NewValue("testingBase"),
config.MustParseKey("pkgExt:name"): config.NewValue("testingExt"),
},
}
snap, err := TestOp(Update).RunStep(
p.GetProject(), p.GetTarget(t, nil), p.Options, false, p.BackendClient, nil, "up")
require.NoError(t, err)
assert.NotNil(t, snap)
assert.Len(t, snap.Resources, 4)
// Check the state of the parameterized provider is what we expect
prov := snap.Resources[2]
assert.Equal(t, tokens.Type("pulumi:providers:pkgExt"), prov.Type)
assert.Equal(t, resource.NewPropertyMapFromMap(map[string]any{
"version": "0.5.0",
"name": "testingExt",
"__internal": map[string]any{
"name": "pkgA",
"version": "1.0.0",
"parameterization": "cmVwbGFjZW1lbnQ=",
},
}), prov.Inputs)
snap, err = TestOp(Refresh).RunStep(
p.GetProject(), p.GetTarget(t, snap), p.Options, false, p.BackendClient, nil, "refresh")
require.NoError(t, err)
assert.NotNil(t, snap)
assert.Len(t, snap.Resources, 4)
snap, err = TestOp(Destroy).RunStep(
p.GetProject(), p.GetTarget(t, snap), p.Options, false, p.BackendClient, nil, "destroy")
require.NoError(t, err)
assert.NotNil(t, snap)
assert.Len(t, snap.Resources, 0)
}