2018-05-22 19:43:36 +00:00
|
|
|
// 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.
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
|
|
|
|
package integration
|
|
|
|
|
|
|
|
import (
|
2020-11-18 08:56:26 +00:00
|
|
|
"fmt"
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2021-03-17 13:20:05 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/cmdutil"
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func RunCommand(t *testing.T, name string, args []string, wd string, opts *ProgramTestOptions) error {
|
2024-03-04 09:06:56 +00:00
|
|
|
return RunCommandPulumiHome(t, name, args, wd, opts, "")
|
|
|
|
}
|
|
|
|
|
|
|
|
// RunCommandPulumiHome executes the specified command and additional arguments, wrapping any output in the
|
|
|
|
// specialized test output streams that list the location the test is running in, and sets the PULUMI_HOME
|
|
|
|
// environment variable.
|
|
|
|
func RunCommandPulumiHome(
|
|
|
|
t *testing.T,
|
|
|
|
name string,
|
|
|
|
args []string,
|
|
|
|
wd string,
|
|
|
|
opts *ProgramTestOptions,
|
|
|
|
pulumiHome string,
|
|
|
|
) error {
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
path := args[0]
|
|
|
|
command := strings.Join(args, " ")
|
2020-11-17 18:19:24 +00:00
|
|
|
t.Logf("**** Invoke '%v' in '%v'", command, wd)
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
|
2018-04-04 22:31:01 +00:00
|
|
|
env := os.Environ()
|
|
|
|
if opts.Env != nil {
|
|
|
|
env = append(env, opts.Env...)
|
|
|
|
}
|
2019-04-18 22:25:15 +00:00
|
|
|
env = append(env, "PULUMI_DEBUG_COMMANDS=true")
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
env = append(env, "PULUMI_RETAIN_CHECKPOINTS=true")
|
|
|
|
env = append(env, "PULUMI_CONFIG_PASSPHRASE=correct horse battery staple")
|
2024-03-04 09:06:56 +00:00
|
|
|
if pulumiHome != "" {
|
2024-04-19 06:20:33 +00:00
|
|
|
env = append(env, "PULUMI_HOME="+pulumiHome)
|
2024-03-04 09:06:56 +00:00
|
|
|
}
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
|
|
|
|
cmd := exec.Cmd{
|
|
|
|
Path: path,
|
|
|
|
Dir: wd,
|
|
|
|
Args: args,
|
|
|
|
Env: env,
|
|
|
|
}
|
|
|
|
|
|
|
|
startTime := time.Now()
|
|
|
|
|
|
|
|
var runout []byte
|
|
|
|
var runerr error
|
|
|
|
if opts.Verbose || os.Getenv("PULUMI_VERBOSE_TEST") != "" {
|
|
|
|
cmd.Stdout = opts.Stdout
|
|
|
|
cmd.Stderr = opts.Stderr
|
|
|
|
runerr = cmd.Run()
|
|
|
|
} else {
|
|
|
|
runout, runerr = cmd.CombinedOutput()
|
|
|
|
}
|
|
|
|
|
|
|
|
endTime := time.Now()
|
|
|
|
|
|
|
|
if opts.ReportStats != nil {
|
|
|
|
// Note: This data is archived and used by external analytics tools. Take care if changing the schema or format
|
|
|
|
// of this data.
|
|
|
|
opts.ReportStats.ReportCommand(TestCommandStats{
|
|
|
|
StartTime: startTime.Format("2006/01/02 15:04:05"),
|
|
|
|
EndTime: endTime.Format("2006/01/02 15:04:05"),
|
|
|
|
ElapsedSeconds: float64((endTime.Sub(startTime)).Nanoseconds()) / 1000000000,
|
|
|
|
StepName: name,
|
|
|
|
CommandLine: command,
|
|
|
|
StackName: string(opts.GetStackName()),
|
|
|
|
TestID: wd,
|
|
|
|
TestName: filepath.Base(opts.Dir),
|
|
|
|
IsError: runerr != nil,
|
2018-01-30 21:10:32 +00:00
|
|
|
CloudURL: opts.CloudURL,
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if runerr != nil {
|
2020-11-17 18:19:24 +00:00
|
|
|
t.Logf("Invoke '%v' failed: %s\n", command, cmdutil.DetailedError(runerr))
|
2020-11-18 08:56:26 +00:00
|
|
|
|
|
|
|
if !opts.Verbose {
|
2021-09-22 17:55:20 +00:00
|
|
|
stderr := opts.Stderr
|
|
|
|
|
|
|
|
if stderr == nil {
|
|
|
|
stderr = os.Stderr
|
|
|
|
}
|
|
|
|
|
2020-11-18 08:56:26 +00:00
|
|
|
// Make sure we write the output in case of a failure to stderr so
|
|
|
|
// tests can assert the shape of the error message.
|
2021-09-22 17:55:20 +00:00
|
|
|
_, _ = fmt.Fprintf(stderr, "%s\n", string(runout))
|
2020-11-18 08:56:26 +00:00
|
|
|
}
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If we collected any program output, write it to a log file -- success or failure.
|
|
|
|
if len(runout) > 0 {
|
|
|
|
if logFile, err := writeCommandOutput(name, wd, runout); err != nil {
|
2020-11-17 18:19:24 +00:00
|
|
|
t.Logf("Failed to write output: %v", err)
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
} else {
|
2020-11-17 18:19:24 +00:00
|
|
|
t.Logf("Wrote output to %s", logFile)
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
}
|
2019-01-26 00:34:37 +00:00
|
|
|
} else {
|
2020-11-17 18:19:24 +00:00
|
|
|
t.Log("Command completed without output")
|
Restructure test framework to ease multiple languages (#799)
This change restructures the test framework code a bit, to make it
easier to introduce additional languages. Our knowledge of Yarn and
Node.js project structure, for instance, was previously baked in to
the test logic, in a way that was hard to make, for instance, Yarn
optional. (In Python, of course, it will not be used.) To better
support this, I've moved some state onto a new programTester struct
that we can use to lazily find binaries required during the testing
(such as Yarn, Pip, and so on). I'm committing this separately so
that I can minimize merge conflicts in the Python work.
2018-01-13 01:10:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return runerr
|
|
|
|
}
|
|
|
|
|
|
|
|
func withOptionalYarnFlags(args []string) []string {
|
|
|
|
flags := os.Getenv("YARNFLAGS")
|
|
|
|
|
|
|
|
if flags != "" {
|
|
|
|
return append(args, flags)
|
|
|
|
}
|
|
|
|
|
|
|
|
return args
|
|
|
|
}
|
|
|
|
|
|
|
|
// addFlagIfNonNil will take a set of command-line flags, and add a new one if the provided flag value is not empty.
|
|
|
|
func addFlagIfNonNil(args []string, flag, flagValue string) []string {
|
|
|
|
if flagValue != "" {
|
|
|
|
args = append(args, flag, flagValue)
|
|
|
|
}
|
|
|
|
return args
|
|
|
|
}
|