mirror of https://github.com/pulumi/pulumi.git
443 lines
15 KiB
Go
443 lines
15 KiB
Go
// Copyright 2016-2023, 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 deploytest
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"testing"
|
|
|
|
"github.com/blang/semver"
|
|
"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/tokens"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestProvider(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("SignalCancellation", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has CancelF", func(t *testing.T) {
|
|
t.Parallel()
|
|
var called bool
|
|
prov := &Provider{
|
|
CancelF: func() error {
|
|
called = true
|
|
return errors.New("expected error")
|
|
},
|
|
}
|
|
assert.Error(t, prov.SignalCancellation(context.Background()))
|
|
assert.True(t, called)
|
|
})
|
|
t.Run("no CancelF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
assert.NoError(t, prov.SignalCancellation(context.Background()))
|
|
})
|
|
})
|
|
t.Run("Close", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
assert.NoError(t, prov.Close())
|
|
// Ensure idempotent.
|
|
assert.NoError(t, prov.Close())
|
|
})
|
|
t.Run("GetPluginInfo", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{
|
|
Name: "expected-name",
|
|
Version: semver.MustParse("1.0.0"),
|
|
}
|
|
info, err := prov.GetPluginInfo(context.Background())
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "expected-name", info.Name)
|
|
// Ensure reference is passed correctly.
|
|
assert.Equal(t, &prov.Version, info.Version)
|
|
})
|
|
t.Run("GetSchema", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has GetSchemaF", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
expectedVersion := semver.MustParse("1.0.0")
|
|
var called bool
|
|
prov := &Provider{
|
|
GetSchemaF: func(request plugin.GetSchemaRequest) ([]byte, error) {
|
|
assert.Equal(t, 1, request.Version)
|
|
assert.Equal(t, "expected-subpackage", request.SubpackageName)
|
|
assert.Equal(t, &expectedVersion, request.SubpackageVersion)
|
|
called = true
|
|
return nil, expectedErr
|
|
},
|
|
}
|
|
_, err := prov.GetSchema(context.Background(), plugin.GetSchemaRequest{
|
|
Version: 1,
|
|
SubpackageName: "expected-subpackage",
|
|
SubpackageVersion: &expectedVersion,
|
|
})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
assert.True(t, called)
|
|
})
|
|
t.Run("no GetSchemaF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
b, err := prov.GetSchema(context.Background(), plugin.GetSchemaRequest{})
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, []byte("{}"), b.Schema)
|
|
})
|
|
})
|
|
t.Run("CheckConfig", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has CheckConfigF", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
var called bool
|
|
prov := &Provider{
|
|
CheckConfigF: func(
|
|
urn resource.URN, olds, news resource.PropertyMap, allowUnknowns bool,
|
|
) (resource.PropertyMap, []plugin.CheckFailure, error) {
|
|
assert.Equal(t, resource.URN("expected-urn"), urn)
|
|
assert.Equal(t, resource.NewStringProperty("old-value"), olds["old"])
|
|
assert.Equal(t, resource.NewStringProperty("new-value"), news["new"])
|
|
called = true
|
|
return nil, nil, expectedErr
|
|
},
|
|
}
|
|
_, err := prov.CheckConfig(context.Background(), plugin.CheckConfigRequest{
|
|
URN: resource.URN("expected-urn"),
|
|
Olds: resource.PropertyMap{
|
|
"old": resource.NewStringProperty("old-value"),
|
|
},
|
|
News: resource.PropertyMap{
|
|
"new": resource.NewStringProperty("new-value"),
|
|
},
|
|
AllowUnknowns: true,
|
|
})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
assert.True(t, called)
|
|
})
|
|
t.Run("no CheckConfigF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
resp, err := prov.CheckConfig(context.Background(), plugin.CheckConfigRequest{
|
|
News: resource.PropertyMap{
|
|
"expected": resource.NewStringProperty("expected-value"),
|
|
},
|
|
AllowUnknowns: true,
|
|
})
|
|
assert.NoError(t, err)
|
|
assert.Empty(t, resp.Failures)
|
|
// Should return the news.
|
|
assert.Equal(t, resource.NewStringProperty("expected-value"), resp.Properties["expected"])
|
|
})
|
|
})
|
|
t.Run("Construct", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has ConstructF", func(t *testing.T) {
|
|
t.Run("inject error", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
var dialCalled bool
|
|
var constructCalled bool
|
|
expectedResmon := &ResourceMonitor{}
|
|
prov := &Provider{
|
|
DialMonitorF: func(ctx context.Context, endpoint string) (*ResourceMonitor, error) {
|
|
assert.Equal(t, "expected-endpoint", endpoint)
|
|
dialCalled = true
|
|
// Returns no error to avoid short-circuiting due to the monitor being provided
|
|
// an invalid resource monitor address.
|
|
return expectedResmon, nil
|
|
},
|
|
ConstructF: func(
|
|
monitor *ResourceMonitor,
|
|
typ, name string, parent resource.URN, inputs resource.PropertyMap,
|
|
info plugin.ConstructInfo, options plugin.ConstructOptions,
|
|
) (plugin.ConstructResult, error) {
|
|
assert.Equal(t, expectedResmon, monitor)
|
|
constructCalled = true
|
|
return plugin.ConstructResult{}, expectedErr
|
|
},
|
|
}
|
|
_, err := prov.Construct(context.Background(), plugin.ConstructRequest{
|
|
Type: tokens.Type("some-type"),
|
|
Name: "name",
|
|
Info: plugin.ConstructInfo{
|
|
MonitorAddress: "expected-endpoint",
|
|
},
|
|
Parent: resource.URN("<parent-urn>"),
|
|
})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
assert.True(t, dialCalled)
|
|
assert.True(t, constructCalled)
|
|
})
|
|
t.Run("dial error", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("invalid address", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{
|
|
ConstructF: func(
|
|
monitor *ResourceMonitor,
|
|
typ, name string, parent resource.URN, inputs resource.PropertyMap,
|
|
info plugin.ConstructInfo, options plugin.ConstructOptions,
|
|
) (plugin.ConstructResult, error) {
|
|
assert.Fail(t, "Construct should not be called")
|
|
return plugin.ConstructResult{}, nil
|
|
},
|
|
}
|
|
_, err := prov.Construct(context.Background(), plugin.ConstructRequest{
|
|
Type: tokens.Type("some-type"),
|
|
Name: "name",
|
|
Parent: resource.URN("<parent-urn>"),
|
|
})
|
|
assert.ErrorContains(t, err, "could not connect to resource monitor")
|
|
})
|
|
t.Run("injected error", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
var dialCalled bool
|
|
prov := &Provider{
|
|
DialMonitorF: func(ctx context.Context, endpoint string) (*ResourceMonitor, error) {
|
|
dialCalled = true
|
|
// Returns no error to avoid short-circuiting due to the monitor being provided
|
|
// an invalid resource monitor address.
|
|
return nil, expectedErr
|
|
},
|
|
ConstructF: func(
|
|
monitor *ResourceMonitor,
|
|
typ, name string, parent resource.URN, inputs resource.PropertyMap,
|
|
info plugin.ConstructInfo, options plugin.ConstructOptions,
|
|
) (plugin.ConstructResult, error) {
|
|
assert.Fail(t, "Construct should not be called")
|
|
return plugin.ConstructResult{}, nil
|
|
},
|
|
}
|
|
_, err := prov.Construct(context.Background(), plugin.ConstructRequest{
|
|
Type: tokens.Type("some-type"),
|
|
Name: "name",
|
|
Parent: resource.URN("<parent-urn>"),
|
|
})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
assert.True(t, dialCalled)
|
|
})
|
|
})
|
|
})
|
|
t.Run("no ConstructF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{
|
|
DialMonitorF: func(ctx context.Context, endpoint string) (*ResourceMonitor, error) {
|
|
assert.Fail(t, "DialMonitor should not be called")
|
|
return nil, nil
|
|
},
|
|
}
|
|
_, err := prov.Construct(context.Background(), plugin.ConstructRequest{
|
|
Type: tokens.Type("some-type"),
|
|
Name: "name",
|
|
Parent: resource.URN("<parent-urn>"),
|
|
})
|
|
assert.NoError(t, err)
|
|
})
|
|
})
|
|
t.Run("Invoke", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has InvokeF", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedPropertyMap := resource.PropertyMap{
|
|
"key": resource.NewStringProperty("expected-value"),
|
|
}
|
|
var called bool
|
|
prov := &Provider{
|
|
InvokeF: func(tok tokens.ModuleMember, inputs resource.PropertyMap,
|
|
) (resource.PropertyMap, []plugin.CheckFailure, error) {
|
|
assert.Equal(t, tokens.ModuleMember("expected-tok"), tok)
|
|
called = true
|
|
return expectedPropertyMap, nil, nil
|
|
},
|
|
}
|
|
resp, err := prov.Invoke(context.Background(), plugin.InvokeRequest{
|
|
Tok: "expected-tok",
|
|
})
|
|
assert.NoError(t, err)
|
|
assert.True(t, called)
|
|
assert.Equal(t, expectedPropertyMap, resp.Properties)
|
|
})
|
|
t.Run("no InvokeF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
resp, err := prov.Invoke(context.Background(), plugin.InvokeRequest{})
|
|
assert.NoError(t, err)
|
|
assert.Empty(t, resp.Failures)
|
|
assert.Equal(t, resource.PropertyMap{}, resp.Properties)
|
|
})
|
|
})
|
|
t.Run("StreamInvoke", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has StreamInvokeF", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
prov := &Provider{
|
|
StreamInvokeF: func(
|
|
tok tokens.ModuleMember, args resource.PropertyMap,
|
|
onNext func(resource.PropertyMap) error,
|
|
) ([]plugin.CheckFailure, error) {
|
|
assert.Equal(t, tokens.ModuleMember("expected-tok"), tok)
|
|
return nil, expectedErr
|
|
},
|
|
}
|
|
_, err := prov.StreamInvoke(context.Background(), plugin.StreamInvokeRequest{Tok: "expected-tok"})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
})
|
|
t.Run("no StreamInvokeF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
_, err := prov.StreamInvoke(context.Background(), plugin.StreamInvokeRequest{})
|
|
assert.ErrorContains(t, err, "StreamInvoke unimplemented")
|
|
})
|
|
})
|
|
t.Run("Call", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has CallF", func(t *testing.T) {
|
|
t.Run("inject error", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
var dialCalled bool
|
|
var callCalled bool
|
|
expectedResmon := &ResourceMonitor{}
|
|
prov := &Provider{
|
|
DialMonitorF: func(ctx context.Context, endpoint string) (*ResourceMonitor, error) {
|
|
dialCalled = true
|
|
// Returns no error to avoid short-circuiting due to the monitor being provided
|
|
// an invalid resource monitor address.
|
|
return expectedResmon, nil
|
|
},
|
|
CallF: func(
|
|
monitor *ResourceMonitor, tok tokens.ModuleMember, args resource.PropertyMap,
|
|
info plugin.CallInfo, options plugin.CallOptions,
|
|
) (plugin.CallResult, error) {
|
|
assert.Equal(t, expectedResmon, monitor)
|
|
assert.Equal(t, tokens.ModuleMember("expected-tok"), tok)
|
|
callCalled = true
|
|
return plugin.CallResult{}, expectedErr
|
|
},
|
|
}
|
|
_, err := prov.Call(context.Background(), plugin.CallRequest{Tok: "expected-tok"})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
assert.True(t, dialCalled)
|
|
assert.True(t, callCalled)
|
|
})
|
|
t.Run("dial error", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("invalid address", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{
|
|
CallF: func(
|
|
monitor *ResourceMonitor, tok tokens.ModuleMember, args resource.PropertyMap,
|
|
info plugin.CallInfo, options plugin.CallOptions,
|
|
) (plugin.CallResult, error) {
|
|
assert.Fail(t, "Call should not be called")
|
|
return plugin.CallResult{}, nil
|
|
},
|
|
}
|
|
_, err := prov.Call(context.Background(), plugin.CallRequest{})
|
|
assert.ErrorContains(t, err, "could not connect to resource monitor")
|
|
})
|
|
t.Run("injected error", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
var dialCalled bool
|
|
prov := &Provider{
|
|
DialMonitorF: func(ctx context.Context, endpoint string) (*ResourceMonitor, error) {
|
|
dialCalled = true
|
|
// Returns no error to avoid short-circuiting due to the monitor being provided
|
|
// an invalid resource monitor address.
|
|
return nil, expectedErr
|
|
},
|
|
CallF: func(
|
|
monitor *ResourceMonitor, tok tokens.ModuleMember, args resource.PropertyMap,
|
|
info plugin.CallInfo, options plugin.CallOptions,
|
|
) (plugin.CallResult, error) {
|
|
assert.Fail(t, "Call should not be called")
|
|
return plugin.CallResult{}, expectedErr
|
|
},
|
|
}
|
|
_, err := prov.Call(context.Background(), plugin.CallRequest{})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
assert.True(t, dialCalled)
|
|
})
|
|
})
|
|
})
|
|
t.Run("no CallF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{
|
|
DialMonitorF: func(ctx context.Context, endpoint string) (*ResourceMonitor, error) {
|
|
assert.Fail(t, "Dial should not be called")
|
|
return nil, nil
|
|
},
|
|
}
|
|
_, err := prov.Call(context.Background(), plugin.CallRequest{})
|
|
assert.NoError(t, err)
|
|
})
|
|
})
|
|
t.Run("GetMapping", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has GetMappingF", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
prov := &Provider{
|
|
GetMappingF: func(key, provider string) ([]byte, string, error) {
|
|
assert.Equal(t, "expected-key", key)
|
|
assert.Equal(t, "expected-provider", provider)
|
|
return nil, "", expectedErr
|
|
},
|
|
}
|
|
_, err := prov.GetMapping(context.Background(), plugin.GetMappingRequest{
|
|
Key: "expected-key",
|
|
Provider: "expected-provider",
|
|
})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
})
|
|
t.Run("no GetMappingF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
resp, err := prov.GetMapping(context.Background(), plugin.GetMappingRequest{})
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "", resp.Provider)
|
|
assert.Nil(t, resp.Data)
|
|
})
|
|
})
|
|
t.Run("GetMappings", func(t *testing.T) {
|
|
t.Parallel()
|
|
t.Run("has GetMappingsF", func(t *testing.T) {
|
|
t.Parallel()
|
|
expectedErr := errors.New("expected error")
|
|
prov := &Provider{
|
|
GetMappingsF: func(key string) ([]string, error) {
|
|
assert.Equal(t, "expected-key", key)
|
|
return nil, expectedErr
|
|
},
|
|
}
|
|
_, err := prov.GetMappings(context.Background(), plugin.GetMappingsRequest{Key: "expected-key"})
|
|
assert.ErrorIs(t, err, expectedErr)
|
|
})
|
|
t.Run("no GetMappingsF", func(t *testing.T) {
|
|
t.Parallel()
|
|
prov := &Provider{}
|
|
mappingStrs, err := prov.GetMappings(context.Background(), plugin.GetMappingsRequest{})
|
|
assert.NoError(t, err)
|
|
assert.Empty(t, mappingStrs)
|
|
})
|
|
})
|
|
}
|