pulumi/pkg/codegen/internal/test/program_driver.go
2021-09-21 10:00:44 -07:00

203 lines
4.6 KiB
Go

package test
import (
"bytes"
"io/ioutil"
"os"
"path/filepath"
"strings"
"testing"
"github.com/hashicorp/hcl/v2"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/pulumi/pulumi/pkg/v3/codegen"
"github.com/pulumi/pulumi/pkg/v3/codegen/hcl2"
"github.com/pulumi/pulumi/pkg/v3/codegen/hcl2/syntax"
"github.com/pulumi/pulumi/pkg/v3/codegen/internal/utils"
)
type programTest struct {
ProgramFile string
Description string
Skip codegen.StringSet
ExpectNYIDiags codegen.StringSet
}
var testdataPath = filepath.Join("..", "internal", "test", "testdata")
var programTests = []programTest{
{
ProgramFile: "aws-s3-folder",
Description: "AWS S3 Folder",
ExpectNYIDiags: codegen.NewStringSet("python", "nodejs", "dotnet"),
},
{
ProgramFile: "aws-eks",
Description: "AWS EKS",
},
{
ProgramFile: "aws-fargate",
Description: "AWS Fargate",
},
{
ProgramFile: "aws-s3-logging",
Description: "AWS S3 with logging",
},
{
ProgramFile: "aws-webserver",
Description: "AWS Webserver",
},
{
ProgramFile: "azure-native",
Description: "Azure Native",
Skip: codegen.NewStringSet("go"),
},
{
ProgramFile: "azure-sa",
Description: "Azure SA",
},
{
ProgramFile: "kubernetes-operator",
Description: "K8s Operator",
},
{
ProgramFile: "kubernetes-pod",
Description: "K8s Pod",
},
{
ProgramFile: "kubernetes-template",
Description: "K8s Template",
},
{
ProgramFile: "random-pet",
Description: "Random Pet",
},
{
ProgramFile: "resource-options",
Description: "Resource Options",
},
{
ProgramFile: "secret",
Description: "Secret",
},
{
ProgramFile: "functions",
Description: "Functions",
},
}
type langConfig struct {
extension string
outputFile string
}
// TestProgramCodegen runs the complete set of program code generation tests against a particular
// language's code generator.
//
// A program code generation test consists of a PCL file (.pp extension) and a set of expected outputs
// for each language.
//
// The PCL file is the only piece that must be manually authored. Once the schema has been written, the expected outputs
// can be generated by running `PULUMI_ACCEPT=true go test ./..." from the `pkg/codegen` directory.
//nolint: revive
func TestProgramCodegen(
t *testing.T,
language string,
genProgram func(program *hcl2.Program) (map[string][]byte, hcl.Diagnostics, error),
) {
for _, tt := range programTests {
t.Run(tt.Description, func(t *testing.T) {
if tt.Skip.Has(language) {
t.Skip()
return
}
expectNYIDiags := false
if tt.ExpectNYIDiags.Has(language) {
expectNYIDiags = true
}
var cfg langConfig
switch language {
case "python":
cfg = langConfig{
extension: "py",
outputFile: "__main__.py",
}
case "nodejs":
cfg = langConfig{
extension: "ts",
outputFile: "index.ts",
}
case "go":
cfg = langConfig{
extension: "go",
outputFile: "main.go",
}
case "dotnet":
cfg = langConfig{
extension: "cs",
outputFile: "MyStack.cs",
}
default:
t.Fatalf("language %s not recognized", language)
}
pclFile := filepath.Join(testdataPath, tt.ProgramFile+".pp")
contents, err := ioutil.ReadFile(pclFile)
if err != nil {
t.Fatalf("could not read %v: %v", pclFile, err)
}
expectedFile := pclFile + "." + cfg.extension
expected, err := ioutil.ReadFile(expectedFile)
if err != nil && os.Getenv("PULUMI_ACCEPT") == "" {
t.Fatalf("could not read %v: %v", expectedFile, err)
}
parser := syntax.NewParser()
err = parser.ParseFile(bytes.NewReader(contents), tt.ProgramFile+".pp")
if err != nil {
t.Fatalf("could not read %v: %v", pclFile, err)
}
if parser.Diagnostics.HasErrors() {
t.Fatalf("failed to parse files: %v", parser.Diagnostics)
}
program, diags, err := hcl2.BindProgram(parser.Files, hcl2.PluginHost(utils.NewHost(testdataPath)))
if err != nil {
t.Fatalf("could not bind program: %v", err)
}
if diags.HasErrors() {
t.Fatalf("failed to bind program: %v", diags)
}
files, diags, err := genProgram(program)
assert.NoError(t, err)
if expectNYIDiags {
var tmpDiags hcl.Diagnostics
for _, d := range diags {
if !strings.HasPrefix(d.Summary, "not yet implemented") {
tmpDiags = append(tmpDiags, d)
}
}
diags = tmpDiags
}
if diags.HasErrors() {
t.Fatalf("failed to generate program: %v", diags)
}
if os.Getenv("PULUMI_ACCEPT") != "" {
err := ioutil.WriteFile(expectedFile, files[cfg.outputFile], 0600)
require.NoError(t, err)
return
}
assert.Equal(t, string(expected), string(files[cfg.outputFile]))
})
}
}