2020-06-17 21:02:45 +00:00
|
|
|
package schema
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/url"
|
2022-09-10 07:22:11 +00:00
|
|
|
"os"
|
2020-06-17 21:02:45 +00:00
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
2022-11-03 23:46:41 +00:00
|
|
|
"github.com/blang/semver"
|
2020-06-17 21:02:45 +00:00
|
|
|
"github.com/pgavlin/goldmark/ast"
|
|
|
|
"github.com/pgavlin/goldmark/testutil"
|
2022-11-03 23:46:41 +00:00
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/testing/utils"
|
2020-06-17 21:02:45 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2022-11-03 23:46:41 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2020-06-17 21:02:45 +00:00
|
|
|
)
|
|
|
|
|
2022-09-14 03:03:50 +00:00
|
|
|
// Note to future engineers: keep each file tested as a single test, do not use `t.Run` in the inner
|
|
|
|
// loops.
|
|
|
|
//
|
|
|
|
// Time to complete on these tests increases from ~2s to 30s or more and the number of lines logged
|
|
|
|
// to stdout from 46 lines to over 1,000,000 lines of output. This corresponds to the roughly 1
|
|
|
|
// million doc items tested across each file.
|
|
|
|
//
|
|
|
|
// Aside from just being verbose, the voluminous output makes `gotestsum` analysis less useful and
|
|
|
|
// prevents use of the `ci-matrix` tool.
|
|
|
|
|
2022-02-07 11:10:04 +00:00
|
|
|
var testdataPath = filepath.Join("..", "testing", "test", "testdata")
|
2020-06-17 21:02:45 +00:00
|
|
|
|
|
|
|
var nodeAssertions = testutil.DefaultNodeAssertions().Union(testutil.NodeAssertions{
|
|
|
|
KindShortcode: func(t *testing.T, sourceExpected, sourceActual []byte, expected, actual ast.Node) bool {
|
|
|
|
shortcodeExpected, shortcodeActual := expected.(*Shortcode), actual.(*Shortcode)
|
|
|
|
return testutil.AssertEqualBytes(t, shortcodeExpected.Name, shortcodeActual.Name)
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
type doc struct {
|
|
|
|
entity string
|
|
|
|
content string
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDocsForProperty(parent string, p *Property) []doc {
|
|
|
|
entity := path.Join(parent, p.Name)
|
|
|
|
return []doc{
|
|
|
|
{entity: entity + "/description", content: p.Comment},
|
|
|
|
{entity: entity + "/deprecationMessage", content: p.DeprecationMessage},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDocsForObjectType(path string, t *ObjectType) []doc {
|
|
|
|
if t == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
docs := []doc{{entity: path + "/description", content: t.Comment}}
|
|
|
|
for _, p := range t.Properties {
|
|
|
|
docs = append(docs, getDocsForProperty(path+"/properties", p)...)
|
|
|
|
}
|
|
|
|
return docs
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDocsForFunction(f *Function) []doc {
|
|
|
|
entity := "#/functions/" + url.PathEscape(f.Token)
|
|
|
|
docs := []doc{
|
|
|
|
{entity: entity + "/description", content: f.Comment},
|
|
|
|
{entity: entity + "/deprecationMessage", content: f.DeprecationMessage},
|
|
|
|
}
|
|
|
|
docs = append(docs, getDocsForObjectType(entity+"/inputs/properties", f.Inputs)...)
|
2023-01-11 22:17:14 +00:00
|
|
|
|
|
|
|
if f.ReturnType != nil {
|
|
|
|
if objectType, ok := f.ReturnType.(*ObjectType); ok && objectType != nil {
|
|
|
|
docs = append(docs, getDocsForObjectType(entity+"/outputs/properties", objectType)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:02:45 +00:00
|
|
|
return docs
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDocsForResource(r *Resource, isProvider bool) []doc {
|
|
|
|
var entity string
|
|
|
|
if isProvider {
|
|
|
|
entity = "#/provider"
|
|
|
|
} else {
|
|
|
|
entity = "#/resources/" + url.PathEscape(r.Token)
|
|
|
|
}
|
|
|
|
|
|
|
|
docs := []doc{
|
|
|
|
{entity: entity + "/description", content: r.Comment},
|
|
|
|
{entity: entity + "/deprecationMessage", content: r.DeprecationMessage},
|
|
|
|
}
|
|
|
|
for _, p := range r.InputProperties {
|
|
|
|
docs = append(docs, getDocsForProperty(entity+"/inputProperties", p)...)
|
|
|
|
}
|
|
|
|
for _, p := range r.Properties {
|
|
|
|
docs = append(docs, getDocsForProperty(entity+"/properties", p)...)
|
|
|
|
}
|
|
|
|
docs = append(docs, getDocsForObjectType(entity+"/stateInputs", r.StateInputs)...)
|
|
|
|
return docs
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDocsForPackage(pkg *Package) []doc {
|
|
|
|
var allDocs []doc
|
|
|
|
for _, p := range pkg.Config {
|
|
|
|
allDocs = append(allDocs, getDocsForProperty("#/config/variables", p)...)
|
|
|
|
}
|
|
|
|
for _, f := range pkg.Functions {
|
|
|
|
allDocs = append(allDocs, getDocsForFunction(f)...)
|
|
|
|
}
|
|
|
|
allDocs = append(allDocs, getDocsForResource(pkg.Provider, true)...)
|
|
|
|
for _, r := range pkg.Resources {
|
|
|
|
allDocs = append(allDocs, getDocsForResource(r, false)...)
|
|
|
|
}
|
|
|
|
for _, t := range pkg.Types {
|
|
|
|
if obj, ok := t.(*ObjectType); ok {
|
|
|
|
allDocs = append(allDocs, getDocsForObjectType("#/types", obj)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return allDocs
|
|
|
|
}
|
|
|
|
|
2023-10-14 08:32:43 +00:00
|
|
|
//nolint:paralleltest // needs to set plugin acquisition env var
|
2020-06-17 21:02:45 +00:00
|
|
|
func TestParseAndRenderDocs(t *testing.T) {
|
2022-09-10 07:22:11 +00:00
|
|
|
files, err := os.ReadDir(testdataPath)
|
2020-06-17 21:02:45 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("could not read test data: %v", err)
|
|
|
|
}
|
|
|
|
|
2023-10-14 08:32:43 +00:00
|
|
|
//nolint:paralleltest // needs to set plugin acquisition env var
|
2020-06-17 21:02:45 +00:00
|
|
|
for _, f := range files {
|
2022-03-04 08:17:41 +00:00
|
|
|
f := f
|
2022-10-22 00:32:55 +00:00
|
|
|
if filepath.Ext(f.Name()) != ".json" || strings.Contains(f.Name(), "awsx") {
|
2020-06-17 21:02:45 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Run(f.Name(), func(t *testing.T) {
|
2023-10-14 08:32:43 +00:00
|
|
|
t.Setenv("PULUMI_DISABLE_AUTOMATIC_PLUGIN_ACQUISITION", "false")
|
2022-03-04 08:17:41 +00:00
|
|
|
|
2020-06-17 21:02:45 +00:00
|
|
|
path := filepath.Join(testdataPath, f.Name())
|
2023-01-06 22:39:16 +00:00
|
|
|
contents, err := os.ReadFile(path)
|
2020-06-17 21:02:45 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("could not read %v: %v", path, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var spec PackageSpec
|
|
|
|
if err = json.Unmarshal(contents, &spec); err != nil {
|
|
|
|
t.Fatalf("could not unmarshal package spec: %v", err)
|
|
|
|
}
|
|
|
|
pkg, err := ImportSpec(spec, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("could not import package: %v", err)
|
|
|
|
}
|
|
|
|
|
2022-03-04 08:17:41 +00:00
|
|
|
//nolint:paralleltest // these are large, compute heavy tests. keep them in a single thread
|
2020-06-17 21:02:45 +00:00
|
|
|
for _, doc := range getDocsForPackage(pkg) {
|
2022-03-04 08:17:41 +00:00
|
|
|
doc := doc
|
2022-09-14 03:03:50 +00:00
|
|
|
original := []byte(doc.content)
|
|
|
|
expected := ParseDocs(original)
|
|
|
|
rendered := []byte(RenderDocsToString(original, expected))
|
|
|
|
actual := ParseDocs(rendered)
|
|
|
|
if !testutil.AssertSameStructure(t, original, rendered, expected, actual, nodeAssertions) {
|
|
|
|
t.Logf("original: %v", doc.content)
|
|
|
|
t.Logf("rendered: %v", string(rendered))
|
|
|
|
}
|
2020-06-17 21:02:45 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-03 23:46:41 +00:00
|
|
|
func pkgInfo(t *testing.T, filename string) (string, *semver.Version) {
|
|
|
|
filename = strings.TrimSuffix(filename, ".json")
|
|
|
|
idx := 0
|
|
|
|
for {
|
|
|
|
i := strings.IndexByte(filename[idx:], '-') + idx
|
|
|
|
require.Truef(t, i != -1, "Could not parse %q into (pkg, version)", filename)
|
|
|
|
name := filename[:i]
|
|
|
|
version := filename[i+1:]
|
|
|
|
if v, err := semver.Parse(version); err == nil {
|
|
|
|
return name, &v
|
|
|
|
}
|
|
|
|
idx = i + 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-17 21:02:45 +00:00
|
|
|
func TestReferenceRenderer(t *testing.T) {
|
2022-03-04 08:17:41 +00:00
|
|
|
t.Parallel()
|
|
|
|
|
2022-09-10 07:22:11 +00:00
|
|
|
files, err := os.ReadDir(testdataPath)
|
2020-06-17 21:02:45 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("could not read test data: %v", err)
|
|
|
|
}
|
|
|
|
|
2022-11-03 23:46:41 +00:00
|
|
|
seenNames := map[string]struct{}{}
|
|
|
|
|
2022-03-04 08:17:41 +00:00
|
|
|
//nolint:paralleltest // false positive because range var isn't used directly in t.Run(name) arg
|
2020-06-17 21:02:45 +00:00
|
|
|
for _, f := range files {
|
2022-03-04 08:17:41 +00:00
|
|
|
f := f
|
2022-11-03 23:46:41 +00:00
|
|
|
if filepath.Ext(f.Name()) != ".json" || f.Name() == "types.json" {
|
2020-06-17 21:02:45 +00:00
|
|
|
continue
|
|
|
|
}
|
2022-11-03 23:46:41 +00:00
|
|
|
name, version := pkgInfo(t, f.Name())
|
|
|
|
|
|
|
|
if _, ok := seenNames[name]; ok {
|
|
|
|
continue
|
|
|
|
}
|
all: Fix revive issues
Fixes the following issues found by revive
included in the latest release of golangci-lint.
Full list of issues:
**pkg**
```
backend/display/object_diff.go:47:10: superfluous-else: if block ends with a break statement, so drop this else and outdent its block (move short variable declaration to its own line if necessary) (revive)
backend/display/object_diff.go:716:12: redefines-builtin-id: redefinition of the built-in function delete (revive)
backend/display/object_diff.go:742:14: redefines-builtin-id: redefinition of the built-in function delete (revive)
backend/display/object_diff.go:983:10: superfluous-else: if block ends with a continue statement, so drop this else and outdent its block (revive)
backend/httpstate/backend.go:1814:4: redefines-builtin-id: redefinition of the built-in function cap (revive)
backend/httpstate/backend.go:1824:5: redefines-builtin-id: redefinition of the built-in function cap (revive)
backend/httpstate/client/client.go:444:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
backend/httpstate/client/client.go:455:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
cmd/pulumi/org.go:113:4: if-return: redundant if ...; err != nil check, just return error instead. (revive)
cmd/pulumi/util.go:216:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
codegen/docs/gen.go:428:2: redefines-builtin-id: redefinition of the built-in function copy (revive)
codegen/hcl2/model/expression.go:2151:5: redefines-builtin-id: redefinition of the built-in function close (revive)
codegen/hcl2/syntax/comments.go:151:2: redefines-builtin-id: redefinition of the built-in function close (revive)
codegen/hcl2/syntax/comments.go:329:3: redefines-builtin-id: redefinition of the built-in function close (revive)
codegen/hcl2/syntax/comments.go:381:5: redefines-builtin-id: redefinition of the built-in function close (revive)
codegen/nodejs/gen.go:1367:5: redefines-builtin-id: redefinition of the built-in function copy (revive)
codegen/python/gen_program_expressions.go:136:2: redefines-builtin-id: redefinition of the built-in function close (revive)
codegen/python/gen_program_expressions.go:142:3: redefines-builtin-id: redefinition of the built-in function close (revive)
codegen/report/report.go:126:6: redefines-builtin-id: redefinition of the built-in function panic (revive)
codegen/schema/docs_test.go:210:10: superfluous-else: if block ends with a continue statement, so drop this else and outdent its block (move short variable declaration to its own line if necessary) (revive)
codegen/schema/schema.go:790:2: redefines-builtin-id: redefinition of the built-in type any (revive)
codegen/schema/schema.go:793:4: redefines-builtin-id: redefinition of the built-in type any (revive)
resource/deploy/plan.go:506:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
resource/deploy/snapshot_test.go:59:3: redefines-builtin-id: redefinition of the built-in function copy (revive)
resource/deploy/state_builder.go:108:2: redefines-builtin-id: redefinition of the built-in function copy (revive)
```
**sdk**
```
go/common/resource/plugin/context.go:142:2: redefines-builtin-id: redefinition of the built-in function copy (revive)
go/common/resource/plugin/plugin.go:142:12: superfluous-else: if block ends with a break statement, so drop this else and outdent its block (revive)
go/common/resource/properties_diff.go:114:2: redefines-builtin-id: redefinition of the built-in function len (revive)
go/common/resource/properties_diff.go:117:4: redefines-builtin-id: redefinition of the built-in function len (revive)
go/common/resource/properties_diff.go:122:4: redefines-builtin-id: redefinition of the built-in function len (revive)
go/common/resource/properties_diff.go:127:4: redefines-builtin-id: redefinition of the built-in function len (revive)
go/common/resource/properties_diff.go:132:4: redefines-builtin-id: redefinition of the built-in function len (revive)
go/common/util/deepcopy/copy.go:30:1: redefines-builtin-id: redefinition of the built-in function copy (revive)
go/common/workspace/creds.go:242:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
go/pulumi-language-go/main.go:569:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
go/pulumi-language-go/main.go:706:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
go/pulumi/run_test.go:925:2: redefines-builtin-id: redefinition of the built-in type any (revive)
go/pulumi/run_test.go:933:3: redefines-builtin-id: redefinition of the built-in type any (revive)
nodejs/cmd/pulumi-language-nodejs/main.go:778:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
python/cmd/pulumi-language-python/main.go:1011:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
python/cmd/pulumi-language-python/main.go:863:2: if-return: redundant if ...; err != nil check, just return error instead. (revive)
python/python.go:230:2: redefines-builtin-id: redefinition of the built-in function print (revive)
```
**tests**
```
integration/integration_util_test.go:282:11: superfluous-else: if block ends with a continue statement, so drop this else and outdent its block (move short variable declaration to its own line if necessary) (revive)
```
2023-03-20 23:48:02 +00:00
|
|
|
seenNames[name] = struct{}{}
|
2020-06-17 21:02:45 +00:00
|
|
|
|
|
|
|
t.Run(f.Name(), func(t *testing.T) {
|
2022-03-04 08:17:41 +00:00
|
|
|
t.Parallel()
|
|
|
|
|
2022-11-03 23:46:41 +00:00
|
|
|
host := utils.NewHost(testdataPath)
|
|
|
|
defer host.Close()
|
|
|
|
loader := NewPluginLoader(host)
|
|
|
|
pkg, err := loader.LoadPackage(name, version)
|
2020-06-17 21:02:45 +00:00
|
|
|
if err != nil {
|
2022-11-03 23:46:41 +00:00
|
|
|
t.Fatalf("could not import package %s,%s: %v", name, version, err)
|
2020-06-17 21:02:45 +00:00
|
|
|
}
|
|
|
|
|
2022-03-04 08:17:41 +00:00
|
|
|
//nolint:paralleltest // these are large, compute heavy tests. keep them in a single thread
|
2020-06-17 21:02:45 +00:00
|
|
|
for _, doc := range getDocsForPackage(pkg) {
|
2022-03-04 08:17:41 +00:00
|
|
|
doc := doc
|
2022-09-14 03:03:50 +00:00
|
|
|
|
|
|
|
text := []byte(fmt.Sprintf("[entity](%s)", doc.entity))
|
2023-02-23 20:51:11 +00:00
|
|
|
expected := strings.ReplaceAll(doc.entity, "/", "_") + "\n"
|
2022-09-14 03:03:50 +00:00
|
|
|
|
|
|
|
parsed := ParseDocs(text)
|
|
|
|
actual := []byte(RenderDocsToString(text, parsed, WithReferenceRenderer(
|
|
|
|
func(r *Renderer, w io.Writer, src []byte, l *ast.Link, enter bool) (ast.WalkStatus, error) {
|
|
|
|
if !enter {
|
|
|
|
return ast.WalkContinue, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
replaced := bytes.Replace(l.Destination, []byte{'/'}, []byte{'_'}, -1)
|
|
|
|
if _, err := r.MarkdownRenderer().Write(w, replaced); err != nil {
|
|
|
|
return ast.WalkStop, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ast.WalkSkipChildren, nil
|
|
|
|
})))
|
|
|
|
|
|
|
|
assert.Equal(t, expected, string(actual))
|
2020-06-17 21:02:45 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|