mirror of https://github.com/pulumi/pulumi.git
542 lines
17 KiB
Go
542 lines
17 KiB
Go
package test
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen"
|
|
)
|
|
|
|
// Defines an extra check logic that accepts the directory with the
|
|
// generated code, typically `$TestDir/$test.Directory/$language`.
|
|
type CodegenCheck func(t *testing.T, codedir string)
|
|
|
|
type SDKTest struct {
|
|
Directory string
|
|
Description string
|
|
|
|
// Extra checks for this test. They keys of this map
|
|
// are of the form "$language/$check" such as "go/compile".
|
|
Checks map[string]CodegenCheck
|
|
|
|
// Skip checks, identified by "$language/$check".
|
|
// "$language/any" is special, skipping generating the
|
|
// code as well as all tests.
|
|
Skip codegen.StringSet
|
|
|
|
// Do not compile the generated code for the languages in this set.
|
|
// This is a helper form of `Skip`.
|
|
SkipCompileCheck codegen.StringSet
|
|
|
|
// Mutex to ensure only a single test operates on directory at a time
|
|
Mutex sync.Mutex
|
|
}
|
|
|
|
// ShouldSkipTest indicates if a given test for a given language should be run.
|
|
func (tt *SDKTest) ShouldSkipTest(language, test string) bool {
|
|
// Only language-specific checks.
|
|
if !strings.HasPrefix(test, language+"/") {
|
|
return true
|
|
}
|
|
|
|
// Obey SkipCompileCheck to skip compile and test targets.
|
|
if tt.SkipCompileCheck != nil &&
|
|
tt.SkipCompileCheck.Has(language) &&
|
|
(test == fmt.Sprintf("%s/compile", language) ||
|
|
test == fmt.Sprintf("%s/test", language)) {
|
|
return true
|
|
}
|
|
|
|
// Obey Skip.
|
|
if tt.Skip != nil && tt.Skip.Has(test) {
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// ShouldSkipCodegen determines if codegen should be run. ShouldSkipCodegen=true
|
|
// further implies no other tests will be run.
|
|
func (tt *SDKTest) ShouldSkipCodegen(language string) bool {
|
|
return tt.Skip.Has(language + "/any")
|
|
}
|
|
|
|
const (
|
|
python = "python"
|
|
nodejs = "nodejs"
|
|
dotnet = "dotnet"
|
|
golang = "go"
|
|
)
|
|
|
|
var allLanguages = codegen.NewStringSet("python/any", "nodejs/any", "dotnet/any", "go/any", "docs/any")
|
|
|
|
var PulumiPulumiSDKTests = []*SDKTest{
|
|
{
|
|
Directory: "naming-collisions",
|
|
Description: "Schema with types that could potentially produce collisions.",
|
|
},
|
|
{
|
|
Directory: "dash-named-schema",
|
|
Description: "Simple schema with a two part name (foo-bar)",
|
|
},
|
|
{
|
|
Directory: "external-resource-schema",
|
|
Description: "External resource schema",
|
|
SkipCompileCheck: codegen.NewStringSet(golang),
|
|
},
|
|
{
|
|
Directory: "nested-module",
|
|
Description: "Nested module",
|
|
},
|
|
{
|
|
Directory: "simplified-invokes",
|
|
Description: "Simplified invokes",
|
|
Skip: codegen.NewStringSet("python/any", "go/any"),
|
|
},
|
|
{
|
|
Directory: "nested-module-thirdparty",
|
|
Description: "Third-party nested module",
|
|
},
|
|
{
|
|
Directory: "plain-schema-gh6957",
|
|
Description: "Repro for #6957",
|
|
},
|
|
{
|
|
Directory: "resource-args-python-case-insensitive",
|
|
Description: "Resource args with same named resource and type case insensitive",
|
|
},
|
|
{
|
|
Directory: "resource-args-python",
|
|
Description: "Resource args with same named resource and type",
|
|
},
|
|
{
|
|
Directory: "simple-enum-schema",
|
|
Description: "Simple schema with enum types",
|
|
},
|
|
{
|
|
Directory: "simple-plain-schema",
|
|
Description: "Simple schema with plain properties",
|
|
},
|
|
{
|
|
Directory: "simple-plain-schema-with-root-package",
|
|
Description: "Simple schema with root package set",
|
|
},
|
|
{
|
|
Directory: "simple-schema-pyproject",
|
|
Description: "A simple schema that generates a pyproject.toml file",
|
|
Skip: codegen.NewStringSet("go/any", "nodejs/any", "dotnet/any"),
|
|
},
|
|
{
|
|
Directory: "simple-resource-schema",
|
|
Description: "Simple schema with local resource properties",
|
|
},
|
|
{
|
|
Directory: "simple-resource-schema-custom-pypackage-name",
|
|
Description: "Simple schema with local resource properties and custom Python package name",
|
|
},
|
|
{
|
|
Directory: "simple-methods-schema",
|
|
Description: "Simple schema with methods",
|
|
SkipCompileCheck: codegen.NewStringSet(nodejs, golang),
|
|
},
|
|
{
|
|
Directory: "simple-methods-schema-single-value-returns",
|
|
Description: "Simple schema with methods that return single values",
|
|
},
|
|
{
|
|
Directory: "simple-yaml-schema",
|
|
Description: "Simple schema encoded using YAML",
|
|
},
|
|
{
|
|
Directory: "provider-config-schema",
|
|
Description: "Simple provider config schema",
|
|
// For golang skip check, see https://github.com/pulumi/pulumi/issues/11567
|
|
SkipCompileCheck: codegen.NewStringSet(dotnet, golang),
|
|
},
|
|
{
|
|
Directory: "replace-on-change",
|
|
Description: "Simple use of replaceOnChange in schema",
|
|
},
|
|
{
|
|
Directory: "resource-property-overlap",
|
|
Description: "A resource with the same name as its property",
|
|
SkipCompileCheck: codegen.NewStringSet(dotnet, nodejs),
|
|
},
|
|
{
|
|
Directory: "hyphen-url",
|
|
Description: "A resource url with a hyphen in its path",
|
|
Skip: codegen.NewStringSet("go/any"),
|
|
},
|
|
{
|
|
Directory: "output-funcs",
|
|
Description: "Tests targeting the $fn_output helper code generation feature",
|
|
},
|
|
{
|
|
Directory: "output-funcs-edgeorder",
|
|
Description: "Regresses Node compilation issues on a subset of azure-native",
|
|
SkipCompileCheck: codegen.NewStringSet(golang, python),
|
|
Skip: codegen.NewStringSet("nodejs/test"),
|
|
},
|
|
{
|
|
Directory: "output-funcs-tfbridge20",
|
|
Description: "Similar to output-funcs, but with compatibility: tfbridge20, to simulate pulumi-aws use case",
|
|
SkipCompileCheck: codegen.NewStringSet(python),
|
|
},
|
|
{
|
|
Directory: "cyclic-types",
|
|
Description: "Cyclic object types",
|
|
},
|
|
{
|
|
Directory: "regress-node-8110",
|
|
Description: "Test the fix for pulumi/pulumi#8110 nodejs compilation error",
|
|
Skip: codegen.NewStringSet("go/test", "dotnet/test"),
|
|
},
|
|
{
|
|
Directory: "dashed-import-schema",
|
|
Description: "Ensure that we handle all valid go import paths",
|
|
Skip: codegen.NewStringSet("go/test", "dotnet/test"),
|
|
},
|
|
{
|
|
Directory: "plain-and-default",
|
|
Description: "Ensure that a resource with a plain default property works correctly",
|
|
},
|
|
{
|
|
Directory: "plain-object-defaults",
|
|
Description: "Ensure that object defaults are generated (repro #8132)",
|
|
},
|
|
{
|
|
Directory: "plain-object-disable-defaults",
|
|
Description: "Ensure that we can still compile safely when defaults are disabled",
|
|
},
|
|
{
|
|
Directory: "regress-8403",
|
|
Description: "Regress pulumi/pulumi#8403",
|
|
SkipCompileCheck: codegen.NewStringSet(python),
|
|
},
|
|
{
|
|
Directory: "different-package-name-conflict",
|
|
Description: "different packages with the same resource",
|
|
Skip: allLanguages,
|
|
},
|
|
{
|
|
Directory: "different-enum",
|
|
Description: "An enum in a different package namespace",
|
|
Skip: codegen.NewStringSet("dotnet/compile"),
|
|
},
|
|
{
|
|
Directory: "azure-native-nested-types",
|
|
Description: "Condensed example of nested collection types from Azure Native",
|
|
Skip: codegen.NewStringSet("go/any"),
|
|
},
|
|
{
|
|
Directory: "regress-go-8664",
|
|
Description: "Regress pulumi/pulumi#8664 affecting Go",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "regress-go-10527",
|
|
Description: "Regress pulumi/pulumi#10527 affecting Go",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "regress-go-12971",
|
|
Description: "Regress pulumi/pulumi#12971 affecting Go",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "regress-py-12546",
|
|
Description: "Regress pulumi/pulumi#12546 affecting Python",
|
|
Skip: allLanguages.Except("python/any"),
|
|
},
|
|
{
|
|
Directory: "docs-collision",
|
|
Description: "Tests that resources and functions with the same name do not clobber each other.",
|
|
Skip: allLanguages.Except("docs/any"),
|
|
},
|
|
{
|
|
Directory: "other-owned",
|
|
Description: "CSharp rootNamespaces",
|
|
// We only test in dotnet, because we are testing a change in a dotnet
|
|
// language property. Other tests should pass, but do not put the
|
|
// relevant feature under test. To save time, we skip them.
|
|
//
|
|
// We need to see dotnet changes (paths) in the docs too.
|
|
Skip: allLanguages.Except("dotnet/any").Except("docs/any"),
|
|
},
|
|
{
|
|
Directory: "external-node-compatibility",
|
|
// In this case, this test's schema has kubernetes20 set, but is referencing a type from Google Native
|
|
// which doesn't have any compatibility modes set, so the referenced type should be `AuditConfigArgs`
|
|
// (with the `Args` suffix) and not `AuditConfig`.
|
|
Description: "Ensure external package compatibility modes are used when referencing external types",
|
|
Skip: allLanguages.Except("nodejs/any"),
|
|
},
|
|
{
|
|
Directory: "external-go-import-aliases",
|
|
// Google Native has its own import aliases, so those should be respected, unless there are local aliases.
|
|
// AWS Classic doesn't have any import aliases, so none should be used, unless there are local aliases.
|
|
Description: "Ensure external import aliases are honored, and any local import aliases override them",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "external-python-same-module-name",
|
|
Description: "Ensure referencing external types/resources with the same module name are referenced correctly",
|
|
Skip: allLanguages.Except("python/any"),
|
|
},
|
|
{
|
|
Directory: "enum-reference",
|
|
Description: "Ensure referencing external types/resources with referenced enums import correctly",
|
|
},
|
|
{
|
|
Directory: "enum-reference-python",
|
|
Description: "Ensure referencing external types/resources with referenced enums import correctly in Python",
|
|
Skip: allLanguages.Except("python/any"),
|
|
},
|
|
{
|
|
Directory: "external-enum",
|
|
Description: "Ensure we generate valid tokens for external enums",
|
|
Skip: codegen.NewStringSet("dotnet/any"),
|
|
},
|
|
{
|
|
Directory: "internal-dependencies-go",
|
|
Description: "Emit Go internal dependencies",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "go-overridden-internal-module-name",
|
|
Description: "Go SDK where the internal module name is overridden to be 'utilities'",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "go-plain-ref-repro",
|
|
Description: "Generate a resource that accepts a plain input type",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "go-nested-collections",
|
|
Description: "Generate a resource that outputs [][][]Foo",
|
|
Skip: allLanguages.Except("go/any"),
|
|
},
|
|
{
|
|
Directory: "functions-secrets",
|
|
// Secret properties for non-Output<T> returning functions cannot be secret because they are plain.
|
|
Description: "functions that have properties that are secrets in the schema",
|
|
},
|
|
{
|
|
Directory: "secrets",
|
|
Description: "Generate a resource with secret properties",
|
|
SkipCompileCheck: codegen.NewStringSet(dotnet),
|
|
},
|
|
{
|
|
Directory: "regress-py-tfbridge-611",
|
|
Description: "Regresses pulumi/pulumi-terraform-bridge#611",
|
|
Skip: allLanguages.Except("python/any").Union(codegen.NewStringSet("python/test", "python/py_compile")),
|
|
},
|
|
{
|
|
Directory: "hyphenated-symbols",
|
|
Description: "Test that types can have names with hyphens in them",
|
|
Skip: allLanguages.Except("go/any").Except("python/any"),
|
|
},
|
|
{
|
|
Directory: "provider-type-schema",
|
|
Description: "A schema with a type called Provider schema",
|
|
},
|
|
{
|
|
Directory: "embedded-crd-types",
|
|
Description: "A schema with CRD types with package names different from the main package",
|
|
Skip: codegen.NewStringSet("dotnet/any"),
|
|
},
|
|
{
|
|
Directory: "regress-py-14012",
|
|
Description: "Regresses https://github.com/pulumi/pulumi/issues/14012",
|
|
Skip: allLanguages.Except("python/any"),
|
|
},
|
|
{
|
|
Directory: "configure-prop-names",
|
|
Description: "Checks that python _configure handles camelcased and snakecased prop names correctly",
|
|
Skip: allLanguages.Except("python/any"),
|
|
},
|
|
}
|
|
|
|
var genSDKOnly bool
|
|
|
|
func NoSDKCodegenChecks() bool {
|
|
return genSDKOnly
|
|
}
|
|
|
|
func init() {
|
|
noChecks := false
|
|
if env, ok := os.LookupEnv("PULUMI_TEST_SDK_NO_CHECKS"); ok {
|
|
noChecks, _ = strconv.ParseBool(env)
|
|
}
|
|
flag.BoolVar(&genSDKOnly, "sdk.no-checks", noChecks, "when set, skips all post-SDK-generation checks")
|
|
|
|
// NOTE: the testing package will call flag.Parse.
|
|
}
|
|
|
|
// SDKCodegenOptions describes the set of codegen tests for a language.
|
|
type SDKCodegenOptions struct {
|
|
// Name of the programming language.
|
|
Language string
|
|
|
|
// Language-aware code generator; such as `GeneratePackage`.
|
|
// from `codegen/dotnet`.
|
|
GenPackage GenPkgSignature
|
|
|
|
// Extra checks for all the tests. They keys of this map are
|
|
// of the form "$language/$check" such as "go/compile".
|
|
Checks map[string]CodegenCheck
|
|
|
|
// The tests to run. A testcase `tt` are assumed to be located at
|
|
// ../testing/test/testdata/${tt.Directory}
|
|
TestCases []*SDKTest
|
|
}
|
|
|
|
// TestSDKCodegen runs the complete set of SDK code generation tests
|
|
// against a particular language's code generator. It also verifies
|
|
// that the generated code is structurally sound.
|
|
//
|
|
// The test files live in `pkg/codegen/testing/test/testdata` and
|
|
// are registered in `var sdkTests` in `sdk_driver.go`.
|
|
//
|
|
// An SDK code generation test files consists of a schema and a set of
|
|
// expected outputs for each language. Each test is structured as a
|
|
// directory that contains that information:
|
|
//
|
|
// testdata/
|
|
// my-simple-schema/ # i.e. `simple-enum-schema`
|
|
// schema.(json|yaml)
|
|
// go/
|
|
// python/
|
|
// nodejs/
|
|
// dotnet/
|
|
// ...
|
|
//
|
|
// The schema is the only piece that *must* be manually authored.
|
|
//
|
|
// Once the schema has been written, the actual codegen outputs can be
|
|
// generated by running the following in `pkg/codegen` directory:
|
|
//
|
|
// PULUMI_ACCEPT=true go test ./...
|
|
//
|
|
// This will rebuild subfolders such as `go/` from scratch and store
|
|
// the set of code-generated file names in `go/codegen-manifest.json`.
|
|
// If these outputs look correct, they need to be checked into git and
|
|
// will then serve as the expected values for the normal test runs:
|
|
//
|
|
// go test ./...
|
|
//
|
|
// That is, the normal test runs will fail if changes to codegen or
|
|
// schema lead to a diff in the generated file set. If the diff is
|
|
// intentional, it can be accepted again via `PULUMI_ACCEPT=true`.
|
|
//
|
|
// To support running unit tests over the generated code, the tests
|
|
// also support mixing in manually written `$lang-extras` files into
|
|
// the generated tree. For example, given the following input:
|
|
//
|
|
// testdata/
|
|
// my-simple-schema/
|
|
// schema.json
|
|
// go/
|
|
// go-extras/
|
|
// tests/
|
|
// go_test.go
|
|
//
|
|
// The system will copy `go-extras/tests/go_test.go` into
|
|
// `go/tests/go_test.go` before performing compilation and unit test
|
|
// checks over the project generated in `go`.
|
|
func TestSDKCodegen(t *testing.T, opts *SDKCodegenOptions) { // revive:disable-line
|
|
if runtime.GOOS == "windows" {
|
|
t.Skip("TestSDKCodegen is skipped on Windows")
|
|
}
|
|
|
|
testDir := filepath.Join("..", "testing", "test", "testdata")
|
|
|
|
require.NotNil(t, opts.TestCases, "No test cases were provided. This was probably a mistake")
|
|
for _, tt := range opts.TestCases {
|
|
tt := tt // avoid capturing loop variable `sdkTest` in the closure
|
|
|
|
t.Run(tt.Directory, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tt.Mutex.Lock()
|
|
t.Cleanup(tt.Mutex.Unlock)
|
|
|
|
t.Log(tt.Description)
|
|
|
|
dirPath := filepath.Join(testDir, filepath.FromSlash(tt.Directory))
|
|
|
|
schemaPath := filepath.Join(dirPath, "schema.json")
|
|
if _, err := os.Stat(schemaPath); err != nil && os.IsNotExist(err) {
|
|
schemaPath = filepath.Join(dirPath, "schema.yaml")
|
|
}
|
|
|
|
if tt.ShouldSkipCodegen(opts.Language) {
|
|
t.Logf("Skipping generation + tests for %s", tt.Directory)
|
|
return
|
|
}
|
|
|
|
files, err := GeneratePackageFilesFromSchema(schemaPath, opts.GenPackage)
|
|
require.NoError(t, err)
|
|
|
|
if !RewriteFilesWhenPulumiAccept(t, dirPath, opts.Language, files) {
|
|
expectedFiles, err := LoadBaseline(dirPath, opts.Language)
|
|
require.NoError(t, err)
|
|
|
|
if !ValidateFileEquality(t, files, expectedFiles) {
|
|
t.Fail()
|
|
}
|
|
}
|
|
|
|
if genSDKOnly {
|
|
return
|
|
}
|
|
|
|
CopyExtraFiles(t, dirPath, opts.Language)
|
|
|
|
// Merge language-specific global and
|
|
// test-specific checks, with test-specific
|
|
// having precedence.
|
|
allChecks := make(map[string]CodegenCheck)
|
|
for k, v := range opts.Checks {
|
|
allChecks[k] = v
|
|
}
|
|
for k, v := range tt.Checks {
|
|
allChecks[k] = v
|
|
}
|
|
|
|
// Sort the checks in alphabetical order.
|
|
var checkOrder []string
|
|
for check := range allChecks {
|
|
checkOrder = append(checkOrder, check)
|
|
}
|
|
sort.Strings(checkOrder)
|
|
|
|
codeDir := filepath.Join(dirPath, opts.Language)
|
|
|
|
// Perform the checks.
|
|
//nolint:paralleltest // test functions are ordered
|
|
for _, check := range checkOrder {
|
|
check := check
|
|
t.Run(check, func(t *testing.T) {
|
|
if tt.ShouldSkipTest(opts.Language, check) {
|
|
t.Skip()
|
|
}
|
|
checkFun := allChecks[check]
|
|
checkFun(t, codeDir)
|
|
})
|
|
}
|
|
})
|
|
}
|
|
}
|