mirror of https://github.com/pulumi/pulumi.git
188 lines
4.8 KiB
Go
188 lines
4.8 KiB
Go
// Copyright 2016-2020, 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 test
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
"gopkg.in/yaml.v3"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/schema"
|
|
)
|
|
|
|
// GenPkgSignature corresponds to the shape of the codegen GeneratePackage functions.
|
|
type GenPkgSignature func(string, *schema.Package, map[string][]byte) (map[string][]byte, error)
|
|
|
|
// GeneratePackageFilesFromSchema loads a schema and generates files using the provided GeneratePackage function.
|
|
func GeneratePackageFilesFromSchema(schemaPath string, genPackageFunc GenPkgSignature) (map[string][]byte, error) {
|
|
// Read in, decode, and import the schema.
|
|
schemaBytes, err := ioutil.ReadFile(schemaPath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ext := filepath.Ext(schemaPath)
|
|
|
|
var pkgSpec schema.PackageSpec
|
|
if ext == ".yaml" || ext == ".yml" {
|
|
err = yaml.Unmarshal(schemaBytes, &pkgSpec)
|
|
} else {
|
|
err = json.Unmarshal(schemaBytes, &pkgSpec)
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pkg, err := schema.ImportSpec(pkgSpec, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return genPackageFunc("test", pkg, nil)
|
|
}
|
|
|
|
// LoadFiles loads the provided list of files from a directory.
|
|
func LoadFiles(dir, lang string, files []string) (map[string][]byte, error) {
|
|
result := map[string][]byte{}
|
|
for _, file := range files {
|
|
fileBytes, err := ioutil.ReadFile(filepath.Join(dir, lang, file))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
result[file] = fileBytes
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
func loadDirectory(fs map[string][]byte, root, path string) error {
|
|
entries, err := os.ReadDir(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, e := range entries {
|
|
entryPath := filepath.Join(path, e.Name())
|
|
if e.IsDir() {
|
|
if err = loadDirectory(fs, root, entryPath); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
contents, err := os.ReadFile(entryPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
name := filepath.ToSlash(entryPath[len(root)+1:])
|
|
|
|
fs[name] = contents
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// LoadBaseline loads the contents of the given baseline directory.
|
|
func LoadBaseline(dir, lang string) (map[string][]byte, error) {
|
|
dir = filepath.Join(dir, lang)
|
|
|
|
fs := map[string][]byte{}
|
|
if err := loadDirectory(fs, dir, dir); err != nil {
|
|
return nil, err
|
|
}
|
|
return fs, nil
|
|
}
|
|
|
|
// ValidateFileEquality compares maps of files for equality.
|
|
func ValidateFileEquality(t *testing.T, actual, expected map[string][]byte) bool {
|
|
ok := true
|
|
for name, file := range expected {
|
|
if !assert.Contains(t, actual, name) || !assert.Equal(t, string(file), string(actual[name]), name) {
|
|
ok = false
|
|
}
|
|
}
|
|
for name := range actual {
|
|
if _, has := expected[name]; !has {
|
|
t.Logf("missing data for %s", name)
|
|
ok = false
|
|
}
|
|
}
|
|
return ok
|
|
}
|
|
|
|
// If PULUMI_ACCEPT is set, writes out actual output to the expected
|
|
// file set, so we can continue enjoying golden tests without manually
|
|
// modifying the expected output.
|
|
func RewriteFilesWhenPulumiAccept(t *testing.T, dir, lang string, actual map[string][]byte) bool {
|
|
if os.Getenv("PULUMI_ACCEPT") == "" {
|
|
return false
|
|
}
|
|
|
|
baseline := filepath.Join(dir, lang)
|
|
|
|
// Remove the baseline directory's current contents.
|
|
entries, err := os.ReadDir(baseline)
|
|
switch {
|
|
case err == nil:
|
|
for _, e := range entries {
|
|
err = os.RemoveAll(filepath.Join(baseline, e.Name()))
|
|
require.NoError(t, err)
|
|
}
|
|
case os.IsNotExist(err):
|
|
// OK
|
|
default:
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
for file, bytes := range actual {
|
|
relPath := filepath.FromSlash(file)
|
|
path := filepath.Join(dir, lang, relPath)
|
|
|
|
if err = os.MkdirAll(filepath.Dir(path), 0755); err != nil && !os.IsExist(err) {
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
err = ioutil.WriteFile(path, bytes, 0600)
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// CheckAllFilesGenerated ensures that the set of expected and actual files generated
|
|
// are exactly equivalent.
|
|
func CheckAllFilesGenerated(t *testing.T, actual, expected map[string][]byte) {
|
|
seen := map[string]bool{}
|
|
for x := range expected {
|
|
seen[x] = true
|
|
}
|
|
for a := range actual {
|
|
assert.Contains(t, seen, a, "Unexpected file generated: %s", a)
|
|
if seen[a] {
|
|
delete(seen, a)
|
|
}
|
|
}
|
|
|
|
for s := range seen {
|
|
assert.Fail(t, "No content generated for expected file %s", s)
|
|
}
|
|
}
|