pulumi/pkg/codegen/docs/gen.go

2262 lines
68 KiB
Go

//go:generate go run bundler.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.
// Pulling out some of the repeated strings tokens into constants would harm readability, so we just ignore the
// goconst linter's warning.
//
//nolint:lll, goconst
package docs
import (
"bytes"
"embed"
"errors"
"fmt"
"html"
"html/template"
"path"
"sort"
"strings"
"github.com/golang/glog"
"github.com/pgavlin/goldmark"
"github.com/pulumi/pulumi-java/pkg/codegen/java"
yaml "github.com/pulumi/pulumi-yaml/pkg/pulumiyaml/codegen"
"github.com/pulumi/pulumi/pkg/v3/codegen"
"github.com/pulumi/pulumi/pkg/v3/codegen/dotnet"
go_gen "github.com/pulumi/pulumi/pkg/v3/codegen/go"
"github.com/pulumi/pulumi/pkg/v3/codegen/nodejs"
"github.com/pulumi/pulumi/pkg/v3/codegen/python"
"github.com/pulumi/pulumi/pkg/v3/codegen/schema"
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
)
//go:embed templates/*.tmpl
var packagedTemplates embed.FS
// NOTE: This lookup map can be removed when all Pulumi-managed packages
// have a DisplayName in their schema. See pulumi/pulumi#7813.
// This lookup table no longer needs to be updated for new providers
// and is considered stale.
//
// titleLookup is a map of package name to the desired display name.
func titleLookup(shortName string) (string, bool) {
v, ok := map[string]string{
"aiven": "Aiven",
"akamai": "Akamai",
"alicloud": "Alibaba Cloud",
"auth0": "Auth0",
"aws": "AWS Classic",
"awsx": "AWSx (Pulumi Crosswalk for AWS)",
"aws-apigateway": "AWS API Gateway",
"aws-miniflux": "Miniflux",
"aws-native": "AWS Native",
"aws-quickstart-aurora-mysql": "AWS QuickStart Aurora MySQL",
"aws-quickstart-aurora-postgres": "AWS QuickStart Aurora PostgreSQL",
"aws-quickstart-redshift": "AWS QuickStart Redshift",
"aws-serverless": "AWS Serverless",
"aws-quickstart-vpc": "AWS QuickStart VPC",
"aws-s3-replicated-bucket": "AWS S3 Replicated Bucket",
"azure": "Azure Classic",
"azure-justrun": "Azure Justrun",
"azure-native": "Azure Native",
"azure-quickstart-acr-geo-replication": "Azure QuickStart ACR Geo Replication",
"azure-quickstart-aks": "Azure QuickStart AKS",
"azure-quickstart-compute": "Azure QuickStart Compute",
"azure-quickstart-sql": "Azure QuickStart SQL",
"azuread": "Azure Active Directory (Azure AD)",
"azuredevops": "Azure DevOps",
"azuresel": "Azure",
"civo": "Civo",
"cloudamqp": "CloudAMQP",
"cloudflare": "Cloudflare",
"cloudinit": "cloud-init",
"confluentcloud": "Confluent Cloud",
"confluent": "Confluent Cloud (Deprecated)",
"consul": "HashiCorp Consul",
"coredns-helm": "CoreDNS (Helm)",
"datadog": "Datadog",
"digitalocean": "DigitalOcean",
"dnsimple": "DNSimple",
"docker": "Docker",
"docker-buildkit": "Docker BuildKit",
"eks": "Amazon EKS",
"equinix-metal": "Equinix Metal",
"f5bigip": "f5 BIG-IP",
"fastly": "Fastly",
"gcp": "Google Cloud (GCP) Classic",
"gcp-global-cloudrun": "Google Global Cloud Run",
"gcp-project-scaffold": "Google Project Scaffolding",
"google-native": "Google Cloud Native",
"github": "GitHub",
"github-serverless-webhook": "GitHub Serverless Webhook",
"gitlab": "GitLab",
"hcloud": "Hetzner Cloud",
"istio-helm": "Istio (Helm)",
"jaeger-helm": "Jaeger (Helm)",
"kafka": "Kafka",
"keycloak": "Keycloak",
"kong": "Kong",
"kubernetes": "Kubernetes",
"libvirt": "libvirt",
"linode": "Linode",
"mailgun": "Mailgun",
"minio": "MinIO",
"mongodbatlas": "MongoDB Atlas",
"mysql": "MySQL",
"newrelic": "New Relic",
"kubernetes-ingress-nginx": "NGINX Ingress Controller (Helm)",
"kubernetes-coredns": "CoreDNS (Helm)",
"kubernetes-cert-manager": "Jetstack Cert Manager (Helm)",
"nomad": "HashiCorp Nomad",
"ns1": "NS1",
"okta": "Okta",
"openstack": "OpenStack",
"opsgenie": "Opsgenie",
"packet": "Packet",
"pagerduty": "PagerDuty",
"pulumi-std": "Pulumi Standard Library",
"postgresql": "PostgreSQL",
"prometheus-helm": "Prometheus (Helm)",
"rabbitmq": "RabbitMQ",
"rancher2": "Rancher2",
"random": "random",
"rke": "Rancher Kubernetes Engine (RKE)",
"run-my-darn-container": "Run My Darn Container",
"shipa": "Shipa",
"signalfx": "SignalFx",
"snowflake": "Snowflake",
"splunk": "Splunk",
"spotinst": "Spotinst",
"sumologic": "Sumo Logic",
"tls": "TLS",
"vault": "Vault",
"venafi": "Venafi",
"vsphere": "vSphere",
"wavefront": "Wavefront",
"yandex": "Yandex",
}[shortName]
return v, ok
}
// Property anchor tag separator, used in a property anchor tag id to separate the
// property and language (e.g. property~lang).
const propertyLangSeparator = "_"
type docGenContext struct {
internalModMap map[string]*modContext
supportedLanguages []string
snippetLanguages []string
templates *template.Template
docHelpers map[string]codegen.DocLanguageHelper
// The language-specific info objects for a certain package (provider).
goPkgInfo go_gen.GoPackageInfo
csharpPkgInfo dotnet.CSharpPackageInfo
nodePkgInfo nodejs.NodePackageInfo
pythonPkgInfo python.PackageInfo
// langModuleNameLookup is a map of module name to its language-specific
// name.
langModuleNameLookup map[string]string
// Maps a *modContext, *schema.Resource, or *schema.Function to the link that was assigned to it.
moduleConflictLinkMap map[interface{}]string
}
// modules is a map of a module name and information
// about it. This is crux of all API docs generation
// as the modContext carries information about the resources,
// functions, as well other modules within each module.
func (dctx *docGenContext) modules() map[string]*modContext {
return dctx.internalModMap
}
func (dctx *docGenContext) setModules(modules map[string]*modContext) {
m := map[string]*modContext{}
for k, v := range modules {
m[k] = v.withDocGenContext(dctx)
}
dctx.internalModMap = m
}
func newDocGenContext() *docGenContext {
supportedLanguages := []string{"csharp", "go", "nodejs", "python", "yaml", "java"}
docHelpers := make(map[string]codegen.DocLanguageHelper)
for _, lang := range supportedLanguages {
switch lang {
case "csharp":
docHelpers[lang] = &dotnet.DocLanguageHelper{}
case "go":
docHelpers[lang] = &go_gen.DocLanguageHelper{}
case "nodejs":
docHelpers[lang] = &nodejs.DocLanguageHelper{}
case "python":
docHelpers[lang] = &python.DocLanguageHelper{}
case "yaml":
docHelpers[lang] = &yaml.DocLanguageHelper{}
case "java":
docHelpers[lang] = &java.DocLanguageHelper{}
}
}
return &docGenContext{
supportedLanguages: supportedLanguages,
snippetLanguages: []string{"csharp", "go", "python", "typescript", "yaml", "java"},
langModuleNameLookup: map[string]string{},
docHelpers: docHelpers,
moduleConflictLinkMap: map[interface{}]string{},
}
}
type typeDetails struct {
inputType bool
}
// header represents the header of each resource markdown file.
type header struct {
Title string
TitleTag string
MetaDesc string
}
// property represents an input or an output property.
type property struct {
// ID is the `id` attribute that will be attached to the DOM element containing the property.
ID string
// DisplayName is the property name with word-breaks.
DisplayName string
Name string
Comment string
Types []propertyType
DeprecationMessage string
Link string
IsRequired bool
IsInput bool
IsReplaceOnChanges bool
}
// enum represents an enum.
type enum struct {
ID string // ID is the `id` attribute attached to the DOM element containing the enum.
DisplayName string // DisplayName is the enum name with word-breaks.
Name string // Name is the language-specific enum name.
Value string
Comment string
DeprecationMessage string
}
// docNestedType represents a complex type.
type docNestedType struct {
Name string
Input bool
AnchorID string
Properties map[string][]property
EnumValues map[string][]enum
}
// propertyType represents the type of a property.
type propertyType struct {
DisplayName string
DescriptionName string // Name used in description list.
Name string
// Link can be a link to an anchor tag on the same
// page, or to another page/site.
Link string
}
// paramSeparator is for data passed to the separator template.
type paramSeparator struct {
Indent string
}
// formalParam represents the formal parameters of a constructor
// or a lookup function.
type formalParam struct {
Name string
Type propertyType
// This is the language specific optional type indicator.
// For example, in nodejs this is the character "?" and in Go
// it's "*".
OptionalFlag string
DefaultValue string
// Comment is an optional description of the parameter.
Comment string
}
type packageDetails struct {
DisplayName string
Repository string
RepositoryName string
License string
Notes string
Version string
}
type resourceDocArgs struct {
Header header
Tool string
// LangChooserLanguages is a comma-separated list of languages to pass to the
// language chooser shortcode. Use this to customize the languages shown for a
// resource. By default, the language chooser will show all languages supported
// by Pulumi for all resources.
LangChooserLanguages string
// Comment represents the introductory resource comment.
Comment string
ExamplesSection []exampleSection
DeprecationMessage string
// Import
ImportDocs string
// ConstructorParams is a map from language to the rendered HTML for the constructor's
// arguments.
ConstructorParams map[string]string
// ConstructorParamsTyped is the typed set of parameters for the constructor, in order.
ConstructorParamsTyped map[string][]formalParam
// ConstructorResource is the resource that is being constructed or
// is the result of a constructor-like function.
ConstructorResource map[string]propertyType
// ArgsRequired is a flag indicating if the args param is required
// when creating a new resource.
ArgsRequired bool
// InputProperties is a map per language and a corresponding slice of
// input properties accepted as args while creating a new resource.
InputProperties map[string][]property
// OutputProperties is a map per language and a corresponding slice of
// output properties returned when a new instance of the resource is
// created.
OutputProperties map[string][]property
// LookupParams is a map of the param string to be rendered per language
// for looking-up a resource.
LookupParams map[string]string
// StateInputs is a map per language and the corresponding slice of
// state input properties required while looking-up an existing resource.
StateInputs map[string][]property
// StateParam is the type name of the state param, if any.
StateParam string
// NestedTypes is a slice of the nested types used in the input and
// output properties.
NestedTypes []docNestedType
// A list of methods associated with the resource.
Methods []methodDocArgs
PackageDetails packageDetails
}
// typeUsage represents a nested type's usage.
type typeUsage struct {
Input bool
Output bool
}
// nestedTypeUsageInfo is a type-alias for a map of Pulumi type-tokens
// and whether or not the type is used as an input and/or output
// properties.
type nestedTypeUsageInfo map[string]typeUsage
func (ss nestedTypeUsageInfo) add(s string, input bool) {
if v, ok := ss[s]; ok {
if input {
v.Input = true
} else {
v.Output = true
}
ss[s] = v
return
}
ss[s] = typeUsage{
Input: input,
Output: !input,
}
}
// contains returns true if the token already exists and matches the
// input or output flag of the token.
func (ss nestedTypeUsageInfo) contains(token string, input bool) bool {
a, ok := ss[token]
if !ok {
return false
}
if input && a.Input {
return true
} else if !input && a.Output {
return true
}
return false
}
type modContext struct {
pkg schema.PackageReference
mod string
inputTypes []*schema.ObjectType
resources []*schema.Resource
functions []*schema.Function
typeDetails map[*schema.ObjectType]*typeDetails
children []*modContext
tool string
docGenContext *docGenContext
}
func (mod *modContext) withDocGenContext(dctx *docGenContext) *modContext {
if mod == nil {
return nil
}
newctx := *mod
newctx.docGenContext = dctx
children := slice.Prealloc[*modContext](len(newctx.children))
for _, c := range newctx.children {
children = append(children, c.withDocGenContext(dctx))
}
newctx.children = children
return &newctx
}
func resourceName(r *schema.Resource) string {
if r.IsProvider {
return "Provider"
}
return strings.Title(tokenToName(r.Token))
}
func (dctx *docGenContext) getLanguageDocHelper(lang string) codegen.DocLanguageHelper {
if h, ok := dctx.docHelpers[lang]; ok {
return h
}
panic(fmt.Errorf("could not find a doc lang helper for %s", lang))
}
type propertyCharacteristics struct {
// input is a flag indicating if the property is an input type.
input bool
}
func (mod *modContext) details(t *schema.ObjectType) *typeDetails {
details, ok := mod.typeDetails[t]
if !ok {
details = &typeDetails{}
if mod.typeDetails == nil {
mod.typeDetails = map[*schema.ObjectType]*typeDetails{}
}
mod.typeDetails[t] = details
}
return details
}
// getLanguageModuleName transforms the current module's name to a
// language-specific name using the language info, if any, for the
// current package.
func (mod *modContext) getLanguageModuleName(lang string) string {
dctx := mod.docGenContext
modName := mod.mod
lookupKey := lang + "_" + modName
if v, ok := mod.docGenContext.langModuleNameLookup[lookupKey]; ok {
return v
}
switch lang {
case "go":
// Go module names use lowercase.
modName = strings.ToLower(modName)
if override, ok := dctx.goPkgInfo.ModuleToPackage[modName]; ok {
modName = override
}
case "csharp":
if override, ok := dctx.csharpPkgInfo.Namespaces[modName]; ok {
modName = override
}
case "nodejs":
if override, ok := dctx.nodePkgInfo.ModuleToPackage[modName]; ok {
modName = override
}
case "python":
if override, ok := dctx.pythonPkgInfo.ModuleNameOverrides[modName]; ok {
modName = override
}
}
mod.docGenContext.langModuleNameLookup[lookupKey] = modName
return modName
}
// cleanTypeString removes any namespaces from the generated type string for all languages.
// The result of this function should be used display purposes only.
func (mod *modContext) cleanTypeString(t schema.Type, langTypeString, lang, modName string, isInput bool) string {
switch lang {
case "go", "python":
langTypeString = cleanOptionalIdentifier(langTypeString, lang)
parts := strings.Split(langTypeString, ".")
return parts[len(parts)-1]
}
cleanCSharpName := func(pkgName, objModName string) string {
// C# types can be wrapped in enumerable types such as List<> or Dictionary<>, so we have to
// only replace the namespace between the < and the > characters.
qualifier := "Inputs"
if !isInput {
qualifier = "Outputs"
}
var csharpNS string
// This type could be at the package-level, so it won't have a module name.
if objModName != "" {
csharpNS = fmt.Sprintf("Pulumi.%s.%s.%s.", title(pkgName, lang), title(objModName, lang), qualifier)
} else {
csharpNS = fmt.Sprintf("Pulumi.%s.%s.", title(pkgName, lang), qualifier)
}
return strings.ReplaceAll(langTypeString, csharpNS, "")
}
cleanNodeJSName := func(objModName string) string {
// The nodejs codegen currently doesn't use the ModuleToPackage override available
// in the k8s package's schema. So we'll manually strip some known module names for k8s.
// TODO[pulumi/pulumi#4325]: Remove this block once the nodejs code gen is able to use the
// package name overrides for modules.
if isKubernetesPackage(mod.pkg) {
langTypeString = strings.ReplaceAll(langTypeString, "k8s.io.", "")
langTypeString = strings.ReplaceAll(langTypeString, "apiserver.", "")
langTypeString = strings.ReplaceAll(langTypeString, "rbac.authorization.v1.", "")
langTypeString = strings.ReplaceAll(langTypeString, "rbac.authorization.v1alpha1.", "")
langTypeString = strings.ReplaceAll(langTypeString, "rbac.authorization.v1beta1.", "")
}
objModName = strings.ReplaceAll(objModName, "/", ".") + "."
return strings.ReplaceAll(langTypeString, objModName, "")
}
switch t := t.(type) {
case *schema.ObjectType:
// Strip "Args" suffixes from display names for everything but Python inputs.
if lang != "python" || (lang == "python" && !isInput) {
name := tokenToName(t.Token)
nameWithArgs := name + "Args"
// If the langTypeString looks like it's a concatenation of its name and "Args", strip out the "Args".
if strings.Contains(langTypeString, nameWithArgs) {
langTypeString = strings.ReplaceAll(langTypeString, nameWithArgs, name)
}
}
}
switch t := t.(type) {
case *schema.ArrayType:
if schema.IsPrimitiveType(t.ElementType) {
break
}
return mod.cleanTypeString(t.ElementType, langTypeString, lang, modName, isInput)
case *schema.UnionType:
for _, e := range t.ElementTypes {
if schema.IsPrimitiveType(e) {
continue
}
return mod.cleanTypeString(e, langTypeString, lang, modName, isInput)
}
case *schema.ObjectType:
objTypeModName := mod.pkg.TokenToModule(t.Token)
if objTypeModName != mod.mod {
modName = mod.getLanguageModuleName(lang)
}
}
if lang == "nodejs" {
return cleanNodeJSName(modName)
} else if lang == "csharp" {
return cleanCSharpName(mod.pkg.Name(), modName)
}
return strings.ReplaceAll(langTypeString, modName, "")
}
// typeString returns a property type suitable for docs with its display name and the anchor link to
// a type if the type of the property is an array or an object.
func (mod *modContext) typeString(t schema.Type, lang string, characteristics propertyCharacteristics, insertWordBreaks bool) propertyType {
t = codegen.PlainType(t)
docLanguageHelper := mod.docGenContext.getLanguageDocHelper(lang)
modName := mod.getLanguageModuleName(lang)
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get package definition for %q", mod.pkg.Name())
langTypeString := docLanguageHelper.GetLanguageTypeString(def, modName, t, characteristics.input)
if optional, ok := t.(*schema.OptionalType); ok {
t = optional.ElementType
}
// If the type is an object type, let's also wrap it with a link to the supporting type
// on the same page using an anchor tag.
var href string
switch t := t.(type) {
case *schema.ArrayType:
elementLangType := mod.typeString(t.ElementType, lang, characteristics, false)
href = elementLangType.Link
case *schema.ObjectType:
tokenName := tokenToName(t.Token)
// Links to anchor tags on the same page must be lower-cased.
href = "#" + strings.ToLower(tokenName)
case *schema.EnumType:
tokenName := tokenToName(t.Token)
// Links to anchor tags on the same page must be lower-cased.
href = "#" + strings.ToLower(tokenName)
case *schema.UnionType:
var elements []string
for _, e := range t.ElementTypes {
elementLangType := mod.typeString(e, lang, characteristics, false)
elements = append(elements, elementLangType.DisplayName)
}
langTypeString = strings.Join(elements, " | ")
}
// Strip the namespace/module prefix for the type's display name.
displayName := langTypeString
if !schema.IsPrimitiveType(t) {
displayName = mod.cleanTypeString(t, langTypeString, lang, modName, characteristics.input)
}
displayName = cleanOptionalIdentifier(displayName, lang)
langTypeString = cleanOptionalIdentifier(langTypeString, lang)
// Name and DisplayName should be html-escaped to avoid throwing off rendering for template types in languages like
// csharp, Java etc. If word-breaks need to be inserted, then the type string should be html-escaped first.
displayName = html.EscapeString(displayName)
if insertWordBreaks {
displayName = wbr(displayName)
}
return propertyType{
Name: html.EscapeString(langTypeString),
DisplayName: displayName,
Link: href,
}
}
// cleanOptionalIdentifier removes the type identifier (i.e. "?" in "string?").
func cleanOptionalIdentifier(s, lang string) string {
switch lang {
case "nodejs":
return strings.TrimSuffix(s, "?")
case "go":
return strings.TrimPrefix(s, "*")
case "csharp":
return strings.TrimSuffix(s, "?")
case "python":
if strings.HasPrefix(s, "Optional[") && strings.HasSuffix(s, "]") {
s = strings.TrimPrefix(s, "Optional[")
s = strings.TrimSuffix(s, "]")
return s
}
}
return s
}
// Resources typically take the same set of parameters to their constructors, and these
// are the default comments/descriptions for them.
const (
ctorNameArgComment = "The unique name of the resource."
ctorArgsArgComment = "The arguments to resource properties."
ctorOptsArgComment = "Bag of options to control resource's behavior."
)
func (mod *modContext) genConstructorTS(r *schema.Resource, argsOptional bool) []formalParam {
name := resourceName(r)
docLangHelper := mod.docGenContext.getLanguageDocHelper("nodejs")
var argsType string
optsType := "CustomResourceOptions"
// The args type for k8s package differs from the rest depending on whether we are dealing with
// overlay resources or regular k8s resources.
if isKubernetesPackage(mod.pkg) {
if mod.isKubernetesOverlayModule() {
if name == "CustomResource" {
argsType = name + "Args"
} else {
argsType = name + "Opts"
}
} else {
// The non-schema-based k8s codegen does not apply a suffix to the input types.
argsType = name
}
if mod.isComponentResource() {
optsType = "ComponentResourceOptions"
}
} else {
argsType = name + "Args"
}
argsFlag := ""
if argsOptional {
argsFlag = "?"
}
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
return []formalParam{
{
Name: "name",
Type: propertyType{
Name: "string",
},
Comment: ctorNameArgComment,
},
{
Name: "args",
OptionalFlag: argsFlag,
Type: propertyType{
Name: argsType,
Link: "#inputs",
},
Comment: ctorArgsArgComment,
},
{
Name: "opts",
OptionalFlag: "?",
Type: propertyType{
Name: optsType,
Link: docLangHelper.GetDocLinkForPulumiType(def, optsType),
},
Comment: ctorOptsArgComment,
},
}
}
func (mod *modContext) genConstructorGo(r *schema.Resource, argsOptional bool) []formalParam {
name := resourceName(r)
argsType := name + "Args"
argsFlag := ""
if argsOptional {
argsFlag = "*"
}
docLangHelper := mod.docGenContext.getLanguageDocHelper("go")
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
return []formalParam{
{
Name: "ctx",
OptionalFlag: "*",
Type: propertyType{
Name: "Context",
Link: docLangHelper.GetDocLinkForPulumiType(def, "Context"),
},
Comment: "Context object for the current deployment.",
},
{
Name: "name",
Type: propertyType{
Name: "string",
},
Comment: ctorNameArgComment,
},
{
Name: "args",
OptionalFlag: argsFlag,
Type: propertyType{
Name: argsType,
Link: "#inputs",
},
Comment: ctorArgsArgComment,
},
{
Name: "opts",
OptionalFlag: "...",
Type: propertyType{
Name: "ResourceOption",
Link: docLangHelper.GetDocLinkForPulumiType(def, "ResourceOption"),
},
Comment: ctorOptsArgComment,
},
}
}
func (mod *modContext) genConstructorCS(r *schema.Resource, argsOptional bool) []formalParam {
name := resourceName(r)
optsType := "CustomResourceOptions"
if isKubernetesPackage(mod.pkg) && mod.isComponentResource() {
optsType = "ComponentResourceOptions"
}
var argsFlag string
var argsDefault string
if argsOptional {
// If the number of required input properties was zero, we can make the args object optional.
argsDefault = " = null"
argsFlag = "?"
}
docLangHelper := mod.docGenContext.getLanguageDocHelper("csharp")
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
return []formalParam{
{
Name: "name",
Type: propertyType{
Name: "string",
},
Comment: ctorNameArgComment,
},
{
Name: "args",
OptionalFlag: argsFlag,
DefaultValue: argsDefault,
Type: propertyType{
Name: name + "Args",
Link: "#inputs",
},
Comment: ctorArgsArgComment,
},
{
Name: "opts",
OptionalFlag: "?",
DefaultValue: " = null",
Type: propertyType{
Name: optsType,
Link: docLangHelper.GetDocLinkForPulumiType(def, fmt.Sprintf("Pulumi.%s", optsType)),
},
Comment: ctorOptsArgComment,
},
}
}
func (mod *modContext) genConstructorYaml() []formalParam {
return []formalParam{
{
Name: "properties",
Comment: ctorArgsArgComment,
},
{
Name: "options",
Comment: ctorOptsArgComment,
},
}
}
func (mod *modContext) genConstructorJava(r *schema.Resource, argsOverload bool) []formalParam {
name := resourceName(r)
optsType := "CustomResourceOptions"
if mod.isComponentResource() {
optsType = "ComponentResourceOptions"
}
docLangHelper := mod.docGenContext.getLanguageDocHelper("java")
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
result := []formalParam{
{
Name: "name",
Type: propertyType{
Name: "String",
},
Comment: ctorNameArgComment,
},
{
Name: "args",
Type: propertyType{
Name: name + "Args",
Link: "#inputs",
},
Comment: ctorArgsArgComment,
},
}
if !argsOverload {
result = append(result, formalParam{
Name: "options",
OptionalFlag: "@Nullable",
Type: propertyType{
Name: optsType,
Link: docLangHelper.GetDocLinkForPulumiType(def, optsType),
},
Comment: ctorOptsArgComment,
})
}
return result
}
func (mod *modContext) genConstructorPython(r *schema.Resource, argsOptional, argsOverload bool) []formalParam {
docLanguageHelper := mod.docGenContext.getLanguageDocHelper("python")
isK8sOverlayMod := mod.isKubernetesOverlayModule()
isDockerImageResource := mod.pkg.Name() == "docker" && resourceName(r) == "Image"
// Kubernetes overlay resources use a different ordering of formal params in Python.
if isK8sOverlayMod && r.IsOverlay {
return getKubernetesOverlayPythonFormalParams(mod.mod)
} else if isDockerImageResource {
return getDockerImagePythonFormalParams()
}
// We perform at least three appends before iterating over input types.
params := slice.Prealloc[formalParam](3 + len(r.InputProperties))
params = append(params, formalParam{
Name: "resource_name",
Type: propertyType{
Name: "str",
},
Comment: ctorNameArgComment,
})
if argsOverload {
// Determine whether we need to use the alternate args class name (e.g. `<Resource>InitArgs` instead of
// `<Resource>Args`) due to an input type with the same name as the resource in the same module.
resName := resourceName(r)
resArgsName := fmt.Sprintf("%sArgs", resName)
for _, inputType := range mod.inputTypes {
inputTypeName := strings.Title(tokenToName(inputType.Token))
if resName == inputTypeName {
resArgsName = fmt.Sprintf("%sInitArgs", resName)
}
}
optionalFlag, defaultVal, descriptionName := "", "", resArgsName
typeName := descriptionName
if argsOptional {
optionalFlag, defaultVal, typeName = "optional", " = None", fmt.Sprintf("Optional[%s]", typeName)
}
params = append(params, formalParam{
Name: "args",
OptionalFlag: optionalFlag,
DefaultValue: defaultVal,
Type: propertyType{
Name: typeName,
DescriptionName: descriptionName,
Link: "#inputs",
},
Comment: ctorArgsArgComment,
})
}
params = append(params, formalParam{
Name: "opts",
OptionalFlag: "optional",
DefaultValue: " = None",
Type: propertyType{
Name: "Optional[ResourceOptions]",
DescriptionName: "ResourceOptions",
Link: "/docs/reference/pkg/python/pulumi/#pulumi.ResourceOptions",
},
Comment: ctorOptsArgComment,
})
if argsOverload {
return params
}
for _, p := range r.InputProperties {
// If the property defines a const value, then skip it.
// For example, in k8s, `apiVersion` and `kind` are often hard-coded
// in the SDK and are not really user-provided input properties.
if p.ConstValue != nil {
continue
}
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
typ := docLanguageHelper.GetLanguageTypeString(def, mod.mod, codegen.PlainType(codegen.OptionalType(p)), true /*input*/)
params = append(params, formalParam{
Name: python.InitParamName(p.Name),
DefaultValue: " = None",
Type: propertyType{
Name: typ,
},
})
}
return params
}
func (mod *modContext) genNestedTypes(member interface{}, resourceType bool) []docNestedType {
dctx := mod.docGenContext
tokens := nestedTypeUsageInfo{}
// Collect all of the types for this "member" as a map of resource names
// and if it appears in an input object and/or output object.
mod.getTypes(member, tokens)
sortedTokens := slice.Prealloc[string](len(tokens))
for token := range tokens {
sortedTokens = append(sortedTokens, token)
}
sort.Strings(sortedTokens)
var typs []docNestedType
for _, token := range sortedTokens {
for iter := mod.pkg.Types().Range(); iter.Next(); {
t, err := iter.Type()
contract.AssertNoErrorf(err, "error iterating types")
switch typ := t.(type) {
case *schema.ObjectType:
if typ.Token != token || len(typ.Properties) == 0 || typ.IsInputShape() {
continue
}
// Create a map to hold the per-language properties of this object.
props := make(map[string][]property)
for _, lang := range dctx.supportedLanguages {
props[lang] = mod.getProperties(typ.Properties, lang, true, true, false)
}
name := strings.Title(tokenToName(typ.Token))
typs = append(typs, docNestedType{
Name: wbr(name),
AnchorID: strings.ToLower(name),
Properties: props,
})
case *schema.EnumType:
if typ.Token != token || len(typ.Elements) == 0 {
continue
}
name := strings.Title(tokenToName(typ.Token))
enums := make(map[string][]enum)
for _, lang := range dctx.supportedLanguages {
docLangHelper := dctx.getLanguageDocHelper(lang)
var langEnumValues []enum
for _, e := range typ.Elements {
enumName, err := docLangHelper.GetEnumName(e, name)
if err != nil {
panic(err)
}
enumID := strings.ToLower(name + propertyLangSeparator + lang)
langEnumValues = append(langEnumValues, enum{
ID: enumID,
DisplayName: wbr(enumName),
Name: enumName,
Value: fmt.Sprintf("%v", e.Value),
Comment: e.Comment,
DeprecationMessage: e.DeprecationMessage,
})
}
enums[lang] = langEnumValues
}
typs = append(typs, docNestedType{
Name: wbr(name),
AnchorID: strings.ToLower(name),
EnumValues: enums,
})
}
}
}
sort.Slice(typs, func(i, j int) bool {
return typs[i].Name < typs[j].Name
})
return typs
}
// getProperties returns a slice of properties that can be rendered for docs for
// the provided slice of properties in the schema.
func (mod *modContext) getProperties(properties []*schema.Property, lang string, input, nested, isProvider bool,
) []property {
return mod.getPropertiesWithIDPrefixAndExclude(properties, lang, input, nested, isProvider, "", nil)
}
func (mod *modContext) getPropertiesWithIDPrefixAndExclude(properties []*schema.Property, lang string, input, nested,
isProvider bool, idPrefix string, exclude func(name string) bool,
) []property {
dctx := mod.docGenContext
if len(properties) == 0 {
return nil
}
docProperties := slice.Prealloc[property](len(properties))
for _, prop := range properties {
if prop == nil {
continue
}
if exclude != nil && exclude(prop.Name) {
continue
}
// If the property has a const value, then don't show it as an input property.
// Even though it is a valid property, it is used by the language code gen to
// generate the appropriate defaults for it. These cannot be overridden by users.
if prop.ConstValue != nil {
continue
}
characteristics := propertyCharacteristics{input: input}
langDocHelper := dctx.getLanguageDocHelper(lang)
name, err := langDocHelper.GetPropertyName(prop)
if err != nil {
panic(err)
}
propLangName := name
propID := idPrefix + strings.ToLower(propLangName+propertyLangSeparator+lang)
propTypes := make([]propertyType, 0)
if typ, isUnion := codegen.UnwrapType(prop.Type).(*schema.UnionType); isUnion {
for _, elementType := range typ.ElementTypes {
propTypes = append(propTypes, mod.typeString(elementType, lang, characteristics, true))
}
} else {
propTypes = append(propTypes, mod.typeString(prop.Type, lang, characteristics, true))
}
comment := prop.Comment
// Default values for Provider inputs correspond to environment variables, so add that info to the docs.
if isProvider && input && prop.DefaultValue != nil && len(prop.DefaultValue.Environment) > 0 {
var suffix string
if len(prop.DefaultValue.Environment) > 1 {
suffix = "s"
}
comment += fmt.Sprintf(" It can also be sourced from the following environment variable%s: ", suffix)
for i, v := range prop.DefaultValue.Environment {
comment += fmt.Sprintf("`%s`", v)
if i != len(prop.DefaultValue.Environment)-1 {
comment += ", "
}
}
}
docProperties = append(docProperties, property{
ID: propID,
DisplayName: wbr(propLangName),
Name: propLangName,
Comment: comment,
DeprecationMessage: prop.DeprecationMessage,
IsRequired: prop.IsRequired(),
IsInput: input,
// We indicate that a property will replace if either
// a) we will force the replace at the engine level
// b) we are told that the provider will require a replace
IsReplaceOnChanges: prop.ReplaceOnChanges || prop.WillReplaceOnChanges,
Link: "#" + propID,
Types: propTypes,
})
}
// Sort required props to move them to the top of the properties list, then by name.
sort.SliceStable(docProperties, func(i, j int) bool {
pi, pj := docProperties[i], docProperties[j]
switch {
case pi.IsRequired != pj.IsRequired:
return pi.IsRequired && !pj.IsRequired
default:
return pi.Name < pj.Name
}
})
return docProperties
}
func getDockerImagePythonFormalParams() []formalParam {
return []formalParam{
{
Name: "image_name",
},
{
Name: "build",
},
{
Name: "local_image_name",
DefaultValue: "=None",
},
{
Name: "registry",
DefaultValue: "=None",
},
{
Name: "skip_push",
DefaultValue: "=None",
},
{
Name: "opts",
DefaultValue: "=None",
},
}
}
// Returns the rendered HTML for the resource's constructor, as well as the specific arguments.
func (mod *modContext) genConstructors(r *schema.Resource, allOptionalInputs bool) (map[string]string, map[string][]formalParam) {
dctx := mod.docGenContext
renderedParams := make(map[string]string)
formalParams := make(map[string][]formalParam)
// Add an extra language for Python's ResourceArg __init__ overload.
langs := append(dctx.supportedLanguages, "pythonargs")
// Add an extra language for Java's ResourceArg overload.
langs = append(langs, "javaargs")
for _, lang := range langs {
var (
paramTemplate string
params []formalParam
)
b := &bytes.Buffer{}
paramSeparatorTemplate := "param_separator"
ps := paramSeparator{}
switch lang {
case "nodejs":
params = mod.genConstructorTS(r, allOptionalInputs)
paramTemplate = "ts_formal_param"
case "go":
params = mod.genConstructorGo(r, allOptionalInputs)
paramTemplate = "go_formal_param"
case "csharp":
params = mod.genConstructorCS(r, allOptionalInputs)
paramTemplate = "csharp_formal_param"
case "java":
fallthrough
case "javaargs":
argsOverload := lang == "javaargs"
params = mod.genConstructorJava(r, argsOverload)
paramTemplate = "java_formal_param"
case "python":
fallthrough
case "pythonargs":
argsOverload := lang == "pythonargs"
params = mod.genConstructorPython(r, allOptionalInputs, argsOverload)
paramTemplate = "py_formal_param"
paramSeparatorTemplate = "py_param_separator"
ps = paramSeparator{Indent: strings.Repeat(" ", len("def (")+len(resourceName(r)))}
case "yaml":
params = mod.genConstructorYaml()
}
if paramTemplate != "" {
for i, p := range params {
if i != 0 {
if err := dctx.templates.ExecuteTemplate(b, paramSeparatorTemplate, ps); err != nil {
panic(err)
}
}
if err := dctx.templates.ExecuteTemplate(b, paramTemplate, p); err != nil {
panic(err)
}
}
}
renderedParams[lang] = b.String()
formalParams[lang] = params
}
return renderedParams, formalParams
}
// getConstructorResourceInfo returns a map of per-language information about
// the resource being constructed.
func (mod *modContext) getConstructorResourceInfo(resourceTypeName, tok string) map[string]propertyType {
dctx := mod.docGenContext
docLangHelper := dctx.getLanguageDocHelper("yaml")
resourceMap := make(map[string]propertyType)
resourceDisplayName := resourceTypeName
for _, lang := range dctx.supportedLanguages {
// Use the module to package lookup to transform the module name to its normalized package name.
modName := mod.getLanguageModuleName(lang)
// Reset the type name back to the display name.
resourceTypeName = resourceDisplayName
switch lang {
case "nodejs", "go", "python", "java":
// Intentionally left blank.
case "csharp":
namespace := title(mod.pkg.Name(), lang)
if ns, ok := dctx.csharpPkgInfo.Namespaces[mod.pkg.Name()]; ok {
namespace = ns
}
if mod.mod == "" {
resourceTypeName = fmt.Sprintf("Pulumi.%s.%s", namespace, resourceTypeName)
break
}
resourceTypeName = fmt.Sprintf("Pulumi.%s.%s.%s", namespace, modName, resourceTypeName)
case "yaml":
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
resourceMap[lang] = propertyType{
Name: resourceTypeName,
DisplayName: docLangHelper.GetLanguageTypeString(def, mod.mod, &schema.ResourceType{Token: tok}, false),
}
continue
default:
panic(fmt.Errorf("cannot generate constructor info for unhandled language %q", lang))
}
parts := strings.Split(resourceTypeName, ".")
displayName := parts[len(parts)-1]
resourceMap[lang] = propertyType{
Name: resourceDisplayName,
DisplayName: displayName,
}
}
return resourceMap
}
func (mod *modContext) getTSLookupParams(r *schema.Resource, stateParam string) []formalParam {
dctx := mod.docGenContext
docLangHelper := dctx.getLanguageDocHelper("nodejs")
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
return []formalParam{
{
Name: "name",
Type: propertyType{
Name: "string",
},
},
{
Name: "id",
Type: propertyType{
Name: "Input<ID>",
Link: docLangHelper.GetDocLinkForPulumiType(def, "ID"),
},
},
{
Name: "state",
OptionalFlag: "?",
Type: propertyType{
Name: stateParam,
},
},
{
Name: "opts",
OptionalFlag: "?",
Type: propertyType{
Name: "CustomResourceOptions",
Link: docLangHelper.GetDocLinkForPulumiType(def, "CustomResourceOptions"),
},
},
}
}
func (mod *modContext) getGoLookupParams(r *schema.Resource, stateParam string) []formalParam {
dctx := mod.docGenContext
docLangHelper := dctx.getLanguageDocHelper("go")
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
return []formalParam{
{
Name: "ctx",
OptionalFlag: "*",
Type: propertyType{
Name: "Context",
Link: docLangHelper.GetDocLinkForPulumiType(def, "Context"),
},
},
{
Name: "name",
Type: propertyType{
Name: "string",
},
},
{
Name: "id",
Type: propertyType{
Name: "IDInput",
Link: docLangHelper.GetDocLinkForPulumiType(def, "IDInput"),
},
},
{
Name: "state",
OptionalFlag: "*",
Type: propertyType{
Name: stateParam,
},
},
{
Name: "opts",
OptionalFlag: "...",
Type: propertyType{
Name: "ResourceOption",
Link: docLangHelper.GetDocLinkForPulumiType(def, "ResourceOption"),
},
},
}
}
func (mod *modContext) getCSLookupParams(r *schema.Resource, stateParam string) []formalParam {
dctx := mod.docGenContext
docLangHelper := dctx.getLanguageDocHelper("csharp")
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
return []formalParam{
{
Name: "name",
Type: propertyType{
Name: "string",
},
},
{
Name: "id",
Type: propertyType{
Name: "Input<string>",
Link: docLangHelper.GetDocLinkForPulumiType(def, "Pulumi.Input"),
},
},
{
Name: "state",
OptionalFlag: "?",
Type: propertyType{
Name: stateParam,
},
},
{
Name: "opts",
OptionalFlag: "?",
DefaultValue: " = null",
Type: propertyType{
Name: "CustomResourceOptions",
Link: docLangHelper.GetDocLinkForPulumiType(def, "Pulumi.CustomResourceOptions"),
},
},
}
}
func (mod *modContext) getJavaLookupParams(r *schema.Resource, stateParam string) []formalParam {
dctx := mod.docGenContext
docLangHelper := dctx.getLanguageDocHelper("java")
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
return []formalParam{
{
Name: "name",
Type: propertyType{
Name: "String",
},
},
{
Name: "id",
Type: propertyType{
Name: "Output<String>",
Link: docLangHelper.GetDocLinkForPulumiType(def, "Output"),
},
},
{
Name: "state",
Type: propertyType{
Name: stateParam,
},
},
{
Name: "options",
Type: propertyType{
Name: "CustomResourceOptions",
Link: docLangHelper.GetDocLinkForPulumiType(def, "CustomResourceOptions"),
},
},
}
}
func (mod *modContext) getPythonLookupParams(r *schema.Resource, stateParam string) []formalParam {
dctx := mod.docGenContext
// The input properties for a resource needs to be exploded as
// individual constructor params.
docLanguageHelper := dctx.getLanguageDocHelper("python")
params := slice.Prealloc[formalParam](len(r.StateInputs.Properties))
for _, p := range r.StateInputs.Properties {
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
typ := docLanguageHelper.GetLanguageTypeString(def, mod.mod, codegen.PlainType(codegen.OptionalType(p)), true /*input*/)
params = append(params, formalParam{
Name: python.PyName(p.Name),
DefaultValue: " = None",
Type: propertyType{
Name: typ,
},
})
}
return params
}
// genLookupParams generates a map of per-language way of rendering the formal parameters of the lookup function
// used to lookup an existing resource.
func (mod *modContext) genLookupParams(r *schema.Resource, stateParam string) map[string]string {
dctx := mod.docGenContext
lookupParams := make(map[string]string)
if r.StateInputs == nil {
return lookupParams
}
for _, lang := range dctx.supportedLanguages {
var (
paramTemplate string
params []formalParam
)
b := &bytes.Buffer{}
paramSeparatorTemplate := "param_separator"
ps := paramSeparator{}
switch lang {
case "nodejs":
params = mod.getTSLookupParams(r, stateParam)
paramTemplate = "ts_formal_param"
case "go":
params = mod.getGoLookupParams(r, stateParam)
paramTemplate = "go_formal_param"
case "csharp":
params = mod.getCSLookupParams(r, stateParam)
paramTemplate = "csharp_formal_param"
case "java":
params = mod.getJavaLookupParams(r, stateParam)
paramTemplate = "java_formal_param"
case "python":
params = mod.getPythonLookupParams(r, stateParam)
paramTemplate = "py_formal_param"
paramSeparatorTemplate = "py_param_separator"
ps = paramSeparator{Indent: strings.Repeat(" ", len("def get("))}
}
n := len(params)
for i, p := range params {
if err := dctx.templates.ExecuteTemplate(b, paramTemplate, p); err != nil {
panic(err)
}
if i != n-1 {
if err := dctx.templates.ExecuteTemplate(b, paramSeparatorTemplate, ps); err != nil {
panic(err)
}
}
}
lookupParams[lang] = b.String()
}
return lookupParams
}
// filterOutputProperties removes the input properties from the output properties list
// (since input props are implicitly output props), returning only "output" props.
func filterOutputProperties(inputProps []*schema.Property, props []*schema.Property) []*schema.Property {
var outputProps []*schema.Property
inputMap := make(map[string]bool, len(inputProps))
for _, p := range inputProps {
inputMap[p.Name] = true
}
for _, p := range props {
if _, found := inputMap[p.Name]; !found {
outputProps = append(outputProps, p)
}
}
return outputProps
}
func (mod *modContext) genResourceHeader(r *schema.Resource) header {
resourceName := resourceName(r)
var metaDescription string
var titleTag string
if mod.mod == "" {
metaDescription = fmt.Sprintf("Documentation for the %s.%s resource "+
"with examples, input properties, output properties, "+
"lookup functions, and supporting types.", mod.pkg.Name(), resourceName)
titleTag = fmt.Sprintf("%s.%s", mod.pkg.Name(), resourceName)
} else {
metaDescription = fmt.Sprintf("Documentation for the %s.%s.%s resource "+
"with examples, input properties, output properties, "+
"lookup functions, and supporting types.", mod.pkg.Name(), mod.mod, resourceName)
titleTag = fmt.Sprintf("%s.%s.%s", mod.pkg.Name(), mod.mod, resourceName)
}
return header{
Title: resourceName,
TitleTag: titleTag,
MetaDesc: metaDescription,
}
}
// genResource is the entrypoint for generating a doc for a resource
// from its Pulumi schema.
func (mod *modContext) genResource(r *schema.Resource) resourceDocArgs {
dctx := mod.docGenContext
// Create a resource module file into which all of this resource's types will go.
name := resourceName(r)
inputProps := make(map[string][]property)
outputProps := make(map[string][]property)
stateInputs := make(map[string][]property)
var filteredOutputProps []*schema.Property
// Provider resources do not have output properties, so there won't be anything to filter.
if !r.IsProvider {
filteredOutputProps = filterOutputProperties(r.InputProperties, r.Properties)
}
// All custom resources have an implicit `id` output property, that we must inject into the docs.
if !r.IsComponent {
filteredOutputProps = append(filteredOutputProps, &schema.Property{
Name: "id",
Comment: "The provider-assigned unique ID for this managed resource.",
Type: schema.StringType,
})
}
for _, lang := range dctx.supportedLanguages {
inputProps[lang] = mod.getProperties(r.InputProperties, lang, true, false, r.IsProvider)
outputProps[lang] = mod.getProperties(filteredOutputProps, lang, false, false, r.IsProvider)
if r.IsProvider {
continue
}
if r.StateInputs != nil {
stateProps := mod.getProperties(r.StateInputs.Properties, lang, true, false, r.IsProvider)
for i := 0; i < len(stateProps); i++ {
id := "state_" + stateProps[i].ID
stateProps[i].ID = id
stateProps[i].Link = "#" + id
}
stateInputs[lang] = stateProps
}
}
allOptionalInputs := true
for _, prop := range r.InputProperties {
// If at least one prop is required, then break.
if prop.IsRequired() {
allOptionalInputs = false
break
}
}
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %s", mod.pkg.Name())
packageDetails := packageDetails{
DisplayName: getPackageDisplayName(def.Name),
Repository: def.Repository,
RepositoryName: getRepositoryName(def.Repository),
License: def.License,
Notes: def.Attribution,
}
renderedCtorParams, typedCtorParams := mod.genConstructors(r, allOptionalInputs)
stateParam := name + "State"
docInfo := dctx.decomposeDocstring(r.Comment)
data := resourceDocArgs{
Header: mod.genResourceHeader(r),
Tool: mod.tool,
Comment: docInfo.description,
DeprecationMessage: r.DeprecationMessage,
ExamplesSection: docInfo.examples,
ImportDocs: docInfo.importDetails,
ConstructorParams: renderedCtorParams,
ConstructorParamsTyped: typedCtorParams,
ConstructorResource: mod.getConstructorResourceInfo(name, r.Token),
ArgsRequired: !allOptionalInputs,
InputProperties: inputProps,
OutputProperties: outputProps,
LookupParams: mod.genLookupParams(r, stateParam),
StateInputs: stateInputs,
StateParam: stateParam,
NestedTypes: mod.genNestedTypes(r, true /*resourceType*/),
Methods: mod.genMethods(r),
PackageDetails: packageDetails,
}
return data
}
func (mod *modContext) getNestedTypes(t schema.Type, types nestedTypeUsageInfo, input bool) {
switch t := t.(type) {
case *schema.InputType:
mod.getNestedTypes(t.ElementType, types, input)
case *schema.OptionalType:
mod.getNestedTypes(t.ElementType, types, input)
case *schema.ArrayType:
mod.getNestedTypes(t.ElementType, types, input)
case *schema.MapType:
mod.getNestedTypes(t.ElementType, types, input)
case *schema.ObjectType:
if types.contains(t.Token, input) {
break
}
types.add(t.Token, input)
for _, p := range t.Properties {
mod.getNestedTypes(p.Type, types, input)
}
case *schema.EnumType:
types.add(t.Token, false)
case *schema.UnionType:
for _, e := range t.ElementTypes {
mod.getNestedTypes(e, types, input)
}
}
}
func (mod *modContext) getTypes(member interface{}, types nestedTypeUsageInfo) {
glog.V(3).Infoln("getting nested types for module", mod.mod)
switch t := member.(type) {
case *schema.ObjectType:
for _, p := range t.Properties {
mod.getNestedTypes(p.Type, types, false)
}
case *schema.Resource:
for _, p := range t.Properties {
mod.getNestedTypes(p.Type, types, false)
}
for _, p := range t.InputProperties {
mod.getNestedTypes(p.Type, types, true)
}
for _, m := range t.Methods {
mod.getTypes(m.Function, types)
}
case *schema.Function:
if t.Inputs != nil && !t.MultiArgumentInputs {
mod.getNestedTypes(t.Inputs, types, true)
}
if t.ReturnType != nil {
if objectType, ok := t.ReturnType.(*schema.ObjectType); ok && objectType != nil {
mod.getNestedTypes(objectType, types, false)
}
}
}
}
// getModuleFileName returns the file name to use for a module.
func (mod *modContext) getModuleFileName() string {
dctx := mod.docGenContext
if !isKubernetesPackage(mod.pkg) {
return mod.mod
}
// For k8s packages, use the Go-language info to get the file name
// for the module.
if override, ok := dctx.goPkgInfo.ModuleToPackage[mod.mod]; ok {
return override
}
return mod.mod
}
// moduleConflictResolver holds module-level information for resolving naming conflicts.
// It shares information with the top-level docGenContext
// to ensure the same name is used across modules that reference each other.
type moduleConflictResolver struct {
dctx *docGenContext
seen map[string]struct{}
}
func (dctx *docGenContext) newModuleConflictResolver() moduleConflictResolver {
return moduleConflictResolver{
dctx: dctx,
seen: map[string]struct{}{},
}
}
// getSafeName returns a documentation name for an item
// that is unique within the module.
//
// if the item has already been resolved by any module,
// the previously-resolved name is returned.
func (r *moduleConflictResolver) getSafeName(name string, item interface{}) string {
if safeName, ok := r.dctx.moduleConflictLinkMap[item]; ok {
return safeName
}
var prefixes []string
switch item.(type) {
case *schema.Resource:
prefixes = []string{"", "res-"}
case *schema.Function:
prefixes = []string{"", "fn-"}
case *modContext:
prefixes = []string{"", "mod-"}
default:
prefixes = []string{""}
}
for _, prefix := range prefixes {
candidate := prefix + name
if _, exists := r.seen[candidate]; exists {
continue
}
r.seen[candidate] = struct{}{}
r.dctx.moduleConflictLinkMap[item] = candidate
return candidate
}
glog.Error("skipping unresolvable duplicate file name: ", name)
return ""
}
func (mod *modContext) gen(fs codegen.Fs) error {
glog.V(4).Infoln("genIndex for", mod.mod)
modName := mod.getModuleFileName()
conflictResolver := mod.docGenContext.newModuleConflictResolver()
def, err := mod.pkg.Definition()
contract.AssertNoErrorf(err, "failed to get definition for package %q", mod.pkg.Name())
modTitle := modName
if modTitle == "" {
// An empty string indicates that this is the root module.
if def.DisplayName != "" {
modTitle = def.DisplayName
} else {
modTitle = getPackageDisplayName(mod.pkg.Name())
}
}
// addFileTemplated executes template tmpl with data,
// and adds a file $dirName/_index.md with the result.
addFileTemplated := func(dirName, tmpl string, data interface{}) error {
var buff bytes.Buffer
if err := mod.docGenContext.templates.ExecuteTemplate(&buff, tmpl, data); err != nil {
return err
}
p := path.Join(modName, dirName, "_index.md")
fs.Add(p, buff.Bytes())
return nil
}
// If there are submodules, list them.
modules := slice.Prealloc[indexEntry](len(mod.children))
for _, mod := range mod.children {
modName := mod.getModuleFileName()
displayName := modFilenameToDisplayName(modName)
safeName := conflictResolver.getSafeName(displayName, mod)
if safeName == "" {
continue // unresolved conflict
}
modules = append(modules, indexEntry{
Link: getModuleLink(safeName),
DisplayName: displayName,
})
}
sortIndexEntries(modules)
// If there are resources in the root, list them.
resources := slice.Prealloc[indexEntry](len(mod.resources))
for _, r := range mod.resources {
title := resourceName(r)
link := getResourceLink(title)
link = conflictResolver.getSafeName(link, r)
if link == "" {
continue // unresolved conflict
}
data := mod.genResource(r)
if err := addFileTemplated(link, "resource.tmpl", data); err != nil {
return err
}
resources = append(resources, indexEntry{
Link: link + "/",
DisplayName: title,
})
}
sortIndexEntries(resources)
// If there are functions in the root, list them.
functions := slice.Prealloc[indexEntry](len(mod.functions))
for _, f := range mod.functions {
name := tokenToName(f.Token)
link := getFunctionLink(name)
link = conflictResolver.getSafeName(link, f)
if link == "" {
continue // unresolved conflict
}
data := mod.genFunction(f)
if err := addFileTemplated(link, "function.tmpl", data); err != nil {
return err
}
functions = append(functions, indexEntry{
Link: link + "/",
DisplayName: strings.Title(name),
})
}
sortIndexEntries(functions)
version := ""
if mod.pkg.Version() != nil {
version = mod.pkg.Version().String()
}
packageDetails := packageDetails{
DisplayName: getPackageDisplayName(def.Name),
Repository: def.Repository,
RepositoryName: getRepositoryName(def.Repository),
License: def.License,
Notes: def.Attribution,
Version: version,
}
var modTitleTag string
var packageDescription string
// The same index.tmpl template is used for both top level package and module pages, if modules not present,
// assume top level package index page when formatting title tags otherwise, if contains modules, assume modules
// top level page when generating title tags.
if len(modules) > 0 {
modTitleTag = fmt.Sprintf("%s Package", getPackageDisplayName(modTitle))
} else {
modTitleTag = fmt.Sprintf("%s.%s", mod.pkg.Name(), modTitle)
packageDescription = fmt.Sprintf("Explore the resources and functions of the %s.%s module.",
mod.pkg.Name(), modTitle)
}
// Generate the index file.
idxData := indexData{
Tool: mod.tool,
PackageDescription: packageDescription,
Title: modTitle,
TitleTag: modTitleTag,
Resources: resources,
Functions: functions,
Modules: modules,
PackageDetails: packageDetails,
}
// If this is the root module, write out the package description.
if mod.mod == "" {
idxData.PackageDescription = mod.pkg.Description()
}
return addFileTemplated("", "index.tmpl", idxData)
}
// indexEntry represents an individual entry on an index page.
type indexEntry struct {
Link string
DisplayName string
}
// indexData represents the index file data to be rendered as _index.md.
type indexData struct {
Tool string
Title string
TitleTag string
PackageDescription string
Functions []indexEntry
Resources []indexEntry
Modules []indexEntry
PackageDetails packageDetails
}
func sortIndexEntries(entries []indexEntry) {
if len(entries) == 0 {
return
}
sort.Slice(entries, func(i, j int) bool {
return entries[i].DisplayName < entries[j].DisplayName
})
}
// getPackageDisplayName uses the title lookup map to look for a
// display name for the given title.
func getPackageDisplayName(title string) string {
// If title not found in titleLookup map, default back to title given.
if val, ok := titleLookup(title); ok {
return val
}
return title
}
// getRepositoryName returns the repository name based on the repository's URL.
func getRepositoryName(repoURL string) string {
return strings.TrimPrefix(repoURL, "https://github.com/")
}
func (dctx *docGenContext) getMod(
pkg schema.PackageReference,
token string,
tokenPkg schema.PackageReference,
modules map[string]*modContext,
tool string,
add bool,
) *modContext {
modName := pkg.TokenToModule(token)
mod, ok := modules[modName]
if !ok {
mod = &modContext{
pkg: pkg,
mod: modName,
tool: tool,
docGenContext: dctx,
}
if modName != "" && codegen.PkgEquals(tokenPkg, pkg) {
parentName := path.Dir(modName)
// If the parent name is blank, it means this is the package-level.
if parentName == "." || parentName == "" {
parentName = ":index:"
} else {
parentName = ":" + parentName + ":"
}
parent := dctx.getMod(pkg, parentName, tokenPkg, modules, tool, add)
if add {
parent.children = append(parent.children, mod)
}
}
// Save the module only if we're adding and it's for the current package.
// This way, modules for external packages are not saved.
if add && tokenPkg == pkg {
modules[modName] = mod
}
}
return mod
}
func (dctx *docGenContext) generateModulesFromSchemaPackage(tool string, pkg *schema.Package) map[string]*modContext {
// Group resources, types, and functions into modules.
modules := map[string]*modContext{}
// Decode language-specific info.
if err := pkg.ImportLanguages(map[string]schema.Language{
"go": go_gen.Importer,
"python": python.Importer,
"csharp": dotnet.Importer,
"nodejs": nodejs.Importer,
}); err != nil {
panic(err)
}
dctx.goPkgInfo, _ = pkg.Language["go"].(go_gen.GoPackageInfo)
dctx.csharpPkgInfo, _ = pkg.Language["csharp"].(dotnet.CSharpPackageInfo)
dctx.nodePkgInfo, _ = pkg.Language["nodejs"].(nodejs.NodePackageInfo)
dctx.pythonPkgInfo, _ = pkg.Language["python"].(python.PackageInfo)
goLangHelper := dctx.getLanguageDocHelper("go").(*go_gen.DocLanguageHelper)
// Generate the Go package map info now, so we can use that to get the type string
// names later.
goLangHelper.GeneratePackagesMap(pkg, tool, dctx.goPkgInfo)
csharpLangHelper := dctx.getLanguageDocHelper("csharp").(*dotnet.DocLanguageHelper)
csharpLangHelper.Namespaces = dctx.csharpPkgInfo.Namespaces
visitObjects := func(r *schema.Resource) {
visitObjectTypes(r.InputProperties, func(t schema.Type) {
switch T := t.(type) {
case *schema.ObjectType:
dctx.getMod(pkg.Reference(), T.Token, T.PackageReference, modules, tool, true).details(T).inputType = true
}
})
if r.StateInputs != nil {
visitObjectTypes(r.StateInputs.Properties, func(t schema.Type) {
switch T := t.(type) {
case *schema.ObjectType:
dctx.getMod(pkg.Reference(), T.Token, T.PackageReference, modules, tool, true).details(T).inputType = true
}
})
}
}
scanResource := func(r *schema.Resource) {
mod := dctx.getMod(pkg.Reference(), r.Token, r.PackageReference, modules, tool, true)
mod.resources = append(mod.resources, r)
visitObjects(r)
}
scanK8SResource := func(r *schema.Resource) {
mod := getKubernetesMod(pkg, r.Token, modules, tool)
mod.resources = append(mod.resources, r)
visitObjects(r)
}
glog.V(3).Infoln("scanning resources")
if isKubernetesPackage(pkg.Reference()) {
scanK8SResource(pkg.Provider)
for _, r := range pkg.Resources {
scanK8SResource(r)
}
} else {
scanResource(pkg.Provider)
for _, r := range pkg.Resources {
scanResource(r)
}
}
glog.V(3).Infoln("done scanning resources")
for _, f := range pkg.Functions {
if !f.IsMethod {
mod := dctx.getMod(pkg.Reference(), f.Token, f.PackageReference, modules, tool, true)
mod.functions = append(mod.functions, f)
}
}
// Find nested types.
for _, t := range pkg.Types {
switch typ := t.(type) {
case *schema.ObjectType:
mod := dctx.getMod(pkg.Reference(), typ.Token, typ.PackageReference, modules, tool, false)
if mod.details(typ).inputType {
mod.inputTypes = append(mod.inputTypes, typ)
}
}
}
return modules
}
func (dctx *docGenContext) initialize(tool string, pkg *schema.Package) {
dctx.templates = template.New("").Funcs(template.FuncMap{
"htmlSafe": func(html string) template.HTML {
// Markdown fragments in the templates need to be rendered as-is,
// so that html/template package doesn't try to inject data into it,
// which will most certainly fail.
//nolint:gosec
return template.HTML(html)
},
"markdownify": func(html string) template.HTML {
// Convert a string of Markdown into HTML.
var buf bytes.Buffer
if err := goldmark.Convert([]byte(html), &buf); err != nil {
glog.Fatalf("rendering Markdown: %v", err)
}
rendered := buf.String()
// Trim surrounding <p></p> tags.
result := strings.TrimSpace(rendered)
result = strings.TrimPrefix(result, "<p>")
result = strings.TrimSuffix(result, "</p>")
// If there are still <p> tags, there are multiple paragraphs,
// in which case use the original rendered string (untrimmed).
if strings.Contains(result, "<p>") {
result = rendered
}
//nolint:gosec
return template.HTML(result)
},
})
defer glog.Flush()
if _, err := dctx.templates.ParseFS(packagedTemplates, "templates/*.tmpl"); err != nil {
glog.Fatalf("initializing templates: %v", err)
}
// Generate the modules from the schema, and for every module
// run the generator functions to generate markdown files.
dctx.setModules(dctx.generateModulesFromSchemaPackage(tool, pkg))
}
func (dctx *docGenContext) generatePackage(tool string, pkg *schema.Package) (map[string][]byte, error) {
if dctx.modules() == nil {
return nil, errors.New("must call Initialize before generating the docs package")
}
defer glog.Flush()
glog.V(3).Infoln("generating package docs now...")
files := codegen.Fs{}
modules := []string{}
modMap := dctx.modules()
for k := range modMap {
modules = append(modules, k)
}
sort.Strings(modules)
for _, mod := range modules {
if err := modMap[mod].gen(files); err != nil {
return nil, err
}
}
return files, nil
}
// GeneratePackageTree returns a navigable structure starting from the top-most module.
func (dctx *docGenContext) generatePackageTree() ([]PackageTreeItem, error) {
if dctx.modules() == nil {
return nil, errors.New("must call Initialize before generating the docs package")
}
defer glog.Flush()
var packageTree []PackageTreeItem
// "" indicates the top-most module.
if rootMod, ok := dctx.modules()[""]; ok {
tree, err := generatePackageTree(*rootMod)
if err != nil {
glog.Errorf("Error generating the package tree for package: %v", err)
}
packageTree = tree
} else {
glog.Error("A root module entry was not found for the package. Cannot generate the package tree...")
}
return packageTree, nil
}
func visitObjectTypes(properties []*schema.Property, visitor func(t schema.Type)) {
codegen.VisitTypeClosure(properties, func(t schema.Type) {
switch st := t.(type) {
case *schema.EnumType, *schema.ObjectType, *schema.ResourceType:
visitor(st)
}
})
}
// Export a default static context so as not to break external
// consumers of this API; prefer *WithContext API internally to ensure
// tests can run in parallel.
var defaultContext = newDocGenContext()
func Initialize(tool string, pkg *schema.Package) {
defaultContext.initialize(tool, pkg)
}
// GeneratePackage generates docs for each resource given the Pulumi
// schema. The returned map contains the filename with path as the key
// and the contents as its value.
func GeneratePackage(tool string, pkg *schema.Package) (map[string][]byte, error) {
return defaultContext.generatePackage(tool, pkg)
}
// GeneratePackageTree returns a navigable structure starting from the top-most module.
func GeneratePackageTree() ([]PackageTreeItem, error) {
return defaultContext.generatePackageTree()
}