mirror of https://github.com/pulumi/pulumi.git
2681 lines
86 KiB
Go
2681 lines
86 KiB
Go
// Copyright 2016-2018, 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 integration
|
|
|
|
import (
|
|
"context"
|
|
cryptorand "crypto/rand"
|
|
sha256 "crypto/sha256"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"hash/fnv"
|
|
"io"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"regexp"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
multierror "github.com/hashicorp/go-multierror"
|
|
"golang.org/x/mod/modfile"
|
|
"golang.org/x/mod/module"
|
|
"gopkg.in/yaml.v3"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/engine"
|
|
"github.com/pulumi/pulumi/pkg/v3/operations"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/stack"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/apitype"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/env"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/config"
|
|
ptesting "github.com/pulumi/pulumi/sdk/v3/go/common/testing"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tools"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/fsutil"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/retry"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
|
|
"github.com/pulumi/pulumi/sdk/v3/nodejs/npm"
|
|
"github.com/stretchr/testify/assert"
|
|
user "github.com/tweekmonster/luser"
|
|
)
|
|
|
|
const (
|
|
PythonRuntime = "python"
|
|
NodeJSRuntime = "nodejs"
|
|
GoRuntime = "go"
|
|
DotNetRuntime = "dotnet"
|
|
YAMLRuntime = "yaml"
|
|
JavaRuntime = "java"
|
|
)
|
|
|
|
const windowsOS = "windows"
|
|
|
|
var ErrTestFailed = errors.New("test failed")
|
|
|
|
// RuntimeValidationStackInfo contains details related to the stack that runtime validation logic may want to use.
|
|
type RuntimeValidationStackInfo struct {
|
|
StackName tokens.QName
|
|
Deployment *apitype.DeploymentV3
|
|
RootResource apitype.ResourceV3
|
|
Outputs map[string]interface{}
|
|
Events []apitype.EngineEvent
|
|
}
|
|
|
|
// EditDir is an optional edit to apply to the example, as subsequent deployments.
|
|
type EditDir struct {
|
|
Dir string
|
|
ExtraRuntimeValidation func(t *testing.T, stack RuntimeValidationStackInfo)
|
|
|
|
// Additive is true if Dir should be copied *on top* of the test directory.
|
|
// Otherwise Dir *replaces* the test directory, except we keep .pulumi/ and Pulumi.yaml and Pulumi.<stack>.yaml.
|
|
Additive bool
|
|
|
|
// ExpectFailure is true if we expect this test to fail. This is very coarse grained, and will essentially
|
|
// tolerate *any* failure in the program (IDEA: in the future, offer a way to narrow this down more).
|
|
ExpectFailure bool
|
|
|
|
// ExpectNoChanges is true if the edit is expected to not propose any changes.
|
|
ExpectNoChanges bool
|
|
|
|
// Stdout is the writer to use for all stdout messages.
|
|
Stdout io.Writer
|
|
// Stderr is the writer to use for all stderr messages.
|
|
Stderr io.Writer
|
|
// Verbose may be set to true to print messages as they occur, rather than buffering and showing upon failure.
|
|
Verbose bool
|
|
|
|
// Run program directory in query mode.
|
|
QueryMode bool
|
|
}
|
|
|
|
// TestCommandStats is a collection of data related to running a single command during a test.
|
|
type TestCommandStats struct {
|
|
// StartTime is the time at which the command was started
|
|
StartTime string `json:"startTime"`
|
|
// EndTime is the time at which the command exited
|
|
EndTime string `json:"endTime"`
|
|
// ElapsedSeconds is the time at which the command exited
|
|
ElapsedSeconds float64 `json:"elapsedSeconds"`
|
|
// StackName is the name of the stack
|
|
StackName string `json:"stackName"`
|
|
// TestId is the unique ID of the test run
|
|
TestID string `json:"testId"`
|
|
// StepName is the command line which was invoked
|
|
StepName string `json:"stepName"`
|
|
// CommandLine is the command line which was invoked
|
|
CommandLine string `json:"commandLine"`
|
|
// TestName is the name of the directory in which the test was executed
|
|
TestName string `json:"testName"`
|
|
// IsError is true if the command failed
|
|
IsError bool `json:"isError"`
|
|
// The Cloud that the test was run against, or empty for local deployments
|
|
CloudURL string `json:"cloudURL"`
|
|
}
|
|
|
|
// TestStatsReporter reports results and metadata from a test run.
|
|
type TestStatsReporter interface {
|
|
ReportCommand(stats TestCommandStats)
|
|
}
|
|
|
|
// Environment is used to create environments for use by test programs.
|
|
type Environment struct {
|
|
// The name of the environment.
|
|
Name string
|
|
// The definition of the environment.
|
|
Definition map[string]any
|
|
}
|
|
|
|
// ConfigValue is used to provide config values to a test program.
|
|
type ConfigValue struct {
|
|
// The config key to pass to `pulumi config`.
|
|
Key string
|
|
// The config value to pass to `pulumi config`.
|
|
Value string
|
|
// Secret indicates that the `--secret` flag should be specified when calling `pulumi config`.
|
|
Secret bool
|
|
// Path indicates that the `--path` flag should be specified when calling `pulumi config`.
|
|
Path bool
|
|
}
|
|
|
|
// ProgramTestOptions provides options for ProgramTest
|
|
type ProgramTestOptions struct {
|
|
// Dir is the program directory to test.
|
|
Dir string
|
|
// Array of NPM packages which must be `yarn linked` (e.g. {"pulumi", "@pulumi/aws"})
|
|
Dependencies []string
|
|
// Map of package names to versions. The test will use the specified versions of these packages instead of what
|
|
// is declared in `package.json`.
|
|
Overrides map[string]string
|
|
// Automatically use the latest dev version of pulumi SDKs if available.
|
|
InstallDevReleases bool
|
|
// List of environments to create in order.
|
|
CreateEnvironments []Environment
|
|
// List of environments to use.
|
|
Environments []string
|
|
// Map of config keys and values to set (e.g. {"aws:region": "us-east-2"}).
|
|
Config map[string]string
|
|
// Map of secure config keys and values to set (e.g. {"aws:region": "us-east-2"}).
|
|
Secrets map[string]string
|
|
// List of config keys and values to set in order, including Secret and Path options.
|
|
OrderedConfig []ConfigValue
|
|
// SecretsProvider is the optional custom secrets provider to use instead of the default.
|
|
SecretsProvider string
|
|
// EditDirs is an optional list of edits to apply to the example, as subsequent deployments.
|
|
EditDirs []EditDir
|
|
// ExtraRuntimeValidation is an optional callback for additional validation, called before applying edits.
|
|
ExtraRuntimeValidation func(t *testing.T, stack RuntimeValidationStackInfo)
|
|
// RelativeWorkDir is an optional path relative to `Dir` which should be used as working directory during tests.
|
|
RelativeWorkDir string
|
|
// AllowEmptyPreviewChanges is true if we expect that this test's no-op preview may propose changes (e.g.
|
|
// because the test is sensitive to the exact contents of its working directory and those contents change
|
|
// incidentally between the initial update and the empty update).
|
|
AllowEmptyPreviewChanges bool
|
|
// AllowEmptyUpdateChanges is true if we expect that this test's no-op update may perform changes (e.g.
|
|
// because the test is sensitive to the exact contents of its working directory and those contents change
|
|
// incidentally between the initial update and the empty update).
|
|
AllowEmptyUpdateChanges bool
|
|
// ExpectFailure is true if we expect this test to fail. This is very coarse grained, and will essentially
|
|
// tolerate *any* failure in the program (IDEA: in the future, offer a way to narrow this down more).
|
|
ExpectFailure bool
|
|
// ExpectRefreshChanges may be set to true if a test is expected to have changes yielded by an immediate refresh.
|
|
// This could occur, for example, is a resource's state is constantly changing outside of Pulumi (e.g., timestamps).
|
|
ExpectRefreshChanges bool
|
|
// RetryFailedSteps indicates that failed updates, refreshes, and destroys should be retried after a brief
|
|
// intermission. A maximum of 3 retries will be attempted.
|
|
RetryFailedSteps bool
|
|
// SkipRefresh indicates that the refresh step should be skipped entirely.
|
|
SkipRefresh bool
|
|
// Require a preview after refresh to be a no-op (expect no changes). Has no effect if SkipRefresh is true.
|
|
RequireEmptyPreviewAfterRefresh bool
|
|
// SkipPreview indicates that the preview step should be skipped entirely.
|
|
SkipPreview bool
|
|
// SkipUpdate indicates that the update step should be skipped entirely.
|
|
SkipUpdate bool
|
|
// SkipExportImport skips testing that exporting and importing the stack works properly.
|
|
SkipExportImport bool
|
|
// SkipEmptyPreviewUpdate skips the no-change preview/update that is performed that validates
|
|
// that no changes happen.
|
|
SkipEmptyPreviewUpdate bool
|
|
// SkipStackRemoval indicates that the stack should not be removed. (And so the test's results could be inspected
|
|
// in the Pulumi Service after the test has completed.)
|
|
SkipStackRemoval bool
|
|
// Destroy on cleanup defers stack destruction until the test cleanup step, rather than after
|
|
// program test execution. This is useful for more realistic stack reference testing, allowing one
|
|
// project and stack to be stood up and a second to be run before the first is destroyed.
|
|
//
|
|
// Implies NoParallel because we expect that another caller to ProgramTest will set that
|
|
DestroyOnCleanup bool
|
|
// DestroyExcludeProtected indicates that when the test stack is destroyed,
|
|
// protected resources should be excluded from the destroy operation.
|
|
DestroyExcludeProtected bool
|
|
// Quick implies SkipPreview, SkipExportImport and SkipEmptyPreviewUpdate
|
|
Quick bool
|
|
// RequireService indicates that the test must be run against the Pulumi Service
|
|
RequireService bool
|
|
// PreviewCommandlineFlags specifies flags to add to the `pulumi preview` command line (e.g. "--color=raw")
|
|
PreviewCommandlineFlags []string
|
|
// UpdateCommandlineFlags specifies flags to add to the `pulumi up` command line (e.g. "--color=raw")
|
|
UpdateCommandlineFlags []string
|
|
// QueryCommandlineFlags specifies flags to add to the `pulumi query` command line (e.g. "--color=raw")
|
|
QueryCommandlineFlags []string
|
|
// RunBuild indicates that the build step should be run (e.g. run `yarn build` for `nodejs` programs)
|
|
RunBuild bool
|
|
// RunUpdateTest will ensure that updates to the package version can test for spurious diffs
|
|
RunUpdateTest bool
|
|
// DecryptSecretsInOutput will ensure that stack output is passed `--show-secrets` parameter
|
|
// Used in conjunction with ExtraRuntimeValidation
|
|
DecryptSecretsInOutput bool
|
|
|
|
// CloudURL is an optional URL to override the default Pulumi Service API (https://api.pulumi-staging.io). The
|
|
// PULUMI_ACCESS_TOKEN environment variable must also be set to a valid access token for the target cloud.
|
|
CloudURL string
|
|
|
|
// StackName allows the stack name to be explicitly provided instead of computed from the
|
|
// environment during tests.
|
|
StackName string
|
|
|
|
// If non-empty, specifies the value of the `--tracing` flag to pass
|
|
// to Pulumi CLI, which may be a Zipkin endpoint or a
|
|
// `file:./local.trace` style url for AppDash tracing.
|
|
//
|
|
// Template `{command}` syntax will be expanded to the current
|
|
// command name such as `pulumi-stack-rm`. This is useful for
|
|
// file-based tracing since `ProgramTest` performs multiple
|
|
// CLI invocations that can inadvertently overwrite the trace
|
|
// file.
|
|
Tracing string
|
|
|
|
// NoParallel will opt the test out of being ran in parallel.
|
|
NoParallel bool
|
|
|
|
// PrePulumiCommand specifies a callback that will be executed before each `pulumi` invocation. This callback may
|
|
// optionally return another callback to be invoked after the `pulumi` invocation completes.
|
|
PrePulumiCommand func(verb string) (func(err error) error, error)
|
|
|
|
// ReportStats optionally specifies how to report results from the test for external collection.
|
|
ReportStats TestStatsReporter
|
|
|
|
// Stdout is the writer to use for all stdout messages.
|
|
Stdout io.Writer
|
|
// Stderr is the writer to use for all stderr messages.
|
|
Stderr io.Writer
|
|
// Verbose may be set to true to print messages as they occur, rather than buffering and showing upon failure.
|
|
Verbose bool
|
|
|
|
// DebugLogging may be set to anything >0 to enable excessively verbose debug logging from `pulumi`. This
|
|
// is equivalent to `--logflow --logtostderr -v=N`, where N is the value of DebugLogLevel. This may also
|
|
// be enabled by setting the environment variable PULUMI_TEST_DEBUG_LOG_LEVEL.
|
|
DebugLogLevel int
|
|
// DebugUpdates may be set to true to enable debug logging from `pulumi preview`, `pulumi up`, and
|
|
// `pulumi destroy`. This may also be enabled by setting the environment variable PULUMI_TEST_DEBUG_UPDATES.
|
|
DebugUpdates bool
|
|
|
|
// Bin is a location of a `pulumi` executable to be run. Taken from the $PATH if missing.
|
|
Bin string
|
|
// YarnBin is a location of a `yarn` executable to be run. Taken from the $PATH if missing.
|
|
YarnBin string
|
|
// GoBin is a location of a `go` executable to be run. Taken from the $PATH if missing.
|
|
GoBin string
|
|
// PythonBin is a location of a `python` executable to be run. Taken from the $PATH if missing.
|
|
PythonBin string
|
|
// PipenvBin is a location of a `pipenv` executable to run. Taken from the $PATH if missing.
|
|
PipenvBin string
|
|
// DotNetBin is a location of a `dotnet` executable to be run. Taken from the $PATH if missing.
|
|
DotNetBin string
|
|
|
|
// Additional environment variables to pass for each command we run.
|
|
Env []string
|
|
|
|
// Automatically create and use a virtual environment, rather than using the Pipenv tool. This is now the default
|
|
// behavior, so this option no longer has any affect. To go back to the old behavior use the `UsePipenv` option.
|
|
UseAutomaticVirtualEnv bool
|
|
// Use the Pipenv tool to manage the virtual environment.
|
|
UsePipenv bool
|
|
// Use a shared virtual environment for tests based on the contents of the requirements file. Defaults to false.
|
|
UseSharedVirtualEnv *bool
|
|
// Shared venv path when UseSharedVirtualEnv is true. Defaults to $HOME/.pulumi-test-venvs.
|
|
SharedVirtualEnvPath string
|
|
// Refers to the shared venv directory when UseSharedVirtualEnv is true. Otherwise defaults to venv
|
|
virtualEnvDir string
|
|
|
|
// If set, this hook is called after the `pulumi preview` command has completed.
|
|
PreviewCompletedHook func(dir string) error
|
|
|
|
// JSONOutput indicates that the `--json` flag should be passed to `up`, `preview`,
|
|
// `refresh` and `destroy` commands.
|
|
JSONOutput bool
|
|
|
|
// If set, this hook is called after `pulumi stack export` on the exported file. If `SkipExportImport` is set, this
|
|
// hook is ignored.
|
|
ExportStateValidator func(t *testing.T, stack []byte)
|
|
|
|
// If not nil, specifies the logic of preparing a project by
|
|
// ensuring dependencies. If left as nil, runs default
|
|
// preparation logic by dispatching on whether the project
|
|
// uses Node, Python, .NET or Go.
|
|
PrepareProject func(*engine.Projinfo) error
|
|
|
|
// If not nil, will be run after the project has been prepared.
|
|
PostPrepareProject func(*engine.Projinfo) error
|
|
|
|
// Array of provider plugin dependencies which come from local packages.
|
|
LocalProviders []LocalDependency
|
|
}
|
|
|
|
func (opts *ProgramTestOptions) GetUseSharedVirtualEnv() bool {
|
|
if opts.UseSharedVirtualEnv != nil {
|
|
return *opts.UseSharedVirtualEnv
|
|
}
|
|
return false
|
|
}
|
|
|
|
type LocalDependency struct {
|
|
Package string
|
|
Path string
|
|
}
|
|
|
|
func (opts *ProgramTestOptions) GetDebugLogLevel() int {
|
|
if opts.DebugLogLevel > 0 {
|
|
return opts.DebugLogLevel
|
|
}
|
|
if du := os.Getenv("PULUMI_TEST_DEBUG_LOG_LEVEL"); du != "" {
|
|
if n, e := strconv.Atoi(du); e != nil {
|
|
panic(e)
|
|
} else if n > 0 {
|
|
return n
|
|
}
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func (opts *ProgramTestOptions) GetDebugUpdates() bool {
|
|
return opts.DebugUpdates || os.Getenv("PULUMI_TEST_DEBUG_UPDATES") != ""
|
|
}
|
|
|
|
// GetStackName returns a stack name to use for this test.
|
|
func (opts *ProgramTestOptions) GetStackName() tokens.QName {
|
|
if opts.StackName == "" {
|
|
// Fetch the host and test dir names, cleaned so to contain just [a-zA-Z0-9-_] chars.
|
|
hostname, err := os.Hostname()
|
|
contract.AssertNoErrorf(err, "failure to fetch hostname for stack prefix")
|
|
var host string
|
|
for _, c := range hostname {
|
|
if len(host) >= 10 {
|
|
break
|
|
}
|
|
if (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') ||
|
|
(c >= '0' && c <= '9') || c == '-' || c == '_' {
|
|
host += string(c)
|
|
}
|
|
}
|
|
|
|
var test string
|
|
for _, c := range filepath.Base(opts.Dir) {
|
|
if len(test) >= 10 {
|
|
break
|
|
}
|
|
if (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') ||
|
|
(c >= '0' && c <= '9') || c == '-' || c == '_' {
|
|
test += string(c)
|
|
}
|
|
}
|
|
|
|
b := make([]byte, 4)
|
|
_, err = cryptorand.Read(b)
|
|
contract.AssertNoErrorf(err, "failure to generate random stack suffix")
|
|
|
|
opts.StackName = strings.ToLower("p-it-" + host + "-" + test + "-" + hex.EncodeToString(b))
|
|
}
|
|
|
|
return tokens.QName(opts.StackName)
|
|
}
|
|
|
|
// getEnvName returns the uniquified name for the given environment. The name is made unique by appending the FNV hash
|
|
// of the associated stack's name. This ensures that the name is both unique and deterministic. The name must be
|
|
// deterministic because it is computed by both LifeCycleInitialize and TestLifeCycleDestroy.
|
|
func (opts *ProgramTestOptions) getEnvName(name string) string {
|
|
h := fnv.New32()
|
|
_, err := h.Write([]byte(opts.GetStackName()))
|
|
contract.IgnoreError(err)
|
|
|
|
suffix := hex.EncodeToString(h.Sum(nil))
|
|
return fmt.Sprintf("%v-%v", name, suffix)
|
|
}
|
|
|
|
func (opts *ProgramTestOptions) getEnvNameWithOwner(name string) string {
|
|
owner := os.Getenv("PULUMI_TEST_OWNER")
|
|
if opts.RequireService && owner != "" {
|
|
return fmt.Sprintf("%v/%v", owner, opts.getEnvName(name))
|
|
}
|
|
return opts.getEnvName(name)
|
|
}
|
|
|
|
// Returns the md5 hash of the file at the given path as a string
|
|
func hashFile(path string) (string, error) {
|
|
file, err := os.Open(path)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
defer file.Close()
|
|
buf := make([]byte, 32*1024)
|
|
hash := sha256.New()
|
|
for {
|
|
n, err := file.Read(buf)
|
|
if n > 0 {
|
|
_, err := hash.Write(buf[:n])
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
}
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
}
|
|
sum := string(hash.Sum(nil))
|
|
return sum, nil
|
|
}
|
|
|
|
// GetStackNameWithOwner gets the name of the stack prepended with an owner, if PULUMI_TEST_OWNER is set.
|
|
// We use this in CI to create test stacks in an organization that all developers have access to, for debugging.
|
|
func (opts *ProgramTestOptions) GetStackNameWithOwner() tokens.QName {
|
|
owner := os.Getenv("PULUMI_TEST_OWNER")
|
|
|
|
if opts.RequireService && owner != "" {
|
|
return tokens.QName(fmt.Sprintf("%s/%s", owner, opts.GetStackName()))
|
|
}
|
|
|
|
return opts.GetStackName()
|
|
}
|
|
|
|
// With combines a source set of options with a set of overrides.
|
|
func (opts ProgramTestOptions) With(overrides ProgramTestOptions) ProgramTestOptions {
|
|
if overrides.Dir != "" {
|
|
opts.Dir = overrides.Dir
|
|
}
|
|
if overrides.Dependencies != nil {
|
|
opts.Dependencies = overrides.Dependencies
|
|
}
|
|
if overrides.Overrides != nil {
|
|
opts.Overrides = overrides.Overrides
|
|
}
|
|
if overrides.InstallDevReleases {
|
|
opts.InstallDevReleases = overrides.InstallDevReleases
|
|
}
|
|
if len(overrides.CreateEnvironments) != 0 {
|
|
opts.CreateEnvironments = append(opts.CreateEnvironments, overrides.CreateEnvironments...)
|
|
}
|
|
if len(overrides.Environments) != 0 {
|
|
opts.Environments = append(opts.Environments, overrides.Environments...)
|
|
}
|
|
for k, v := range overrides.Config {
|
|
if opts.Config == nil {
|
|
opts.Config = make(map[string]string)
|
|
}
|
|
opts.Config[k] = v
|
|
}
|
|
for k, v := range overrides.Secrets {
|
|
if opts.Secrets == nil {
|
|
opts.Secrets = make(map[string]string)
|
|
}
|
|
opts.Secrets[k] = v
|
|
}
|
|
if overrides.OrderedConfig != nil {
|
|
opts.OrderedConfig = append(opts.OrderedConfig, overrides.OrderedConfig...)
|
|
}
|
|
if overrides.SecretsProvider != "" {
|
|
opts.SecretsProvider = overrides.SecretsProvider
|
|
}
|
|
if overrides.EditDirs != nil {
|
|
opts.EditDirs = overrides.EditDirs
|
|
}
|
|
if overrides.ExtraRuntimeValidation != nil {
|
|
opts.ExtraRuntimeValidation = overrides.ExtraRuntimeValidation
|
|
}
|
|
if overrides.RelativeWorkDir != "" {
|
|
opts.RelativeWorkDir = overrides.RelativeWorkDir
|
|
}
|
|
if overrides.AllowEmptyPreviewChanges {
|
|
opts.AllowEmptyPreviewChanges = overrides.AllowEmptyPreviewChanges
|
|
}
|
|
if overrides.AllowEmptyUpdateChanges {
|
|
opts.AllowEmptyUpdateChanges = overrides.AllowEmptyUpdateChanges
|
|
}
|
|
if overrides.ExpectFailure {
|
|
opts.ExpectFailure = overrides.ExpectFailure
|
|
}
|
|
if overrides.ExpectRefreshChanges {
|
|
opts.ExpectRefreshChanges = overrides.ExpectRefreshChanges
|
|
}
|
|
if overrides.RetryFailedSteps {
|
|
opts.RetryFailedSteps = overrides.RetryFailedSteps
|
|
}
|
|
if overrides.SkipRefresh {
|
|
opts.SkipRefresh = overrides.SkipRefresh
|
|
}
|
|
if overrides.RequireEmptyPreviewAfterRefresh {
|
|
opts.RequireEmptyPreviewAfterRefresh = overrides.RequireEmptyPreviewAfterRefresh
|
|
}
|
|
if overrides.SkipPreview {
|
|
opts.SkipPreview = overrides.SkipPreview
|
|
}
|
|
if overrides.SkipUpdate {
|
|
opts.SkipUpdate = overrides.SkipUpdate
|
|
}
|
|
if overrides.SkipExportImport {
|
|
opts.SkipExportImport = overrides.SkipExportImport
|
|
}
|
|
if overrides.SkipEmptyPreviewUpdate {
|
|
opts.SkipEmptyPreviewUpdate = overrides.SkipEmptyPreviewUpdate
|
|
}
|
|
if overrides.SkipStackRemoval {
|
|
opts.SkipStackRemoval = overrides.SkipStackRemoval
|
|
}
|
|
if overrides.DestroyOnCleanup {
|
|
opts.DestroyOnCleanup = overrides.DestroyOnCleanup
|
|
}
|
|
if overrides.DestroyExcludeProtected {
|
|
opts.DestroyExcludeProtected = overrides.DestroyExcludeProtected
|
|
}
|
|
if overrides.Quick {
|
|
opts.Quick = overrides.Quick
|
|
}
|
|
if overrides.RequireService {
|
|
opts.RequireService = overrides.RequireService
|
|
}
|
|
if overrides.PreviewCommandlineFlags != nil {
|
|
opts.PreviewCommandlineFlags = append(opts.PreviewCommandlineFlags, overrides.PreviewCommandlineFlags...)
|
|
}
|
|
if overrides.UpdateCommandlineFlags != nil {
|
|
opts.UpdateCommandlineFlags = append(opts.UpdateCommandlineFlags, overrides.UpdateCommandlineFlags...)
|
|
}
|
|
if overrides.QueryCommandlineFlags != nil {
|
|
opts.QueryCommandlineFlags = append(opts.QueryCommandlineFlags, overrides.QueryCommandlineFlags...)
|
|
}
|
|
if overrides.RunBuild {
|
|
opts.RunBuild = overrides.RunBuild
|
|
}
|
|
if overrides.RunUpdateTest {
|
|
opts.RunUpdateTest = overrides.RunUpdateTest
|
|
}
|
|
if overrides.DecryptSecretsInOutput {
|
|
opts.DecryptSecretsInOutput = overrides.DecryptSecretsInOutput
|
|
}
|
|
if overrides.CloudURL != "" {
|
|
opts.CloudURL = overrides.CloudURL
|
|
}
|
|
if overrides.StackName != "" {
|
|
opts.StackName = overrides.StackName
|
|
}
|
|
if overrides.Tracing != "" {
|
|
opts.Tracing = overrides.Tracing
|
|
}
|
|
if overrides.NoParallel {
|
|
opts.NoParallel = overrides.NoParallel
|
|
}
|
|
if overrides.PrePulumiCommand != nil {
|
|
opts.PrePulumiCommand = overrides.PrePulumiCommand
|
|
}
|
|
if overrides.ReportStats != nil {
|
|
opts.ReportStats = overrides.ReportStats
|
|
}
|
|
if overrides.Stdout != nil {
|
|
opts.Stdout = overrides.Stdout
|
|
}
|
|
if overrides.Stderr != nil {
|
|
opts.Stderr = overrides.Stderr
|
|
}
|
|
if overrides.Verbose {
|
|
opts.Verbose = overrides.Verbose
|
|
}
|
|
if overrides.DebugLogLevel != 0 {
|
|
opts.DebugLogLevel = overrides.DebugLogLevel
|
|
}
|
|
if overrides.DebugUpdates {
|
|
opts.DebugUpdates = overrides.DebugUpdates
|
|
}
|
|
if overrides.Bin != "" {
|
|
opts.Bin = overrides.Bin
|
|
}
|
|
if overrides.YarnBin != "" {
|
|
opts.YarnBin = overrides.YarnBin
|
|
}
|
|
if overrides.GoBin != "" {
|
|
opts.GoBin = overrides.GoBin
|
|
}
|
|
if overrides.PipenvBin != "" {
|
|
opts.PipenvBin = overrides.PipenvBin
|
|
}
|
|
if overrides.DotNetBin != "" {
|
|
opts.DotNetBin = overrides.DotNetBin
|
|
}
|
|
if overrides.Env != nil {
|
|
opts.Env = append(opts.Env, overrides.Env...)
|
|
}
|
|
if overrides.UseAutomaticVirtualEnv {
|
|
opts.UseAutomaticVirtualEnv = overrides.UseAutomaticVirtualEnv
|
|
}
|
|
if overrides.UsePipenv {
|
|
opts.UsePipenv = overrides.UsePipenv
|
|
}
|
|
if overrides.UseSharedVirtualEnv != nil {
|
|
opts.UseSharedVirtualEnv = overrides.UseSharedVirtualEnv
|
|
}
|
|
if overrides.SharedVirtualEnvPath != "" {
|
|
opts.SharedVirtualEnvPath = overrides.SharedVirtualEnvPath
|
|
}
|
|
if overrides.PreviewCompletedHook != nil {
|
|
opts.PreviewCompletedHook = overrides.PreviewCompletedHook
|
|
}
|
|
if overrides.JSONOutput {
|
|
opts.JSONOutput = overrides.JSONOutput
|
|
}
|
|
if overrides.ExportStateValidator != nil {
|
|
opts.ExportStateValidator = overrides.ExportStateValidator
|
|
}
|
|
if overrides.PrepareProject != nil {
|
|
opts.PrepareProject = overrides.PrepareProject
|
|
}
|
|
if overrides.PostPrepareProject != nil {
|
|
opts.PostPrepareProject = overrides.PostPrepareProject
|
|
}
|
|
if overrides.LocalProviders != nil {
|
|
opts.LocalProviders = append(opts.LocalProviders, overrides.LocalProviders...)
|
|
}
|
|
return opts
|
|
}
|
|
|
|
type regexFlag struct {
|
|
re *regexp.Regexp
|
|
}
|
|
|
|
func (rf *regexFlag) String() string {
|
|
if rf.re == nil {
|
|
return ""
|
|
}
|
|
return rf.re.String()
|
|
}
|
|
|
|
func (rf *regexFlag) Set(v string) error {
|
|
r, err := regexp.Compile(v)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rf.re = r
|
|
return nil
|
|
}
|
|
|
|
var (
|
|
directoryMatcher regexFlag
|
|
listDirs bool
|
|
pipMutex *fsutil.FileMutex
|
|
)
|
|
|
|
func init() {
|
|
flag.Var(&directoryMatcher, "dirs", "optional list of regexes to use to select integration tests to run")
|
|
flag.BoolVar(&listDirs, "list-dirs", false, "list available integration tests without running them")
|
|
|
|
mutexPath := filepath.Join(os.TempDir(), "pip-mutex.lock")
|
|
pipMutex = fsutil.NewFileMutex(mutexPath)
|
|
}
|
|
|
|
// GetLogs retrieves the logs for a given stack in a particular region making the query provided.
|
|
//
|
|
// [provider] should be one of "aws" or "azure"
|
|
func GetLogs(
|
|
t *testing.T,
|
|
provider, region string,
|
|
stackInfo RuntimeValidationStackInfo,
|
|
query operations.LogQuery,
|
|
) *[]operations.LogEntry {
|
|
snap, err := stack.DeserializeDeploymentV3(
|
|
context.Background(),
|
|
*stackInfo.Deployment,
|
|
stack.DefaultSecretsProvider)
|
|
assert.NoError(t, err)
|
|
|
|
tree := operations.NewResourceTree(snap.Resources)
|
|
if !assert.NotNil(t, tree) {
|
|
return nil
|
|
}
|
|
|
|
cfg := map[config.Key]string{
|
|
config.MustMakeKey(provider, "region"): region,
|
|
}
|
|
ops := tree.OperationsProvider(cfg)
|
|
|
|
// Validate logs from example
|
|
logs, err := ops.GetLogs(query)
|
|
if !assert.NoError(t, err) {
|
|
return nil
|
|
}
|
|
|
|
return logs
|
|
}
|
|
|
|
func prepareProgram(t *testing.T, opts *ProgramTestOptions) {
|
|
// If we're just listing tests, simply print this test's directory.
|
|
if listDirs {
|
|
fmt.Printf("%s\n", opts.Dir)
|
|
}
|
|
|
|
// If we have a matcher, ensure that this test matches its pattern.
|
|
if directoryMatcher.re != nil && !directoryMatcher.re.Match([]byte(opts.Dir)) {
|
|
t.Skipf("Skipping: '%v' does not match '%v'", opts.Dir, directoryMatcher.re)
|
|
}
|
|
|
|
// Disable stack backups for tests to avoid filling up ~/.pulumi/backups with unnecessary
|
|
// backups of test stacks.
|
|
disableCheckpointBackups := env.DIYBackendDisableCheckpointBackups.Var().Name()
|
|
opts.Env = append(opts.Env, disableCheckpointBackups+"=1")
|
|
|
|
// We want tests to default into being ran in parallel, hence the odd double negative.
|
|
if !opts.NoParallel && !opts.DestroyOnCleanup {
|
|
t.Parallel()
|
|
}
|
|
|
|
if os.Getenv("PULUMI_TEST_USE_SERVICE") == "true" {
|
|
opts.RequireService = true
|
|
}
|
|
if opts.RequireService {
|
|
// This token is set in CI jobs, so this escape hatch is here to enable a smooth local dev
|
|
// experience, i.e.: running "make" and not seeing many failures due to a missing token.
|
|
if os.Getenv("PULUMI_ACCESS_TOKEN") == "" {
|
|
t.Skipf("Skipping: PULUMI_ACCESS_TOKEN is not set")
|
|
}
|
|
} else if opts.CloudURL == "" {
|
|
opts.CloudURL = MakeTempBackend(t)
|
|
}
|
|
|
|
// If the test panics, recover and log instead of letting the panic escape the test. Even though *this* test will
|
|
// have run deferred functions and cleaned up, if the panic reaches toplevel it will kill the process and prevent
|
|
// other tests running in parallel from cleaning up.
|
|
defer func() {
|
|
if failure := recover(); failure != nil {
|
|
t.Errorf("panic testing %v: %v", opts.Dir, failure)
|
|
}
|
|
}()
|
|
|
|
// Set up some default values for sending test reports and tracing data. We use environment varaiables to
|
|
// control these globally and set reasonable values for our own use in CI.
|
|
if opts.ReportStats == nil {
|
|
if v := os.Getenv("PULUMI_TEST_REPORT_CONFIG"); v != "" {
|
|
splits := strings.Split(v, ":")
|
|
if len(splits) != 3 {
|
|
t.Errorf("report config should be set to a value of the form: <aws-region>:<bucket-name>:<keyPrefix>")
|
|
}
|
|
|
|
opts.ReportStats = NewS3Reporter(splits[0], splits[1], splits[2])
|
|
}
|
|
}
|
|
|
|
if opts.Tracing == "" {
|
|
opts.Tracing = os.Getenv("PULUMI_TEST_TRACE_ENDPOINT")
|
|
}
|
|
|
|
if opts.UseSharedVirtualEnv == nil {
|
|
if sharedVenv := os.Getenv("PULUMI_TEST_PYTHON_SHARED_VENV"); sharedVenv != "" {
|
|
useSharedVenvBool := sharedVenv == "true"
|
|
opts.UseSharedVirtualEnv = &useSharedVenvBool
|
|
}
|
|
}
|
|
|
|
if opts.virtualEnvDir == "" && !opts.GetUseSharedVirtualEnv() {
|
|
opts.virtualEnvDir = "venv"
|
|
}
|
|
|
|
if opts.SharedVirtualEnvPath == "" {
|
|
opts.SharedVirtualEnvPath = filepath.Join(os.Getenv("HOME"), ".pulumi-test-venvs")
|
|
if sharedVenvPath := os.Getenv("PULUMI_TEST_PYTHON_SHARED_VENV_PATH"); sharedVenvPath != "" {
|
|
opts.SharedVirtualEnvPath = sharedVenvPath
|
|
}
|
|
}
|
|
|
|
if opts.Quick {
|
|
opts.SkipPreview = true
|
|
opts.SkipExportImport = true
|
|
opts.SkipEmptyPreviewUpdate = true
|
|
}
|
|
}
|
|
|
|
// ProgramTest runs a lifecycle of Pulumi commands in a program working directory, using the `pulumi` and `yarn`
|
|
// binaries available on PATH. It essentially executes the following workflow:
|
|
//
|
|
// yarn install
|
|
// yarn link <each opts.Depencies>
|
|
// (+) yarn run build
|
|
// pulumi init
|
|
// (*) pulumi login
|
|
// pulumi stack init integrationtesting
|
|
// pulumi config set <each opts.Config>
|
|
// pulumi config set --secret <each opts.Secrets>
|
|
// pulumi preview
|
|
// pulumi up
|
|
// pulumi stack export --file stack.json
|
|
// pulumi stack import --file stack.json
|
|
// pulumi preview (expected to be empty)
|
|
// pulumi up (expected to be empty)
|
|
// pulumi destroy --yes
|
|
// pulumi stack rm --yes integrationtesting
|
|
//
|
|
// (*) Only if PULUMI_ACCESS_TOKEN is set.
|
|
// (+) Only if `opts.RunBuild` is true.
|
|
//
|
|
// All commands must return success return codes for the test to succeed, unless ExpectFailure is true.
|
|
func ProgramTest(t *testing.T, opts *ProgramTestOptions) {
|
|
prepareProgram(t, opts)
|
|
pt := newProgramTester(t, opts)
|
|
err := pt.TestLifeCycleInitAndDestroy()
|
|
if !errors.Is(err, ErrTestFailed) {
|
|
assert.NoError(t, err)
|
|
}
|
|
}
|
|
|
|
// ProgramTestManualLifeCycle returns a ProgramTester than must be manually controlled in terms of its lifecycle
|
|
func ProgramTestManualLifeCycle(t *testing.T, opts *ProgramTestOptions) *ProgramTester {
|
|
prepareProgram(t, opts)
|
|
pt := newProgramTester(t, opts)
|
|
return pt
|
|
}
|
|
|
|
// ProgramTester contains state associated with running a single test pass.
|
|
type ProgramTester struct {
|
|
t *testing.T // the Go tester for this run.
|
|
opts *ProgramTestOptions // options that control this test run.
|
|
bin string // the `pulumi` binary we are using.
|
|
yarnBin string // the `yarn` binary we are using.
|
|
goBin string // the `go` binary we are using.
|
|
pythonBin string // the `python` binary we are using.
|
|
pipenvBin string // The `pipenv` binary we are using.
|
|
dotNetBin string // the `dotnet` binary we are using.
|
|
updateEventLog string // The path to the engine event log for `pulumi up` in this test.
|
|
maxStepTries int // The maximum number of times to retry a failed pulumi step.
|
|
tmpdir string // the temporary directory we use for our test environment
|
|
projdir string // the project directory we use for this run
|
|
TestFinished bool // whether or not the test if finished
|
|
pulumiHome string // The directory PULUMI_HOME will be set to
|
|
}
|
|
|
|
func newProgramTester(t *testing.T, opts *ProgramTestOptions) *ProgramTester {
|
|
stackName := opts.GetStackName()
|
|
maxStepTries := 1
|
|
if opts.RetryFailedSteps {
|
|
maxStepTries = 3
|
|
}
|
|
home, err := os.MkdirTemp("", "test-env-home")
|
|
assert.NoError(t, err, "creating temp PULUMI_HOME directory")
|
|
return &ProgramTester{
|
|
t: t,
|
|
opts: opts,
|
|
updateEventLog: filepath.Join(os.TempDir(), string(stackName)+"-events.json"),
|
|
maxStepTries: maxStepTries,
|
|
pulumiHome: home,
|
|
}
|
|
}
|
|
|
|
// MakeTempBackend creates a temporary backend directory which will clean up on test exit.
|
|
func MakeTempBackend(t *testing.T) string {
|
|
tempDir := t.TempDir()
|
|
return "file://" + filepath.ToSlash(tempDir)
|
|
}
|
|
|
|
func (pt *ProgramTester) GetTmpDir() string {
|
|
return pt.tmpdir
|
|
}
|
|
|
|
func (pt *ProgramTester) getBin() (string, error) {
|
|
return getCmdBin(&pt.bin, "pulumi", pt.opts.Bin)
|
|
}
|
|
|
|
func (pt *ProgramTester) getYarnBin() (string, error) {
|
|
return getCmdBin(&pt.yarnBin, "yarn", pt.opts.YarnBin)
|
|
}
|
|
|
|
func (pt *ProgramTester) getGoBin() (string, error) {
|
|
return getCmdBin(&pt.goBin, "go", pt.opts.GoBin)
|
|
}
|
|
|
|
// getPythonBin returns a path to the currently-installed `python` binary, or an error if it could not be found.
|
|
func (pt *ProgramTester) getPythonBin() (string, error) {
|
|
if pt.pythonBin == "" {
|
|
pt.pythonBin = pt.opts.PythonBin
|
|
if pt.opts.PythonBin == "" {
|
|
var err error
|
|
// Look for `python3` by default, but fallback to `python` if not found, except on Windows
|
|
// where we look for these in the reverse order because the default python.org Windows
|
|
// installation does not include a `python3` binary, and the existence of a `python3.exe`
|
|
// symlink to `python.exe` on some systems does not work correctly with the Python `venv`
|
|
// module.
|
|
pythonCmds := []string{"python3", "python"}
|
|
if runtime.GOOS == windowsOS {
|
|
pythonCmds = []string{"python", "python3"}
|
|
}
|
|
for _, bin := range pythonCmds {
|
|
pt.pythonBin, err = exec.LookPath(bin)
|
|
// Break on the first cmd we find on the path (if any).
|
|
if err == nil {
|
|
break
|
|
}
|
|
}
|
|
if err != nil {
|
|
return "", fmt.Errorf("Expected to find one of %q on $PATH: %w", pythonCmds, err)
|
|
}
|
|
}
|
|
}
|
|
return pt.pythonBin, nil
|
|
}
|
|
|
|
// getPipenvBin returns a path to the currently-installed Pipenv tool, or an error if the tool could not be found.
|
|
func (pt *ProgramTester) getPipenvBin() (string, error) {
|
|
return getCmdBin(&pt.pipenvBin, "pipenv", pt.opts.PipenvBin)
|
|
}
|
|
|
|
func (pt *ProgramTester) getDotNetBin() (string, error) {
|
|
return getCmdBin(&pt.dotNetBin, "dotnet", pt.opts.DotNetBin)
|
|
}
|
|
|
|
func (pt *ProgramTester) pulumiCmd(name string, args []string) ([]string, error) {
|
|
bin, err := pt.getBin()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
cmd := []string{bin}
|
|
if du := pt.opts.GetDebugLogLevel(); du > 0 {
|
|
cmd = append(cmd, "--logflow", "--logtostderr", "-v="+strconv.Itoa(du))
|
|
}
|
|
cmd = append(cmd, args...)
|
|
if tracing := pt.opts.Tracing; tracing != "" {
|
|
cmd = append(cmd, "--tracing", strings.ReplaceAll(tracing, "{command}", name))
|
|
}
|
|
return cmd, nil
|
|
}
|
|
|
|
func (pt *ProgramTester) yarnCmd(args []string) ([]string, error) {
|
|
bin, err := pt.getYarnBin()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
result := []string{bin}
|
|
result = append(result, args...)
|
|
return withOptionalYarnFlags(result), nil
|
|
}
|
|
|
|
func (pt *ProgramTester) pythonCmd(args []string) ([]string, error) {
|
|
bin, err := pt.getPythonBin()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cmd := []string{bin}
|
|
return append(cmd, args...), nil
|
|
}
|
|
|
|
func (pt *ProgramTester) pipenvCmd(args []string) ([]string, error) {
|
|
bin, err := pt.getPipenvBin()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cmd := []string{bin}
|
|
return append(cmd, args...), nil
|
|
}
|
|
|
|
func (pt *ProgramTester) runCommand(name string, args []string, wd string) error {
|
|
return RunCommandPulumiHome(pt.t, name, args, wd, pt.opts, pt.pulumiHome)
|
|
}
|
|
|
|
// RunPulumiCommand runs a Pulumi command in the project directory.
|
|
// For example:
|
|
//
|
|
// pt.RunPulumiCommand("preview", "--stack", "dev")
|
|
func (pt *ProgramTester) RunPulumiCommand(name string, args ...string) error {
|
|
// pt.runPulumiCommand uses 'name' for logging only.
|
|
// We want it to be part of the actual command.
|
|
args = append([]string{name}, args...)
|
|
return pt.runPulumiCommand(name, args, pt.projdir, false /* expectFailure */)
|
|
}
|
|
|
|
func (pt *ProgramTester) runPulumiCommand(name string, args []string, wd string, expectFailure bool) error {
|
|
cmd, err := pt.pulumiCmd(name, args)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var postFn func(error) error
|
|
if pt.opts.PrePulumiCommand != nil {
|
|
postFn, err = pt.opts.PrePulumiCommand(args[0])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
isUpdate := args[0] == "preview" || args[0] == "up" || args[0] == "destroy" || args[0] == "refresh"
|
|
|
|
// If we're doing a preview or an update and this project is a Python project, we need to run
|
|
// the command in the context of the virtual environment that Pipenv created in order to pick up
|
|
// the correct version of Python. We also need to do this for destroy and refresh so that
|
|
// dynamic providers are run in the right virtual environment.
|
|
// This is only necessary when not using automatic virtual environment support.
|
|
if pt.opts.UsePipenv && isUpdate {
|
|
projinfo, err := pt.getProjinfo(wd)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
|
|
if projinfo.Proj.Runtime.Name() == "python" {
|
|
pipenvBin, err := pt.getPipenvBin()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// "pipenv run" activates the current virtual environment and runs the remainder of the arguments as if it
|
|
// were a command.
|
|
cmd = append([]string{pipenvBin, "run"}, cmd...)
|
|
}
|
|
}
|
|
|
|
_, _, err = retry.Until(context.Background(), retry.Acceptor{
|
|
Accept: func(try int, nextRetryTime time.Duration) (bool, interface{}, error) {
|
|
runerr := pt.runCommand(name, cmd, wd)
|
|
if runerr == nil {
|
|
return true, nil, nil
|
|
} else if _, ok := runerr.(*exec.ExitError); ok && isUpdate && !expectFailure {
|
|
// the update command failed, let's try again, assuming we haven't failed a few times.
|
|
if try+1 >= pt.maxStepTries {
|
|
return false, nil, fmt.Errorf("%v did not succeed after %v tries", cmd, try+1)
|
|
}
|
|
|
|
pt.t.Logf("%v failed: %v; retrying...", cmd, runerr)
|
|
return false, nil, nil
|
|
}
|
|
|
|
// some other error, fail
|
|
return false, nil, runerr
|
|
},
|
|
})
|
|
if postFn != nil {
|
|
if postErr := postFn(err); postErr != nil {
|
|
return multierror.Append(err, postErr)
|
|
}
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (pt *ProgramTester) runYarnCommand(name string, args []string, wd string) error {
|
|
// Yarn will time out if multiple processes are trying to install packages at the same time.
|
|
ptesting.YarnInstallMutex.Lock()
|
|
defer ptesting.YarnInstallMutex.Unlock()
|
|
pt.t.Log("acquired yarn install lock")
|
|
defer pt.t.Log("released yarn install lock")
|
|
|
|
cmd, err := pt.yarnCmd(args)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, _, err = retry.Until(context.Background(), retry.Acceptor{
|
|
Accept: func(try int, nextRetryTime time.Duration) (bool, interface{}, error) {
|
|
runerr := pt.runCommand(name, cmd, wd)
|
|
if runerr == nil {
|
|
return true, nil, nil
|
|
} else if _, ok := runerr.(*exec.ExitError); ok {
|
|
// yarn failed, let's try again, assuming we haven't failed a few times.
|
|
if try+1 >= 3 {
|
|
return false, nil, fmt.Errorf("%v did not complete after %v tries", cmd, try+1)
|
|
}
|
|
|
|
return false, nil, nil
|
|
}
|
|
|
|
// someother error, fail
|
|
return false, nil, runerr
|
|
},
|
|
})
|
|
return err
|
|
}
|
|
|
|
func (pt *ProgramTester) runPythonCommand(name string, args []string, wd string) error {
|
|
cmd, err := pt.pythonCmd(args)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return pt.runCommand(name, cmd, wd)
|
|
}
|
|
|
|
func (pt *ProgramTester) runVirtualEnvCommand(name string, args []string, wd string) error {
|
|
// When installing with `pip install -e`, a PKG-INFO file is created. If two packages are being installed
|
|
// this way simultaneously (which happens often, when running tests), both installations will be writing the
|
|
// same file simultaneously. If one process catches "PKG-INFO" in a half-written state, the one process that
|
|
// observed the torn write will fail to install the package.
|
|
//
|
|
// To avoid this problem, we use pipMutex to explicitly serialize installation operations. Doing so avoids
|
|
// the problem of multiple processes stomping on the same files in the source tree. Note that pipMutex is a
|
|
// file mutex, so this strategy works even if the go test runner chooses to split up text execution across
|
|
// multiple processes. (Furthermore, each test gets an instance of ProgramTester and thus the mutex, so we'd
|
|
// need to be sharing the mutex globally in each test process if we weren't using the file system to lock.)
|
|
if name == "virtualenv-pip-install-package" {
|
|
if err := pipMutex.Lock(); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if pt.opts.Verbose {
|
|
pt.t.Log("acquired pip install lock")
|
|
defer pt.t.Log("released pip install lock")
|
|
}
|
|
defer func() {
|
|
if err := pipMutex.Unlock(); err != nil {
|
|
panic(err)
|
|
}
|
|
}()
|
|
}
|
|
|
|
virtualenvBinPath, err := getVirtualenvBinPath(wd, args[0], pt)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
cmd := append([]string{virtualenvBinPath}, args[1:]...)
|
|
return pt.runCommand(name, cmd, wd)
|
|
}
|
|
|
|
func (pt *ProgramTester) runPipenvCommand(name string, args []string, wd string) error {
|
|
// Pipenv uses setuptools to install and uninstall packages. Setuptools has an installation mode called "develop"
|
|
// that we use to install the package being tested, since it is 1) lightweight and 2) not doing so has its own set
|
|
// of annoying problems.
|
|
//
|
|
// Setuptools develop does three things:
|
|
// 1. It invokes the "egg_info" command in the target package,
|
|
// 2. It creates a special `.egg-link` sentinel file in the current site-packages folder, pointing to the package
|
|
// being installed's path on disk
|
|
// 3. It updates easy-install.pth in site-packages so that pip understand that this package has been installed.
|
|
//
|
|
// Steps 2 and 3 operate entirely within the context of a virtualenv. The state that they mutate is fully contained
|
|
// within the current virtualenv. However, step 1 operates in the context of the package's source tree. Egg info
|
|
// is responsible for producing a minimal "egg" for a particular package, and its largest responsibility is creating
|
|
// a PKG-INFO file for a package. PKG-INFO contains, among other things, the version of the package being installed.
|
|
//
|
|
// If two packages are being installed in "develop" mode simultaneously (which happens often, when running tests),
|
|
// both installations will run "egg_info" on the source tree and both processes will be writing the same files
|
|
// simultaneously. If one process catches "PKG-INFO" in a half-written state, the one process that observed the
|
|
// torn write will fail to install the package (setuptools crashes).
|
|
//
|
|
// To avoid this problem, we use pipMutex to explicitly serialize installation operations. Doing so avoids the
|
|
// problem of multiple processes stomping on the same files in the source tree. Note that pipMutex is a file
|
|
// mutex, so this strategy works even if the go test runner chooses to split up text execution across multiple
|
|
// processes. (Furthermore, each test gets an instance of ProgramTester and thus the mutex, so we'd need to be
|
|
// sharing the mutex globally in each test process if we weren't using the file system to lock.)
|
|
if name == "pipenv-install-package" {
|
|
if err := pipMutex.Lock(); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if pt.opts.Verbose {
|
|
pt.t.Log("acquired pip install lock")
|
|
defer pt.t.Log("released pip install lock")
|
|
}
|
|
defer func() {
|
|
if err := pipMutex.Unlock(); err != nil {
|
|
panic(err)
|
|
}
|
|
}()
|
|
}
|
|
|
|
cmd, err := pt.pipenvCmd(args)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return pt.runCommand(name, cmd, wd)
|
|
}
|
|
|
|
// TestLifeCyclePrepare prepares a test by creating a temporary directory
|
|
func (pt *ProgramTester) TestLifeCyclePrepare() error {
|
|
tmpdir, projdir, err := pt.copyTestToTemporaryDirectory()
|
|
pt.tmpdir = tmpdir
|
|
pt.projdir = projdir
|
|
return err
|
|
}
|
|
|
|
func (pt *ProgramTester) checkTestFailure() error {
|
|
if pt.t.Failed() {
|
|
pt.t.Logf("Canceling further steps due to test failure")
|
|
return ErrTestFailed
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// TestCleanUp cleans up the temporary directory that a test used
|
|
func (pt *ProgramTester) TestCleanUp() {
|
|
testFinished := pt.TestFinished
|
|
if pt.tmpdir != "" {
|
|
if !testFinished || pt.t.Failed() {
|
|
// Test aborted or failed. Maybe copy to "failed tests" directory.
|
|
failedTestsDir := os.Getenv("PULUMI_FAILED_TESTS_DIR")
|
|
if failedTestsDir != "" {
|
|
dest := filepath.Join(failedTestsDir, pt.t.Name()+uniqueSuffix())
|
|
contract.IgnoreError(fsutil.CopyFile(dest, pt.tmpdir, nil))
|
|
}
|
|
} else {
|
|
contract.IgnoreError(os.RemoveAll(pt.tmpdir))
|
|
}
|
|
} else {
|
|
// When tmpdir is empty, we ran "in tree", which means we wrote output
|
|
// to the "command-output" folder in the projdir, and we should clean
|
|
// it up if the test passed
|
|
if testFinished && !pt.t.Failed() {
|
|
contract.IgnoreError(os.RemoveAll(filepath.Join(pt.projdir, commandOutputFolderName)))
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestLifeCycleInitAndDestroy executes the test and cleans up
|
|
func (pt *ProgramTester) TestLifeCycleInitAndDestroy() error {
|
|
err := pt.TestLifeCyclePrepare()
|
|
if err != nil {
|
|
return fmt.Errorf("copying test to temp dir %s: %w", pt.tmpdir, err)
|
|
}
|
|
|
|
pt.TestFinished = false
|
|
if pt.opts.DestroyOnCleanup {
|
|
pt.t.Cleanup(pt.TestCleanUp)
|
|
} else {
|
|
defer pt.TestCleanUp()
|
|
}
|
|
|
|
err = pt.TestLifeCycleInitialize()
|
|
if err != nil {
|
|
return fmt.Errorf("initializing test project: %w", err)
|
|
}
|
|
|
|
destroyStack := func() {
|
|
destroyErr := pt.TestLifeCycleDestroy()
|
|
assert.NoError(pt.t, destroyErr)
|
|
}
|
|
if pt.opts.DestroyOnCleanup {
|
|
// Allow other tests to refer to this stack until the test is complete.
|
|
pt.t.Cleanup(destroyStack)
|
|
} else {
|
|
// Ensure that before we exit, we attempt to destroy and remove the stack.
|
|
defer destroyStack()
|
|
}
|
|
|
|
if err = pt.TestPreviewUpdateAndEdits(); err != nil {
|
|
return fmt.Errorf("running test preview, update, and edits: %w", err)
|
|
}
|
|
|
|
if pt.opts.RunUpdateTest {
|
|
err = upgradeProjectDeps(pt.projdir, pt)
|
|
if err != nil {
|
|
return fmt.Errorf("upgrading project dependencies: %w", err)
|
|
}
|
|
|
|
if err = pt.TestPreviewUpdateAndEdits(); err != nil {
|
|
return fmt.Errorf("running test preview, update, and edits (updateTest): %w", err)
|
|
}
|
|
}
|
|
|
|
pt.TestFinished = true
|
|
return nil
|
|
}
|
|
|
|
func upgradeProjectDeps(projectDir string, pt *ProgramTester) error {
|
|
projInfo, err := pt.getProjinfo(projectDir)
|
|
if err != nil {
|
|
return fmt.Errorf("getting project info: %w", err)
|
|
}
|
|
|
|
switch rt := projInfo.Proj.Runtime.Name(); rt {
|
|
case NodeJSRuntime:
|
|
if err = pt.yarnLinkPackageDeps(projectDir); err != nil {
|
|
return err
|
|
}
|
|
case PythonRuntime:
|
|
if err = pt.installPipPackageDeps(projectDir); err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
return fmt.Errorf("unrecognized project runtime: %s", rt)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// TestLifeCycleInitialize initializes the project directory and stack along with any configuration
|
|
func (pt *ProgramTester) TestLifeCycleInitialize() error {
|
|
dir := pt.projdir
|
|
stackName := pt.opts.GetStackName()
|
|
|
|
// Set the default target Pulumi API if not overridden in options.
|
|
if pt.opts.CloudURL == "" {
|
|
pulumiAPI := os.Getenv("PULUMI_API")
|
|
if pulumiAPI != "" {
|
|
pt.opts.CloudURL = pulumiAPI
|
|
}
|
|
}
|
|
|
|
// Ensure all links are present, the stack is created, and all configs are applied.
|
|
pt.t.Logf("Initializing project (dir %s; stack %s)", dir, stackName)
|
|
|
|
// Login as needed.
|
|
stackInitName := string(pt.opts.GetStackNameWithOwner())
|
|
|
|
if os.Getenv("PULUMI_ACCESS_TOKEN") == "" && pt.opts.CloudURL == "" {
|
|
fmt.Printf("Using existing logged in user for tests. Set PULUMI_ACCESS_TOKEN and/or PULUMI_API to override.\n")
|
|
} else {
|
|
// Set PulumiCredentialsPathEnvVar to our CWD, so we use credentials specific to just this
|
|
// test.
|
|
pt.opts.Env = append(pt.opts.Env, fmt.Sprintf("%s=%s", workspace.PulumiCredentialsPathEnvVar, dir))
|
|
|
|
loginArgs := []string{"login"}
|
|
loginArgs = addFlagIfNonNil(loginArgs, "--cloud-url", pt.opts.CloudURL)
|
|
|
|
// If this is a local OR cloud login, then don't attach the owner to the stack-name.
|
|
if pt.opts.CloudURL != "" {
|
|
stackInitName = string(pt.opts.GetStackName())
|
|
}
|
|
|
|
if err := pt.runPulumiCommand("pulumi-login", loginArgs, dir, false); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// Stack init
|
|
stackInitArgs := []string{"stack", "init", stackInitName}
|
|
if pt.opts.SecretsProvider != "" {
|
|
stackInitArgs = append(stackInitArgs, "--secrets-provider", pt.opts.SecretsProvider)
|
|
}
|
|
if err := pt.runPulumiCommand("pulumi-stack-init", stackInitArgs, dir, false); err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(pt.opts.Config)+len(pt.opts.Secrets) > 0 {
|
|
setAllArgs := []string{"config", "set-all"}
|
|
|
|
for key, value := range pt.opts.Config {
|
|
setAllArgs = append(setAllArgs, "--plaintext", fmt.Sprintf("%s=%s", key, value))
|
|
}
|
|
for key, value := range pt.opts.Secrets {
|
|
setAllArgs = append(setAllArgs, "--secret", fmt.Sprintf("%s=%s", key, value))
|
|
}
|
|
|
|
if err := pt.runPulumiCommand("pulumi-config", setAllArgs, dir, false); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for _, cv := range pt.opts.OrderedConfig {
|
|
configArgs := []string{"config", "set", cv.Key, cv.Value}
|
|
if cv.Secret {
|
|
configArgs = append(configArgs, "--secret")
|
|
}
|
|
if cv.Path {
|
|
configArgs = append(configArgs, "--path")
|
|
}
|
|
if err := pt.runPulumiCommand("pulumi-config", configArgs, dir, false); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// Environments
|
|
for _, env := range pt.opts.CreateEnvironments {
|
|
name := pt.opts.getEnvNameWithOwner(env.Name)
|
|
|
|
envFile, err := func() (string, error) {
|
|
temp, err := os.CreateTemp(pt.t.TempDir(), fmt.Sprintf("pulumi-env-%v-*", env.Name))
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
defer contract.IgnoreClose(temp)
|
|
|
|
enc := yaml.NewEncoder(temp)
|
|
enc.SetIndent(2)
|
|
if err = enc.Encode(env.Definition); err != nil {
|
|
return "", err
|
|
}
|
|
return temp.Name(), nil
|
|
}()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
initArgs := []string{"env", "init", name, "-f", envFile}
|
|
if err := pt.runPulumiCommand("pulumi-env-init", initArgs, dir, false); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if len(pt.opts.Environments) != 0 {
|
|
envs := make([]string, len(pt.opts.Environments))
|
|
for i, e := range pt.opts.Environments {
|
|
envs[i] = pt.opts.getEnvName(e)
|
|
}
|
|
|
|
stackFile := filepath.Join(dir, fmt.Sprintf("Pulumi.%v.yaml", stackName))
|
|
bytes, err := os.ReadFile(stackFile)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
|
|
var stack workspace.ProjectStack
|
|
if err := yaml.Unmarshal(bytes, &stack); err != nil {
|
|
return err
|
|
}
|
|
stack.Environment = workspace.NewEnvironment(envs)
|
|
|
|
bytes, err = yaml.Marshal(stack)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err = os.WriteFile(stackFile, bytes, 0o600); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// TestLifeCycleDestroy destroys a stack and removes it
|
|
func (pt *ProgramTester) TestLifeCycleDestroy() error {
|
|
if pt.projdir != "" {
|
|
// Destroy and remove the stack.
|
|
pt.t.Log("Destroying stack")
|
|
destroy := []string{"destroy", "--non-interactive", "--yes", "--skip-preview"}
|
|
if pt.opts.GetDebugUpdates() {
|
|
destroy = append(destroy, "-d")
|
|
}
|
|
if pt.opts.JSONOutput {
|
|
destroy = append(destroy, "--json")
|
|
}
|
|
if pt.opts.DestroyExcludeProtected {
|
|
destroy = append(destroy, "--exclude-protected")
|
|
}
|
|
if err := pt.runPulumiCommand("pulumi-destroy", destroy, pt.projdir, false); err != nil {
|
|
return err
|
|
}
|
|
|
|
if pt.t.Failed() {
|
|
pt.t.Logf("Test failed, retaining stack '%s'", pt.opts.GetStackNameWithOwner())
|
|
return nil
|
|
}
|
|
|
|
if !pt.opts.SkipStackRemoval {
|
|
err := pt.runPulumiCommand("pulumi-stack-rm", []string{"stack", "rm", "--yes"}, pt.projdir, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for _, env := range pt.opts.CreateEnvironments {
|
|
name := pt.opts.getEnvNameWithOwner(env.Name)
|
|
err := pt.runPulumiCommand("pulumi-env-rm", []string{"env", "rm", "--yes", name}, pt.projdir, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// TestPreviewUpdateAndEdits runs the preview, update, and any relevant edits
|
|
func (pt *ProgramTester) TestPreviewUpdateAndEdits() error {
|
|
dir := pt.projdir
|
|
// Now preview and update the real changes.
|
|
pt.t.Log("Performing primary preview and update")
|
|
initErr := pt.PreviewAndUpdate(dir, "initial", pt.opts.ExpectFailure, false, false)
|
|
|
|
// If the initial preview/update failed, just exit without trying the rest (but make sure to destroy).
|
|
if initErr != nil {
|
|
return fmt.Errorf("initial failure: %w", initErr)
|
|
}
|
|
|
|
// Perform an empty preview and update; nothing is expected to happen here.
|
|
if !pt.opts.SkipExportImport {
|
|
pt.t.Log("Roundtripping checkpoint via stack export and stack import")
|
|
|
|
if err := pt.exportImport(dir); err != nil {
|
|
return fmt.Errorf("empty preview + update: %w", err)
|
|
}
|
|
}
|
|
|
|
if !pt.opts.SkipEmptyPreviewUpdate {
|
|
msg := ""
|
|
if !pt.opts.AllowEmptyUpdateChanges {
|
|
msg = "(no changes expected)"
|
|
}
|
|
pt.t.Logf("Performing empty preview and update%s", msg)
|
|
if err := pt.PreviewAndUpdate(dir, "empty", pt.opts.ExpectFailure,
|
|
!pt.opts.AllowEmptyPreviewChanges, !pt.opts.AllowEmptyUpdateChanges); err != nil {
|
|
return fmt.Errorf("empty preview: %w", err)
|
|
}
|
|
}
|
|
|
|
// Run additional validation provided by the test options, passing in the checkpoint info.
|
|
if err := pt.performExtraRuntimeValidation(pt.opts.ExtraRuntimeValidation, dir); err != nil {
|
|
return err
|
|
}
|
|
|
|
if !pt.opts.SkipRefresh {
|
|
// Perform a refresh and ensure it doesn't yield changes.
|
|
refresh := []string{"refresh", "--non-interactive", "--yes", "--skip-preview"}
|
|
if pt.opts.GetDebugUpdates() {
|
|
refresh = append(refresh, "-d")
|
|
}
|
|
if pt.opts.JSONOutput {
|
|
refresh = append(refresh, "--json")
|
|
}
|
|
if !pt.opts.ExpectRefreshChanges {
|
|
refresh = append(refresh, "--expect-no-changes")
|
|
}
|
|
if err := pt.runPulumiCommand("pulumi-refresh", refresh, dir, false); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Perform another preview and expect no changes in it.
|
|
if pt.opts.RequireEmptyPreviewAfterRefresh {
|
|
preview := []string{"preview", "--non-interactive", "--expect-no-changes"}
|
|
if pt.opts.GetDebugUpdates() {
|
|
preview = append(preview, "-d")
|
|
}
|
|
if pt.opts.JSONOutput {
|
|
preview = append(preview, "--json")
|
|
}
|
|
if err := pt.runPulumiCommand("pulumi-preview-after-refresh", preview, dir, false); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
// If there are any edits, apply them and run a preview and update for each one.
|
|
return pt.testEdits(dir)
|
|
}
|
|
|
|
func (pt *ProgramTester) exportImport(dir string) error {
|
|
exportCmd := []string{"stack", "export", "--file", "stack.json"}
|
|
importCmd := []string{"stack", "import", "--file", "stack.json"}
|
|
|
|
defer func() {
|
|
contract.IgnoreError(os.Remove(filepath.Join(dir, "stack.json")))
|
|
}()
|
|
|
|
if err := pt.runPulumiCommand("pulumi-stack-export", exportCmd, dir, false); err != nil {
|
|
return err
|
|
}
|
|
|
|
if f := pt.opts.ExportStateValidator; f != nil {
|
|
bytes, err := os.ReadFile(filepath.Join(dir, "stack.json"))
|
|
if err != nil {
|
|
pt.t.Logf("Failed to read stack.json: %s", err)
|
|
return err
|
|
}
|
|
pt.t.Logf("Calling ExportStateValidator")
|
|
f(pt.t, bytes)
|
|
|
|
if err := pt.checkTestFailure(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return pt.runPulumiCommand("pulumi-stack-import", importCmd, dir, false)
|
|
}
|
|
|
|
// PreviewAndUpdate runs pulumi preview followed by pulumi up
|
|
func (pt *ProgramTester) PreviewAndUpdate(dir string, name string, shouldFail, expectNopPreview,
|
|
expectNopUpdate bool,
|
|
) error {
|
|
preview := []string{"preview", "--non-interactive", "--diff"}
|
|
update := []string{"up", "--non-interactive", "--yes", "--skip-preview", "--event-log", pt.updateEventLog}
|
|
if pt.opts.GetDebugUpdates() {
|
|
preview = append(preview, "-d")
|
|
update = append(update, "-d")
|
|
}
|
|
if pt.opts.JSONOutput {
|
|
preview = append(preview, "--json")
|
|
update = append(update, "--json")
|
|
}
|
|
if expectNopPreview {
|
|
preview = append(preview, "--expect-no-changes")
|
|
}
|
|
if expectNopUpdate {
|
|
update = append(update, "--expect-no-changes")
|
|
}
|
|
if pt.opts.PreviewCommandlineFlags != nil {
|
|
preview = append(preview, pt.opts.PreviewCommandlineFlags...)
|
|
}
|
|
if pt.opts.UpdateCommandlineFlags != nil {
|
|
update = append(update, pt.opts.UpdateCommandlineFlags...)
|
|
}
|
|
|
|
// If not in quick mode, run an explicit preview.
|
|
if !pt.opts.SkipPreview {
|
|
if err := pt.runPulumiCommand("pulumi-preview-"+name, preview, dir, shouldFail); err != nil {
|
|
if shouldFail {
|
|
pt.t.Log("Permitting failure (ExpectFailure=true for this preview)")
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
if pt.opts.PreviewCompletedHook != nil {
|
|
if err := pt.opts.PreviewCompletedHook(dir); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now run an update.
|
|
if !pt.opts.SkipUpdate {
|
|
if err := pt.runPulumiCommand("pulumi-update-"+name, update, dir, shouldFail); err != nil {
|
|
if shouldFail {
|
|
pt.t.Log("Permitting failure (ExpectFailure=true for this update)")
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
}
|
|
|
|
// If we expected a failure, but none occurred, return an error.
|
|
if shouldFail {
|
|
return errors.New("expected this step to fail, but it succeeded")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (pt *ProgramTester) query(dir string, name string, shouldFail bool) error {
|
|
query := []string{"query", "--non-interactive"}
|
|
if pt.opts.GetDebugUpdates() {
|
|
query = append(query, "-d")
|
|
}
|
|
if pt.opts.QueryCommandlineFlags != nil {
|
|
query = append(query, pt.opts.QueryCommandlineFlags...)
|
|
}
|
|
|
|
// Now run a query.
|
|
if err := pt.runPulumiCommand("pulumi-query-"+name, query, dir, shouldFail); err != nil {
|
|
if shouldFail {
|
|
pt.t.Log("Permitting failure (ExpectFailure=true for this update)")
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
// If we expected a failure, but none occurred, return an error.
|
|
if shouldFail {
|
|
return errors.New("expected this step to fail, but it succeeded")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (pt *ProgramTester) testEdits(dir string) error {
|
|
for i, edit := range pt.opts.EditDirs {
|
|
var err error
|
|
if err = pt.testEdit(dir, i, edit); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (pt *ProgramTester) testEdit(dir string, i int, edit EditDir) error {
|
|
pt.t.Logf("Applying edit '%v' and rerunning preview and update", edit.Dir)
|
|
|
|
if edit.Additive {
|
|
// Just copy new files into dir
|
|
if err := fsutil.CopyFile(dir, edit.Dir, nil); err != nil {
|
|
return fmt.Errorf("Couldn't copy %v into %v: %w", edit.Dir, dir, err)
|
|
}
|
|
} else {
|
|
// Create a new temporary directory
|
|
newDir, err := os.MkdirTemp("", pt.opts.StackName+"-")
|
|
if err != nil {
|
|
return fmt.Errorf("Couldn't create new temporary directory: %w", err)
|
|
}
|
|
|
|
// Delete whichever copy of the test is unused when we return
|
|
dirToDelete := newDir
|
|
defer func() {
|
|
contract.IgnoreError(os.RemoveAll(dirToDelete))
|
|
}()
|
|
|
|
// Copy everything except Pulumi.yaml, Pulumi.<stack-name>.yaml, and .pulumi from source into new directory
|
|
exclusions := make(map[string]bool)
|
|
projectYaml := workspace.ProjectFile + ".yaml"
|
|
configYaml := workspace.ProjectFile + "." + pt.opts.StackName + ".yaml"
|
|
exclusions[workspace.BookkeepingDir] = true
|
|
exclusions[projectYaml] = true
|
|
exclusions[configYaml] = true
|
|
|
|
if err := fsutil.CopyFile(newDir, edit.Dir, exclusions); err != nil {
|
|
return fmt.Errorf("Couldn't copy %v into %v: %w", edit.Dir, newDir, err)
|
|
}
|
|
|
|
// Copy Pulumi.yaml, Pulumi.<stack-name>.yaml, and .pulumi from old directory to new directory
|
|
oldProjectYaml := filepath.Join(dir, projectYaml)
|
|
newProjectYaml := filepath.Join(newDir, projectYaml)
|
|
|
|
oldConfigYaml := filepath.Join(dir, configYaml)
|
|
newConfigYaml := filepath.Join(newDir, configYaml)
|
|
|
|
oldProjectDir := filepath.Join(dir, workspace.BookkeepingDir)
|
|
newProjectDir := filepath.Join(newDir, workspace.BookkeepingDir)
|
|
|
|
if err := fsutil.CopyFile(newProjectYaml, oldProjectYaml, nil); err != nil {
|
|
return fmt.Errorf("Couldn't copy Pulumi.yaml: %w", err)
|
|
}
|
|
|
|
// Copy the config file over if it exists.
|
|
//
|
|
// Pulumi is not required to write a config file if there is no config, so
|
|
// it might not.
|
|
if _, err := os.Stat(oldConfigYaml); !os.IsNotExist(err) {
|
|
if err := fsutil.CopyFile(newConfigYaml, oldConfigYaml, nil); err != nil {
|
|
return fmt.Errorf("Couldn't copy Pulumi.%s.yaml: %w", pt.opts.StackName, err)
|
|
}
|
|
}
|
|
|
|
// Likewise, pulumi is not required to write a book-keeping (.pulumi) file.
|
|
if _, err := os.Stat(oldProjectDir); !os.IsNotExist(err) {
|
|
if err := fsutil.CopyFile(newProjectDir, oldProjectDir, nil); err != nil {
|
|
return fmt.Errorf("Couldn't copy .pulumi: %w", err)
|
|
}
|
|
}
|
|
|
|
// Finally, replace our current temp directory with the new one.
|
|
dirOld := dir + ".old"
|
|
if err := os.Rename(dir, dirOld); err != nil {
|
|
return fmt.Errorf("Couldn't rename %v to %v: %w", dir, dirOld, err)
|
|
}
|
|
|
|
// There's a brief window here where the old temp dir name could be taken from us.
|
|
|
|
if err := os.Rename(newDir, dir); err != nil {
|
|
return fmt.Errorf("Couldn't rename %v to %v: %w", newDir, dir, err)
|
|
}
|
|
|
|
// Keep dir, delete oldDir
|
|
dirToDelete = dirOld
|
|
}
|
|
|
|
err := pt.prepareProjectDir(dir)
|
|
if err != nil {
|
|
return fmt.Errorf("Couldn't prepare project in %v: %w", dir, err)
|
|
}
|
|
|
|
oldStdOut := pt.opts.Stdout
|
|
oldStderr := pt.opts.Stderr
|
|
oldVerbose := pt.opts.Verbose
|
|
if edit.Stdout != nil {
|
|
pt.opts.Stdout = edit.Stdout
|
|
}
|
|
if edit.Stderr != nil {
|
|
pt.opts.Stderr = edit.Stderr
|
|
}
|
|
if edit.Verbose {
|
|
pt.opts.Verbose = true
|
|
}
|
|
|
|
defer func() {
|
|
pt.opts.Stdout = oldStdOut
|
|
pt.opts.Stderr = oldStderr
|
|
pt.opts.Verbose = oldVerbose
|
|
}()
|
|
|
|
if !edit.QueryMode {
|
|
if err = pt.PreviewAndUpdate(dir, fmt.Sprintf("edit-%d", i),
|
|
edit.ExpectFailure, edit.ExpectNoChanges, edit.ExpectNoChanges); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if err = pt.query(dir, fmt.Sprintf("query-%d", i), edit.ExpectFailure); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return pt.performExtraRuntimeValidation(edit.ExtraRuntimeValidation, dir)
|
|
}
|
|
|
|
func (pt *ProgramTester) performExtraRuntimeValidation(
|
|
extraRuntimeValidation func(t *testing.T, stack RuntimeValidationStackInfo), dir string,
|
|
) error {
|
|
if extraRuntimeValidation == nil {
|
|
return nil
|
|
}
|
|
|
|
stackName := pt.opts.GetStackName()
|
|
|
|
// Create a temporary file name for the stack export
|
|
tempDir, err := os.MkdirTemp("", string(stackName))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fileName := filepath.Join(tempDir, "stack.json")
|
|
|
|
// Invoke `pulumi stack export`
|
|
// There are situations where we want to get access to the secrets in the validation
|
|
// this will allow us to get access to them as part of running ExtraRuntimeValidation
|
|
var pulumiCommand []string
|
|
if pt.opts.DecryptSecretsInOutput {
|
|
pulumiCommand = append(pulumiCommand, "stack", "export", "--show-secrets", "--file", fileName)
|
|
} else {
|
|
pulumiCommand = append(pulumiCommand, "stack", "export", "--file", fileName)
|
|
}
|
|
if err = pt.runPulumiCommand("pulumi-export",
|
|
pulumiCommand, dir, false); err != nil {
|
|
return fmt.Errorf("expected to export stack to file: %s: %w", fileName, err)
|
|
}
|
|
|
|
// Open the exported JSON file
|
|
f, err := os.Open(fileName)
|
|
if err != nil {
|
|
return fmt.Errorf("expected to be able to open file with stack exports: %s: %w", fileName, err)
|
|
}
|
|
defer func() {
|
|
contract.IgnoreClose(f)
|
|
contract.IgnoreError(os.RemoveAll(tempDir))
|
|
}()
|
|
|
|
// Unmarshal the Deployment
|
|
var untypedDeployment apitype.UntypedDeployment
|
|
if err = json.NewDecoder(f).Decode(&untypedDeployment); err != nil {
|
|
return err
|
|
}
|
|
var deployment apitype.DeploymentV3
|
|
if err = json.Unmarshal(untypedDeployment.Deployment, &deployment); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Get the root resource and outputs from the deployment
|
|
var rootResource apitype.ResourceV3
|
|
var outputs map[string]interface{}
|
|
for _, res := range deployment.Resources {
|
|
if res.Type == resource.RootStackType && res.Parent == "" {
|
|
rootResource = res
|
|
outputs = res.Outputs
|
|
}
|
|
}
|
|
|
|
events, err := pt.readUpdateEventLog()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Populate stack info object with all of this data to pass to the validation function
|
|
stackInfo := RuntimeValidationStackInfo{
|
|
StackName: pt.opts.GetStackName(),
|
|
Deployment: &deployment,
|
|
RootResource: rootResource,
|
|
Outputs: outputs,
|
|
Events: events,
|
|
}
|
|
|
|
pt.t.Log("Performing extra runtime validation.")
|
|
extraRuntimeValidation(pt.t, stackInfo)
|
|
pt.t.Log("Extra runtime validation complete.")
|
|
|
|
return pt.checkTestFailure()
|
|
}
|
|
|
|
func (pt *ProgramTester) readUpdateEventLog() ([]apitype.EngineEvent, error) {
|
|
events := []apitype.EngineEvent{}
|
|
eventsFile, err := os.Open(pt.updateEventLog)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return events, nil
|
|
}
|
|
return events, fmt.Errorf("expected to be able to open event log file %s: %w",
|
|
pt.updateEventLog, err)
|
|
}
|
|
|
|
defer contract.IgnoreClose(eventsFile)
|
|
|
|
decoder := json.NewDecoder(eventsFile)
|
|
for {
|
|
var event apitype.EngineEvent
|
|
if err = decoder.Decode(&event); err != nil {
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
return events, fmt.Errorf("failed decoding engine event from log file %s: %w",
|
|
pt.updateEventLog, err)
|
|
}
|
|
events = append(events, event)
|
|
}
|
|
|
|
return events, nil
|
|
}
|
|
|
|
// copyTestToTemporaryDirectory creates a temporary directory to run the test in and copies the test to it.
|
|
func (pt *ProgramTester) copyTestToTemporaryDirectory() (string, string, error) {
|
|
// Get the source dir and project info.
|
|
sourceDir := pt.opts.Dir
|
|
projSourceDir := sourceDir
|
|
if wd := pt.opts.RelativeWorkDir; wd != "" {
|
|
projSourceDir = filepath.Join(projSourceDir, wd)
|
|
}
|
|
projinfo, err := pt.getProjinfo(projSourceDir)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("could not get project info from source: %w", err)
|
|
}
|
|
|
|
if pt.opts.Stdout == nil {
|
|
pt.opts.Stdout = os.Stdout
|
|
}
|
|
if pt.opts.Stderr == nil {
|
|
pt.opts.Stderr = os.Stderr
|
|
}
|
|
|
|
pt.t.Logf("sample: %v", sourceDir)
|
|
bin, err := pt.getBin()
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
pt.t.Logf("pulumi: %v\n", bin)
|
|
|
|
stackName := string(pt.opts.GetStackName())
|
|
|
|
// For most projects, we will copy to a temporary directory. For Go projects, however, we must create
|
|
// a folder structure that adheres to GOPATH requirements
|
|
var tmpdir, projdir string
|
|
if projinfo.Proj.Runtime.Name() == "go" {
|
|
targetDir, err := tools.CreateTemporaryGoFolder("stackName")
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("Couldn't create temporary directory: %w", err)
|
|
}
|
|
tmpdir = targetDir
|
|
projdir = targetDir
|
|
} else {
|
|
targetDir, tempErr := os.MkdirTemp("", stackName+"-")
|
|
if tempErr != nil {
|
|
return "", "", fmt.Errorf("Couldn't create temporary directory: %w", tempErr)
|
|
}
|
|
tmpdir = targetDir
|
|
projdir = targetDir
|
|
}
|
|
if wd := pt.opts.RelativeWorkDir; wd != "" {
|
|
projdir = filepath.Join(projdir, wd)
|
|
}
|
|
// Copy the source project.
|
|
if copyErr := fsutil.CopyFile(tmpdir, sourceDir, nil); copyErr != nil {
|
|
return "", "", copyErr
|
|
}
|
|
// Reload the projinfo before making mutating changes (workspace.LoadProject caches the in-memory Project by path)
|
|
projinfo, err = pt.getProjinfo(projdir)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("could not get project info: %w", err)
|
|
}
|
|
|
|
// Add dynamic plugin paths from ProgramTester
|
|
if (projinfo.Proj.Plugins == nil || projinfo.Proj.Plugins.Providers == nil) && pt.opts.LocalProviders != nil {
|
|
projinfo.Proj.Plugins = &workspace.Plugins{
|
|
Providers: make([]workspace.PluginOptions, 0),
|
|
}
|
|
}
|
|
|
|
if pt.opts.LocalProviders != nil {
|
|
for _, provider := range pt.opts.LocalProviders {
|
|
// LocalProviders are relative to the working directory when running tests, NOT relative to the
|
|
// Pulumi.yaml. This is a bit odd, but makes it easier to construct the required paths in each
|
|
// test.
|
|
absPath, err := filepath.Abs(provider.Path)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("could not get absolute path for plugin %s: %w", provider.Path, err)
|
|
}
|
|
|
|
projinfo.Proj.Plugins.Providers = append(projinfo.Proj.Plugins.Providers, workspace.PluginOptions{
|
|
Name: provider.Package,
|
|
Path: absPath,
|
|
})
|
|
}
|
|
}
|
|
|
|
// Absolute path of the source directory, for fixupPath to use below
|
|
absSource, err := filepath.Abs(sourceDir)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("could not get absolute path for source directory %s: %w", sourceDir, err)
|
|
}
|
|
|
|
// Return a fixed up path if it's relative to sourceDir but not beneath it, else just returns the input
|
|
fixupPath := func(path string) (string, error) {
|
|
if filepath.IsAbs(path) {
|
|
return path, nil
|
|
}
|
|
absPlugin := filepath.Join(absSource, path)
|
|
if !strings.HasPrefix(absPlugin, absSource+string(filepath.Separator)) {
|
|
return absPlugin, nil
|
|
}
|
|
return path, nil
|
|
}
|
|
|
|
if projinfo.Proj.Plugins != nil {
|
|
optionSets := [][]workspace.PluginOptions{
|
|
projinfo.Proj.Plugins.Providers,
|
|
projinfo.Proj.Plugins.Languages,
|
|
projinfo.Proj.Plugins.Analyzers,
|
|
}
|
|
for _, options := range optionSets {
|
|
for i, opt := range options {
|
|
path, err := fixupPath(opt.Path)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("could not get fixed path for plugin %s: %w", opt.Path, err)
|
|
}
|
|
options[i].Path = path
|
|
}
|
|
}
|
|
}
|
|
projfile := filepath.Join(projdir, workspace.ProjectFile+".yaml")
|
|
bytes, err := yaml.Marshal(projinfo.Proj)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("error marshalling project %q: %w", projfile, err)
|
|
}
|
|
|
|
if err := os.WriteFile(projfile, bytes, 0o600); err != nil {
|
|
return "", "", fmt.Errorf("error writing project: %w", err)
|
|
}
|
|
|
|
err = pt.prepareProject(projinfo)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("Failed to prepare %v: %w", projdir, err)
|
|
}
|
|
|
|
if pt.opts.PostPrepareProject != nil {
|
|
err = pt.opts.PostPrepareProject(projinfo)
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("Failed to post-prepare %v: %w", projdir, err)
|
|
}
|
|
}
|
|
|
|
// TODO[pulumi/pulumi#5455]: Dynamic providers fail to load when used from multi-lang components.
|
|
// Until that's been fixed, this environment variable can be set by a test, which results in
|
|
// a package.json being emitted in the project directory and `yarn install && yarn link @pulumi/pulumi`
|
|
// being run.
|
|
// When the underlying issue has been fixed, the use of this environment variable should be removed.
|
|
var yarnLinkPulumi bool
|
|
for _, env := range pt.opts.Env {
|
|
if env == "PULUMI_TEST_YARN_LINK_PULUMI=true" {
|
|
yarnLinkPulumi = true
|
|
break
|
|
}
|
|
}
|
|
if yarnLinkPulumi {
|
|
const packageJSON = `{
|
|
"name": "test",
|
|
"peerDependencies": {
|
|
"@pulumi/pulumi": "latest"
|
|
}
|
|
}`
|
|
if err := os.WriteFile(filepath.Join(projdir, "package.json"), []byte(packageJSON), 0o600); err != nil {
|
|
return "", "", err
|
|
}
|
|
if err := pt.runYarnCommand("yarn-link", []string{"link", "@pulumi/pulumi"}, projdir); err != nil {
|
|
return "", "", err
|
|
}
|
|
if err = pt.runYarnCommand("yarn-install", []string{"install"}, projdir); err != nil {
|
|
return "", "", err
|
|
}
|
|
}
|
|
|
|
pt.t.Logf("projdir: %v", projdir)
|
|
return tmpdir, projdir, nil
|
|
}
|
|
|
|
func (pt *ProgramTester) getProjinfo(projectDir string) (*engine.Projinfo, error) {
|
|
// Load up the package so we know things like what language the project is.
|
|
projfile := filepath.Join(projectDir, workspace.ProjectFile+".yaml")
|
|
proj, err := workspace.LoadProject(projfile)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &engine.Projinfo{Proj: proj, Root: projectDir}, nil
|
|
}
|
|
|
|
// prepareProject runs setup necessary to get the project ready for `pulumi` commands.
|
|
func (pt *ProgramTester) prepareProject(projinfo *engine.Projinfo) error {
|
|
if pt.opts.PrepareProject != nil {
|
|
return pt.opts.PrepareProject(projinfo)
|
|
}
|
|
return pt.defaultPrepareProject(projinfo)
|
|
}
|
|
|
|
// prepareProjectDir runs setup necessary to get the project ready for `pulumi` commands.
|
|
func (pt *ProgramTester) prepareProjectDir(projectDir string) error {
|
|
projinfo, err := pt.getProjinfo(projectDir)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return pt.prepareProject(projinfo)
|
|
}
|
|
|
|
// prepareNodeJSProject runs setup necessary to get a Node.js project ready for `pulumi` commands.
|
|
func (pt *ProgramTester) prepareNodeJSProject(projinfo *engine.Projinfo) error {
|
|
if err := ptesting.WriteYarnRCForTest(projinfo.Root); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Get the correct pwd to run Yarn in.
|
|
cwd, _, err := projinfo.GetPwdMain()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
workspaceRoot, err := npm.FindWorkspaceRoot(cwd)
|
|
if err != nil {
|
|
if !errors.Is(err, npm.ErrNotInWorkspace) {
|
|
return err
|
|
}
|
|
// Not in a workspace, don't updated cwd.
|
|
} else {
|
|
pt.t.Logf("detected yarn/npm workspace root at %s", workspaceRoot)
|
|
cwd = workspaceRoot
|
|
}
|
|
|
|
// If dev versions were requested, we need to update the
|
|
// package.json to use them. Note that Overrides take
|
|
// priority over installing dev versions.
|
|
if pt.opts.InstallDevReleases {
|
|
err := pt.runYarnCommand("yarn-add", []string{"add", "@pulumi/pulumi@dev"}, cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// If the test requested some packages to be overridden, we do two things. First, if the package is listed as a
|
|
// direct dependency of the project, we change the version constraint in the package.json. For transitive
|
|
// dependencies, we use yarn's "resolutions" feature to force them to a specific version.
|
|
if len(pt.opts.Overrides) > 0 {
|
|
packageJSON, err := readPackageJSON(cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
resolutions := make(map[string]interface{})
|
|
|
|
for packageName, packageVersion := range pt.opts.Overrides {
|
|
for _, section := range []string{"dependencies", "devDependencies"} {
|
|
if _, has := packageJSON[section]; has {
|
|
entry := packageJSON[section].(map[string]interface{})
|
|
|
|
if _, has := entry[packageName]; has {
|
|
entry[packageName] = packageVersion
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
pt.t.Logf("adding resolution for %s to version %s", packageName, packageVersion)
|
|
resolutions["**/"+packageName] = packageVersion
|
|
}
|
|
|
|
// Wack any existing resolutions section with our newly computed one.
|
|
packageJSON["resolutions"] = resolutions
|
|
|
|
if err := writePackageJSON(cwd, packageJSON); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// Now ensure dependencies are present.
|
|
if err = pt.runYarnCommand("yarn-install", []string{"install"}, cwd); err != nil {
|
|
return err
|
|
}
|
|
|
|
if !pt.opts.RunUpdateTest {
|
|
if err = pt.yarnLinkPackageDeps(cwd); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if pt.opts.RunBuild {
|
|
// And finally compile it using whatever build steps are in the package.json file.
|
|
if err = pt.runYarnCommand("yarn-build", []string{"run", "build"}, cwd); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// readPackageJSON unmarshals the package.json file located in pathToPackage.
|
|
func readPackageJSON(pathToPackage string) (map[string]interface{}, error) {
|
|
f, err := os.Open(filepath.Join(pathToPackage, "package.json"))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("opening package.json: %w", err)
|
|
}
|
|
defer contract.IgnoreClose(f)
|
|
|
|
var ret map[string]interface{}
|
|
if err := json.NewDecoder(f).Decode(&ret); err != nil {
|
|
return nil, fmt.Errorf("decoding package.json: %w", err)
|
|
}
|
|
|
|
return ret, nil
|
|
}
|
|
|
|
func writePackageJSON(pathToPackage string, metadata map[string]interface{}) error {
|
|
// os.Create truncates the already existing file.
|
|
f, err := os.Create(filepath.Join(pathToPackage, "package.json"))
|
|
if err != nil {
|
|
return fmt.Errorf("opening package.json: %w", err)
|
|
}
|
|
defer contract.IgnoreClose(f)
|
|
|
|
encoder := json.NewEncoder(f)
|
|
encoder.SetEscapeHTML(false)
|
|
encoder.SetIndent("", " ")
|
|
if err := encoder.Encode(metadata); err != nil {
|
|
return fmt.Errorf("writing package.json: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// preparePythonProject runs setup necessary to get a Python project ready for `pulumi` commands.
|
|
func (pt *ProgramTester) preparePythonProject(projinfo *engine.Projinfo) error {
|
|
cwd, _, err := projinfo.GetPwdMain()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if pt.opts.UsePipenv {
|
|
if err = pt.preparePythonProjectWithPipenv(cwd); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
venvPath := "venv"
|
|
if cwd != projinfo.Root {
|
|
venvPath = filepath.Join(cwd, "venv")
|
|
}
|
|
|
|
if pt.opts.GetUseSharedVirtualEnv() {
|
|
requirementsPath := filepath.Join(cwd, "requirements.txt")
|
|
requirementsmd5, err := hashFile(requirementsPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
pt.opts.virtualEnvDir = fmt.Sprintf("pulumi-venv-%x", requirementsmd5)
|
|
venvPath = filepath.Join(pt.opts.SharedVirtualEnvPath, pt.opts.virtualEnvDir)
|
|
}
|
|
if err = pt.runPythonCommand("python-venv", []string{"-m", "venv", venvPath}, cwd); err != nil {
|
|
return err
|
|
}
|
|
|
|
projinfo.Proj.Runtime.SetOption("virtualenv", venvPath)
|
|
projfile := filepath.Join(projinfo.Root, workspace.ProjectFile+".yaml")
|
|
if err = projinfo.Proj.Save(projfile); err != nil {
|
|
return fmt.Errorf("saving project: %w", err)
|
|
}
|
|
|
|
if pt.opts.InstallDevReleases {
|
|
command := []string{"python", "-m", "pip", "install", "--pre", "pulumi"}
|
|
if err := pt.runVirtualEnvCommand("virtualenv-pip-install", command, cwd); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
command := []string{"python", "-m", "pip", "install", "-r", "requirements.txt"}
|
|
if err := pt.runVirtualEnvCommand("virtualenv-pip-install", command, cwd); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if !pt.opts.RunUpdateTest {
|
|
if err = pt.installPipPackageDeps(cwd); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (pt *ProgramTester) preparePythonProjectWithPipenv(cwd string) error {
|
|
// Allow ENV var based overload of desired Python version for
|
|
// the Pipenv environment. This is useful in CI scenarios that
|
|
// need to pin a specific version such as 3.9.x vs 3.10.x.
|
|
pythonVersion := os.Getenv("PYTHON_VERSION")
|
|
if pythonVersion == "" {
|
|
pythonVersion = "3"
|
|
}
|
|
|
|
// Create a new Pipenv environment. This bootstraps a new virtual environment containing the version of Python that
|
|
// we requested. Note that this version of Python is sourced from the machine, so you must first install the version
|
|
// of Python that you are requesting on the host machine before building a virtualenv for it.
|
|
|
|
if err := pt.runPipenvCommand("pipenv-new", []string{"--python", pythonVersion}, cwd); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Install the package's dependencies. We do this by running `pip` inside the virtualenv that `pipenv` has created.
|
|
// We don't use `pipenv install` because we don't want a lock file and prefer the similar model of `pip install`
|
|
// which matches what our customers do
|
|
command := []string{"run", "pip", "install", "-r", "requirements.txt"}
|
|
if pt.opts.InstallDevReleases {
|
|
command = []string{"run", "pip", "install", "--pre", "-r", "requirements.txt"}
|
|
}
|
|
err := pt.runPipenvCommand("pipenv-install", command, cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// YarnLinkPackageDeps bring in package dependencies via yarn
|
|
func (pt *ProgramTester) yarnLinkPackageDeps(cwd string) error {
|
|
for _, dependency := range pt.opts.Dependencies {
|
|
if err := pt.runYarnCommand("yarn-link", []string{"link", dependency}, cwd); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// InstallPipPackageDeps brings in package dependencies via pip install
|
|
func (pt *ProgramTester) installPipPackageDeps(cwd string) error {
|
|
var err error
|
|
for _, dep := range pt.opts.Dependencies {
|
|
// If the given filepath isn't absolute, make it absolute. We're about to pass it to pipenv and pipenv is
|
|
// operating inside of a random folder in /tmp.
|
|
if !filepath.IsAbs(dep) {
|
|
dep, err = filepath.Abs(dep)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if pt.opts.UsePipenv {
|
|
if err := pt.runPipenvCommand("pipenv-install-package",
|
|
[]string{"run", "pip", "install", "-e", dep}, cwd); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if err := pt.runVirtualEnvCommand("virtualenv-pip-install-package",
|
|
[]string{"python", "-m", "pip", "install", "-e", dep}, cwd); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func getVirtualenvBinPath(cwd, bin string, pt *ProgramTester) (string, error) {
|
|
virtualEnvBasePath := filepath.Join(cwd, pt.opts.virtualEnvDir)
|
|
if pt.opts.GetUseSharedVirtualEnv() {
|
|
virtualEnvBasePath = filepath.Join(pt.opts.SharedVirtualEnvPath, pt.opts.virtualEnvDir)
|
|
}
|
|
virtualenvBinPath := filepath.Join(virtualEnvBasePath, "bin", bin)
|
|
if runtime.GOOS == windowsOS {
|
|
virtualenvBinPath = filepath.Join(virtualEnvBasePath, "Scripts", bin+".exe")
|
|
}
|
|
if info, err := os.Stat(virtualenvBinPath); err != nil || info.IsDir() {
|
|
return "", fmt.Errorf("Expected %s to exist in virtual environment at %q", bin, virtualenvBinPath)
|
|
}
|
|
return virtualenvBinPath, nil
|
|
}
|
|
|
|
// getSanitizedPkg strips the version string from a go dep
|
|
// Note: most of the pulumi modules don't use major version subdirectories for modules
|
|
func getSanitizedModulePath(pkg string) string {
|
|
re := regexp.MustCompile(`v\d`)
|
|
v := re.FindString(pkg)
|
|
if v != "" {
|
|
return strings.TrimSuffix(strings.ReplaceAll(pkg, v, ""), "/")
|
|
}
|
|
return pkg
|
|
}
|
|
|
|
func getRewritePath(pkg string, gopath string, depRoot string) string {
|
|
var depParts []string
|
|
sanitizedPkg := getSanitizedModulePath(pkg)
|
|
|
|
splitPkg := strings.Split(sanitizedPkg, "/")
|
|
|
|
if depRoot != "" {
|
|
// Get the package name
|
|
// This is the value after "github.com/foo/bar"
|
|
repoName := splitPkg[2]
|
|
basePath := splitPkg[len(splitPkg)-1]
|
|
if basePath == repoName {
|
|
depParts = []string{depRoot, repoName}
|
|
} else {
|
|
depParts = []string{depRoot, repoName, basePath}
|
|
}
|
|
return filepath.Join(depParts...)
|
|
}
|
|
depParts = append([]string{gopath, "src"}, splitPkg...)
|
|
return filepath.Join(depParts...)
|
|
}
|
|
|
|
// Fetchs the GOPATH
|
|
func GoPath() (string, error) {
|
|
gopath := os.Getenv("GOPATH")
|
|
if gopath == "" {
|
|
usr, userErr := user.Current()
|
|
if userErr != nil {
|
|
return "", userErr
|
|
}
|
|
gopath = filepath.Join(usr.HomeDir, "go")
|
|
}
|
|
return gopath, nil
|
|
}
|
|
|
|
// prepareGoProject runs setup necessary to get a Go project ready for `pulumi` commands.
|
|
func (pt *ProgramTester) prepareGoProject(projinfo *engine.Projinfo) error {
|
|
// Go programs are compiled, so we will compile the project first.
|
|
goBin, err := pt.getGoBin()
|
|
if err != nil {
|
|
return fmt.Errorf("locating `go` binary: %w", err)
|
|
}
|
|
|
|
depRoot := os.Getenv("PULUMI_GO_DEP_ROOT")
|
|
gopath, userError := GoPath()
|
|
if userError != nil {
|
|
return userError
|
|
}
|
|
|
|
cwd, _, err := projinfo.GetPwdMain()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// initialize a go.mod for dependency resolution if one doesn't exist
|
|
_, err = os.Stat(filepath.Join(cwd, "go.mod"))
|
|
if err != nil {
|
|
err = pt.runCommand("go-mod-init", []string{goBin, "mod", "init"}, cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// install dev dependencies if requested
|
|
if pt.opts.InstallDevReleases {
|
|
// We're currently only installing pulumi/pulumi dependencies, which always have
|
|
// "master" as the default branch.
|
|
defaultBranch := "master"
|
|
err = pt.runCommand("go-get-dev-deps", []string{
|
|
goBin, "get", "-u", "github.com/pulumi/pulumi/sdk/v3@" + defaultBranch,
|
|
}, cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// link local dependencies
|
|
for _, dep := range pt.opts.Dependencies {
|
|
editStr, err := getEditStr(dep, gopath, depRoot)
|
|
if err != nil {
|
|
return fmt.Errorf("error generating go mod replacement for dep %q: %w", dep, err)
|
|
}
|
|
err = pt.runCommand("go-mod-edit", []string{goBin, "mod", "edit", "-replace", editStr}, cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// tidy to resolve all transitive dependencies including from local dependencies above.
|
|
err = pt.runCommand("go-mod-tidy", []string{goBin, "mod", "tidy"}, cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if pt.opts.RunBuild {
|
|
outBin := filepath.Join(gopath, "bin", string(projinfo.Proj.Name))
|
|
if runtime.GOOS == windowsOS {
|
|
outBin = outBin + ".exe"
|
|
}
|
|
err = pt.runCommand("go-build", []string{goBin, "build", "-o", outBin, "."}, cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = os.Stat(outBin)
|
|
if err != nil {
|
|
return fmt.Errorf("error finding built application artifact: %w", err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func getEditStr(dep string, gopath string, depRoot string) (string, error) {
|
|
checkModName := true
|
|
var err error
|
|
var replacedModName string
|
|
var targetModDir string
|
|
if strings.ContainsRune(dep, '=') {
|
|
parts := strings.Split(dep, "=")
|
|
replacedModName = parts[0]
|
|
targetModDir = parts[1]
|
|
} else if !modfile.IsDirectoryPath(dep) {
|
|
replacedModName = dep
|
|
targetModDir = getRewritePath(dep, gopath, depRoot)
|
|
} else {
|
|
targetModDir = dep
|
|
replacedModName, err = getModName(targetModDir)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
// We've read the package name from the go.mod file, skip redundant check below.
|
|
checkModName = false
|
|
}
|
|
|
|
targetModDir, err = filepath.Abs(targetModDir)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if checkModName {
|
|
targetModName, err := getModName(targetModDir)
|
|
if err != nil {
|
|
return "", fmt.Errorf("no go.mod at directory, set the path to the module explicitly or place "+
|
|
"the dependency in the path specified by PULUMI_GO_DEP_ROOT or the default GOPATH: %w", err)
|
|
}
|
|
targetPrefix, _, ok := module.SplitPathVersion(targetModName)
|
|
if !ok {
|
|
return "", fmt.Errorf("invalid module path for target module %q", targetModName)
|
|
}
|
|
replacedPrefix, _, ok := module.SplitPathVersion(replacedModName)
|
|
if !ok {
|
|
return "", fmt.Errorf("invalid module path for replaced module %q", replacedModName)
|
|
}
|
|
if targetPrefix != replacedPrefix {
|
|
return "", fmt.Errorf("found module path with prefix %s, expected %s", targetPrefix, replacedPrefix)
|
|
}
|
|
}
|
|
|
|
editStr := fmt.Sprintf("%s=%s", replacedModName, targetModDir)
|
|
return editStr, nil
|
|
}
|
|
|
|
func getModName(dir string) (string, error) {
|
|
pkgModPath := filepath.Join(dir, "go.mod")
|
|
pkgModData, err := os.ReadFile(pkgModPath)
|
|
if err != nil {
|
|
return "", fmt.Errorf("error reading go.mod at %s: %w", dir, err)
|
|
}
|
|
pkgMod, err := modfile.Parse(pkgModPath, pkgModData, nil)
|
|
if err != nil {
|
|
return "", fmt.Errorf("error parsing go.mod at %s: %w", dir, err)
|
|
}
|
|
|
|
return pkgMod.Module.Mod.Path, nil
|
|
}
|
|
|
|
// prepareDotNetProject runs setup necessary to get a .NET project ready for `pulumi` commands.
|
|
func (pt *ProgramTester) prepareDotNetProject(projinfo *engine.Projinfo) error {
|
|
dotNetBin, err := pt.getDotNetBin()
|
|
if err != nil {
|
|
return fmt.Errorf("locating `dotnet` binary: %w", err)
|
|
}
|
|
|
|
cwd, _, err := projinfo.GetPwdMain()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
localNuget := os.Getenv("PULUMI_LOCAL_NUGET")
|
|
if localNuget == "" {
|
|
home := os.Getenv("HOME")
|
|
localNuget = filepath.Join(home, ".pulumi-dev", "nuget")
|
|
}
|
|
|
|
if pt.opts.InstallDevReleases {
|
|
err = pt.runCommand("dotnet-add-package",
|
|
[]string{
|
|
dotNetBin, "add", "package", "Pulumi",
|
|
"--prerelease",
|
|
},
|
|
cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for _, dep := range pt.opts.Dependencies {
|
|
|
|
// dotnet add package requires a specific version in case of a pre-release, so we have to look it up.
|
|
globPattern := filepath.Join(localNuget, dep+".?.*.nupkg")
|
|
matches, err := filepath.Glob(globPattern)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to find a local Pulumi NuGet package: %w", err)
|
|
}
|
|
if len(matches) != 1 {
|
|
return fmt.Errorf("attempting to find a local NuGet package %s by searching %s yielded %d results: %v",
|
|
dep,
|
|
globPattern,
|
|
len(matches),
|
|
matches)
|
|
}
|
|
file := filepath.Base(matches[0])
|
|
r := strings.NewReplacer(dep+".", "", ".nupkg", "")
|
|
version := r.Replace(file)
|
|
|
|
// We don't restore because the program might depend on external
|
|
// packages which cannot be found in our local nuget source. A restore
|
|
// will happen automatically as part of the `pulumi up`.
|
|
err = pt.runCommand("dotnet-add-package",
|
|
[]string{
|
|
dotNetBin, "add", "package", dep,
|
|
"-v", version,
|
|
"-s", localNuget,
|
|
"--no-restore",
|
|
},
|
|
cwd)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to add dependency on %s: %w", dep, err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (pt *ProgramTester) prepareYAMLProject(projinfo *engine.Projinfo) error {
|
|
// YAML doesn't need any system setup, and should auto-install required plugins
|
|
return nil
|
|
}
|
|
|
|
func (pt *ProgramTester) prepareJavaProject(projinfo *engine.Projinfo) error {
|
|
// Java doesn't need any system setup, and should auto-install required plugins
|
|
return nil
|
|
}
|
|
|
|
func (pt *ProgramTester) defaultPrepareProject(projinfo *engine.Projinfo) error {
|
|
// Based on the language, invoke the right routine to prepare the target directory.
|
|
switch rt := projinfo.Proj.Runtime.Name(); rt {
|
|
case NodeJSRuntime:
|
|
return pt.prepareNodeJSProject(projinfo)
|
|
case PythonRuntime:
|
|
return pt.preparePythonProject(projinfo)
|
|
case GoRuntime:
|
|
return pt.prepareGoProject(projinfo)
|
|
case DotNetRuntime:
|
|
return pt.prepareDotNetProject(projinfo)
|
|
case YAMLRuntime:
|
|
return pt.prepareYAMLProject(projinfo)
|
|
case JavaRuntime:
|
|
return pt.prepareJavaProject(projinfo)
|
|
default:
|
|
return fmt.Errorf("unrecognized project runtime: %s", rt)
|
|
}
|
|
}
|