pulumi/pkg/testing/integration/util.go
Matt Ellis 872c7661e3 Provide a way to override packages during a test run
Add a new property to ProgramTestOptions, `Overrides` that allows a
test to request a different version of a package is used instead of
what would be listed in the package.json file.

This will be used by our nightly automation to run everything "at head"
2018-11-25 22:02:28 -08:00

137 lines
3.7 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 (
"bytes"
"io"
"io/ioutil"
"os"
"os/exec"
"path/filepath"
"strings"
"time"
"github.com/pkg/errors"
"github.com/pulumi/pulumi/pkg/resource"
"github.com/pulumi/pulumi/pkg/util/contract"
)
// DecodeMapString takes a string of the form key1=value1:key2=value2 and returns a go map.
func DecodeMapString(val string) (map[string]string, error) {
newMap := make(map[string]string)
if val != "" {
for _, overrideClause := range strings.Split(val, ":") {
data := strings.Split(overrideClause, "=")
if len(data) != 2 {
return nil, errors.Errorf(
"could not decode %s as an override, should be of the form <package>=<version>", overrideClause)
}
packageName := data[0]
packageVersion := data[1]
newMap[packageName] = packageVersion
}
}
return newMap, nil
}
// ReplaceInFile does a find and replace for a given string within a file.
func ReplaceInFile(old, new, path string) error {
rawContents, err := ioutil.ReadFile(path)
if err != nil {
return err
}
newContents := strings.Replace(string(rawContents), old, new, -1)
return ioutil.WriteFile(path, []byte(newContents), os.ModePerm)
}
// getCmdBin returns the binary named bin in location loc or, if it hasn't yet been initialized, will lazily
// populate it by either using the default def or, if empty, looking on the current $PATH.
func getCmdBin(loc *string, bin, def string) (string, error) {
if *loc == "" {
*loc = def
if *loc == "" {
var err error
*loc, err = exec.LookPath(bin)
if err != nil {
return "", errors.Wrapf(err, "Expected to find `%s` binary on $PATH", bin)
}
}
}
return *loc, nil
}
func uniqueSuffix() string {
// .<timestamp>.<five random hex characters>
timestamp := time.Now().Format("20060102-150405")
suffix, err := resource.NewUniqueHex("."+timestamp+".", 5, -1)
contract.AssertNoError(err)
return suffix
}
const (
commandOutputFolderName = "command-output"
)
func writeCommandOutput(commandName, runDir string, output []byte) (string, error) {
logFileDir := filepath.Join(runDir, commandOutputFolderName)
if err := os.MkdirAll(logFileDir, 0700); err != nil {
return "", errors.Wrapf(err, "Failed to create '%s'", logFileDir)
}
logFile := filepath.Join(logFileDir, commandName+uniqueSuffix()+".log")
if err := ioutil.WriteFile(logFile, output, 0644); err != nil {
return "", errors.Wrapf(err, "Failed to write '%s'", logFile)
}
return logFile, nil
}
type prefixer struct {
writer io.Writer
prefix []byte
anyOutput bool
}
// newPrefixer wraps an io.Writer, prepending a fixed prefix after each \n emitting on the wrapped writer
func newPrefixer(writer io.Writer, prefix string) *prefixer {
return &prefixer{writer, []byte(prefix), false}
}
var _ io.Writer = (*prefixer)(nil)
func (prefixer *prefixer) Write(p []byte) (int, error) {
n := 0
lines := bytes.SplitAfter(p, []byte{'\n'})
for _, line := range lines {
if len(line) > 0 {
_, err := prefixer.writer.Write(prefixer.prefix)
if err != nil {
return n, err
}
}
m, err := prefixer.writer.Write(line)
n += m
if err != nil {
return n, err
}
}
return n, nil
}