mirror of https://github.com/pulumi/pulumi.git
470 lines
12 KiB
Go
470 lines
12 KiB
Go
// Copyright 2016-2021, 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 main
|
|
|
|
import (
|
|
"context"
|
|
"flag"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/testing/iotest"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/fsutil"
|
|
pulumirpc "github.com/pulumi/pulumi/sdk/v3/proto/go"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestParseRunParams(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
desc string
|
|
give []string
|
|
want runParams
|
|
wantErr string // non-empty if we expect an error
|
|
}{
|
|
{
|
|
desc: "no arguments",
|
|
wantErr: "missing required engine RPC address argument",
|
|
},
|
|
{
|
|
desc: "no options",
|
|
give: []string{"localhost:1234"},
|
|
want: runParams{
|
|
engineAddress: "localhost:1234",
|
|
},
|
|
},
|
|
{
|
|
desc: "tracing",
|
|
give: []string{"-tracing", "foo.trace", "localhost:1234"},
|
|
want: runParams{
|
|
tracing: "foo.trace",
|
|
engineAddress: "localhost:1234",
|
|
},
|
|
},
|
|
{
|
|
desc: "binary",
|
|
give: []string{"-binary", "foo", "localhost:1234"},
|
|
want: runParams{
|
|
engineAddress: "localhost:1234",
|
|
},
|
|
},
|
|
{
|
|
desc: "buildTarget",
|
|
give: []string{"-buildTarget", "foo", "localhost:1234"},
|
|
want: runParams{
|
|
engineAddress: "localhost:1234",
|
|
},
|
|
},
|
|
{
|
|
desc: "root",
|
|
give: []string{"-root", "path/to/root", "localhost:1234"},
|
|
want: runParams{
|
|
engineAddress: "localhost:1234",
|
|
},
|
|
},
|
|
{
|
|
desc: "unknown option",
|
|
give: []string{"-unknown-option", "bar", "localhost:1234"},
|
|
wantErr: "flag provided but not defined: -unknown-option",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
tt := tt
|
|
t.Run(tt.desc, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Use a FlagSet with ContinueOnError for each case
|
|
// instead of using the global flag set.
|
|
//
|
|
// The global flag set uses flag.ExitOnError,
|
|
// so it cannot validate error cases during tests.
|
|
fset := flag.NewFlagSet(t.Name(), flag.ContinueOnError)
|
|
fset.SetOutput(iotest.LogWriter(t))
|
|
|
|
got, err := parseRunParams(fset, tt.give)
|
|
if tt.wantErr != "" {
|
|
assert.ErrorContains(t, err, tt.wantErr)
|
|
} else {
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, &tt.want, got)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestGetPlugin(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cases := []struct {
|
|
Name string
|
|
Mod *modInfo
|
|
Expected *pulumirpc.PluginDependency
|
|
ExpectedError string
|
|
JSON *plugin.PulumiPluginJSON
|
|
JSONPath string
|
|
}{
|
|
{
|
|
Name: "valid-pulumi-mod",
|
|
Mod: &modInfo{
|
|
Path: "github.com/pulumi/pulumi-aws/sdk",
|
|
Version: "v1.29.0",
|
|
},
|
|
Expected: &pulumirpc.PluginDependency{
|
|
Name: "aws",
|
|
Version: "v1.29.0",
|
|
},
|
|
},
|
|
{
|
|
Name: "pulumi-pseduo-version-plugin",
|
|
Mod: &modInfo{
|
|
Path: "github.com/pulumi/pulumi-aws/sdk",
|
|
Version: "v1.29.1-0.20200403140640-efb5e2a48a86",
|
|
},
|
|
Expected: &pulumirpc.PluginDependency{
|
|
Name: "aws",
|
|
Version: "v1.29.0",
|
|
},
|
|
},
|
|
{
|
|
Name: "non-pulumi-mod",
|
|
Mod: &modInfo{
|
|
Path: "github.com/moolumi/pulumi-aws/sdk",
|
|
Version: "v1.29.0",
|
|
},
|
|
ExpectedError: "module is not a pulumi provider",
|
|
},
|
|
{
|
|
Name: "invalid-version-module",
|
|
Mod: &modInfo{
|
|
Path: "github.com/pulumi/pulumi-aws/sdk",
|
|
Version: "42-42-42",
|
|
},
|
|
ExpectedError: "module does not have semver compatible version",
|
|
},
|
|
{
|
|
Name: "pulumi-pulumi-mod",
|
|
Mod: &modInfo{
|
|
Path: "github.com/pulumi/pulumi/sdk",
|
|
Version: "v1.14.0",
|
|
},
|
|
ExpectedError: "module is not a pulumi provider",
|
|
},
|
|
{
|
|
Name: "beta-pulumi-module",
|
|
Mod: &modInfo{
|
|
Path: "github.com/pulumi/pulumi-aws/sdk",
|
|
Version: "v2.0.0-beta.1",
|
|
},
|
|
Expected: &pulumirpc.PluginDependency{
|
|
Name: "aws",
|
|
Version: "v2.0.0-beta.1",
|
|
},
|
|
},
|
|
{
|
|
Name: "non-zero-patch-module", Mod: &modInfo{
|
|
Path: "github.com/pulumi/pulumi-kubernetes/sdk",
|
|
Version: "v1.5.8",
|
|
},
|
|
Expected: &pulumirpc.PluginDependency{
|
|
Name: "kubernetes",
|
|
Version: "v1.5.8",
|
|
},
|
|
},
|
|
{
|
|
Name: "pulumiplugin",
|
|
Mod: &modInfo{
|
|
Path: "github.com/me/myself/i",
|
|
Version: "invalid-Version",
|
|
},
|
|
Expected: &pulumirpc.PluginDependency{
|
|
Name: "thing1",
|
|
Version: "v1.2.3",
|
|
Server: "myserver.com",
|
|
},
|
|
JSON: &plugin.PulumiPluginJSON{
|
|
Resource: true,
|
|
Name: "thing1",
|
|
Version: "v1.2.3",
|
|
Server: "myserver.com",
|
|
},
|
|
},
|
|
{
|
|
Name: "non-resource",
|
|
Mod: &modInfo{},
|
|
ExpectedError: "module is not a pulumi provider",
|
|
JSON: &plugin.PulumiPluginJSON{
|
|
Resource: false,
|
|
},
|
|
},
|
|
{
|
|
Name: "missing-pulumiplugin",
|
|
Mod: &modInfo{
|
|
Dir: "/not/real",
|
|
},
|
|
ExpectedError: "module is not a pulumi provider",
|
|
JSON: &plugin.PulumiPluginJSON{
|
|
Name: "thing2",
|
|
Version: "v1.2.3",
|
|
},
|
|
},
|
|
{
|
|
Name: "pulumiplugin-go-lookup",
|
|
Mod: &modInfo{
|
|
Path: "github.com/me/myself",
|
|
Version: "v1.2.3",
|
|
},
|
|
JSON: &plugin.PulumiPluginJSON{
|
|
Name: "name",
|
|
Resource: true,
|
|
},
|
|
JSONPath: "go",
|
|
Expected: &pulumirpc.PluginDependency{
|
|
Name: "name",
|
|
Version: "v1.2.3",
|
|
},
|
|
},
|
|
{
|
|
Name: "pulumiplugin-go-name-lookup",
|
|
Mod: &modInfo{
|
|
Path: "github.com/me/myself",
|
|
Version: "v1.2.3",
|
|
},
|
|
JSON: &plugin.PulumiPluginJSON{
|
|
Name: "name",
|
|
Resource: true,
|
|
},
|
|
JSONPath: filepath.Join("go", "name"),
|
|
Expected: &pulumirpc.PluginDependency{
|
|
Name: "name",
|
|
Version: "v1.2.3",
|
|
},
|
|
},
|
|
{
|
|
Name: "pulumiplugin-nested-too-deep",
|
|
Mod: &modInfo{
|
|
Path: "path.com/here",
|
|
Version: "v0.0",
|
|
},
|
|
JSONPath: filepath.Join("go", "valid", "invalid"),
|
|
JSON: &plugin.PulumiPluginJSON{
|
|
Name: "name",
|
|
Resource: true,
|
|
},
|
|
ExpectedError: "module is not a pulumi provider",
|
|
},
|
|
{
|
|
Name: "nested-wrong-folder",
|
|
Mod: &modInfo{
|
|
Path: "path.com/here",
|
|
Version: "v0.0",
|
|
},
|
|
JSONPath: filepath.Join("invalid", "valid"),
|
|
JSON: &plugin.PulumiPluginJSON{
|
|
Name: "name",
|
|
Resource: true,
|
|
},
|
|
ExpectedError: "module is not a pulumi provider",
|
|
},
|
|
}
|
|
|
|
for _, c := range cases {
|
|
c := c
|
|
t.Run(c.Name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cwd := t.TempDir()
|
|
if c.Mod.Dir == "" {
|
|
c.Mod.Dir = cwd
|
|
}
|
|
if c.JSON != nil {
|
|
path := filepath.Join(cwd, c.JSONPath)
|
|
err := os.MkdirAll(path, 0o700)
|
|
assert.NoErrorf(t, err, "Failed to setup test folder %s", path)
|
|
bytes, err := c.JSON.JSON()
|
|
assert.NoError(t, err, "Failed to setup test pulumi-plugin.json")
|
|
err = os.WriteFile(filepath.Join(path, "pulumi-plugin.json"), bytes, 0o600)
|
|
assert.NoError(t, err, "Failed to write pulumi-plugin.json")
|
|
}
|
|
|
|
actual, err := c.Mod.getPlugin(t.TempDir())
|
|
if c.ExpectedError != "" {
|
|
assert.EqualError(t, err, c.ExpectedError)
|
|
} else {
|
|
// Kind must be resource. We can thus exclude it from the test.
|
|
if c.Expected.Kind == "" {
|
|
c.Expected.Kind = "resource"
|
|
}
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, c.Expected, actual)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestPluginsAndDependencies_moduleMode(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
root := t.TempDir()
|
|
require.NoError(t,
|
|
fsutil.CopyFile(root, filepath.Join("testdata", "sample"), nil),
|
|
"copy test data")
|
|
|
|
testPluginsAndDependencies(t, filepath.Join(root, "prog"))
|
|
}
|
|
|
|
// Test for https://github.com/pulumi/pulumi/issues/12526.
|
|
// Validates that if a Pulumi program has vendored its dependencies,
|
|
// the language host can still find the plugin and run the program.
|
|
func TestPluginsAndDependencies_vendored(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
root := t.TempDir()
|
|
require.NoError(t,
|
|
fsutil.CopyFile(root, filepath.Join("testdata", "sample"), nil),
|
|
"copy test data")
|
|
|
|
progDir := filepath.Join(root, "prog")
|
|
|
|
// Vendor the dependencies and nuke the sources
|
|
// to ensure that the language host can only use the vendored version.
|
|
cmd := exec.Command("go", "mod", "vendor")
|
|
cmd.Dir = progDir
|
|
cmd.Stdout = iotest.LogWriter(t)
|
|
cmd.Stderr = iotest.LogWriter(t)
|
|
require.NoError(t, cmd.Run(), "vendor dependencies")
|
|
require.NoError(t, os.RemoveAll(filepath.Join(root, "plugin")))
|
|
require.NoError(t, os.RemoveAll(filepath.Join(root, "dep")))
|
|
require.NoError(t, os.RemoveAll(filepath.Join(root, "indirect-dep")))
|
|
|
|
testPluginsAndDependencies(t, progDir)
|
|
}
|
|
|
|
// Regression test for https://github.com/pulumi/pulumi/issues/12963.
|
|
// Verifies that the language host can find plugins and dependencies
|
|
// when the Pulumi program is in a subdirectory of the project root.
|
|
func TestPluginsAndDependencies_subdir(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
t.Run("moduleMode", func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
root := t.TempDir()
|
|
require.NoError(t,
|
|
fsutil.CopyFile(root, filepath.Join("testdata", "sample"), nil),
|
|
"copy test data")
|
|
|
|
testPluginsAndDependencies(t, filepath.Join(root, "prog-subdir", "infra"))
|
|
})
|
|
|
|
t.Run("vendored", func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
root := t.TempDir()
|
|
require.NoError(t,
|
|
fsutil.CopyFile(root, filepath.Join("testdata", "sample"), nil),
|
|
"copy test data")
|
|
|
|
progDir := filepath.Join(root, "prog-subdir", "infra")
|
|
|
|
// Vendor the dependencies and nuke the sources
|
|
// to ensure that the language host can only use the vendored version.
|
|
cmd := exec.Command("go", "mod", "vendor")
|
|
cmd.Dir = progDir
|
|
cmd.Stdout = iotest.LogWriter(t)
|
|
cmd.Stderr = iotest.LogWriter(t)
|
|
require.NoError(t, cmd.Run(), "vendor dependencies")
|
|
require.NoError(t, os.RemoveAll(filepath.Join(root, "plugin")))
|
|
require.NoError(t, os.RemoveAll(filepath.Join(root, "dep")))
|
|
require.NoError(t, os.RemoveAll(filepath.Join(root, "indirect-dep")))
|
|
|
|
testPluginsAndDependencies(t, progDir)
|
|
})
|
|
|
|
t.Run("gowork", func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
root := t.TempDir()
|
|
require.NoError(t,
|
|
fsutil.CopyFile(root, filepath.Join("testdata", "sample"), nil),
|
|
"copy test data")
|
|
|
|
testPluginsAndDependencies(t, filepath.Join(root, "prog-gowork", "prog"))
|
|
})
|
|
}
|
|
|
|
func testPluginsAndDependencies(t *testing.T, progDir string) {
|
|
host := newLanguageHost("0.0.0.0:0", progDir, "")
|
|
ctx := context.Background()
|
|
|
|
t.Run("GetRequiredPlugins", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(ctx, 10*time.Second)
|
|
defer cancel()
|
|
|
|
res, err := host.GetRequiredPlugins(ctx, &pulumirpc.GetRequiredPluginsRequest{
|
|
Project: "deprecated",
|
|
Pwd: progDir,
|
|
Info: &pulumirpc.ProgramInfo{
|
|
RootDirectory: progDir,
|
|
ProgramDirectory: progDir,
|
|
EntryPoint: ".",
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
require.Len(t, res.Plugins, 1)
|
|
plug := res.Plugins[0]
|
|
|
|
assert.Equal(t, "example", plug.Name, "plugin name")
|
|
assert.Equal(t, "v1.2.3", plug.Version, "plugin version")
|
|
assert.Equal(t, "resource", plug.Kind, "plugin kind")
|
|
assert.Equal(t, "example.com/download", plug.Server, "plugin server")
|
|
})
|
|
|
|
t.Run("GetProgramDependencies", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(ctx, 10*time.Second)
|
|
defer cancel()
|
|
|
|
res, err := host.GetProgramDependencies(ctx, &pulumirpc.GetProgramDependenciesRequest{
|
|
Project: "deprecated",
|
|
Pwd: progDir,
|
|
TransitiveDependencies: true,
|
|
Info: &pulumirpc.ProgramInfo{
|
|
RootDirectory: progDir,
|
|
ProgramDirectory: progDir,
|
|
EntryPoint: ".",
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
gotDeps := make(map[string]string) // name => version
|
|
for _, dep := range res.Dependencies {
|
|
gotDeps[dep.Name] = dep.Version
|
|
}
|
|
|
|
assert.Equal(t, map[string]string{
|
|
"example.com/plugin": "v1.2.3",
|
|
"example.com/dep": "v1.5.0",
|
|
"example.com/indirect-dep/v2": "v2.1.0",
|
|
}, gotDeps)
|
|
})
|
|
}
|