2021-10-18 22:18:15 +00:00
|
|
|
// Copyright 2016-2021, Pulumi Corporation.
|
2020-01-21 22:45:48 +00:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
2023-01-06 00:07:45 +00:00
|
|
|
//nolint:lll, goconst
|
2020-01-21 22:45:48 +00:00
|
|
|
package dotnet
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2024-04-22 06:37:50 +00:00
|
|
|
"errors"
|
2020-01-21 22:45:48 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"reflect"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"unicode"
|
|
|
|
|
2024-04-22 06:37:50 +00:00
|
|
|
mapset "github.com/deckarep/golang-set/v2"
|
2021-03-17 13:20:05 +00:00
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/schema"
|
2021-09-08 05:23:30 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/diag"
|
2021-12-02 20:47:33 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/plugin"
|
2023-06-28 16:02:04 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
|
2023-10-20 22:21:58 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
2021-09-08 05:23:30 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/cmdutil"
|
2021-03-17 13:20:05 +00:00
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
2020-01-21 22:45:48 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type typeDetails struct {
|
2021-10-18 22:18:15 +00:00
|
|
|
outputType bool
|
|
|
|
inputType bool
|
|
|
|
stateType bool
|
|
|
|
plainType bool
|
|
|
|
usedInFunctionOutputVersionInputs bool
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2020-04-07 00:01:33 +00:00
|
|
|
// Title converts the input string to a title case
|
|
|
|
// where only the initial letter is upper-cased.
|
|
|
|
func Title(s string) string {
|
2020-01-21 22:45:48 +00:00
|
|
|
if s == "" {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
runes := []rune(s)
|
|
|
|
return string(append([]rune{unicode.ToUpper(runes[0])}, runes[1:]...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func csharpIdentifier(s string) string {
|
2020-05-01 10:45:47 +00:00
|
|
|
// Some schema field names may look like $ref or $schema. Remove the leading $ to make a valid identifier.
|
|
|
|
// This could lead to a clash if both `$foo` and `foo` are defined, but we don't try to de-duplicate now.
|
2023-01-11 15:59:43 +00:00
|
|
|
s = strings.TrimPrefix(s, "$")
|
2020-05-01 10:45:47 +00:00
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
switch s {
|
|
|
|
case "abstract", "as", "base", "bool",
|
|
|
|
"break", "byte", "case", "catch",
|
|
|
|
"char", "checked", "class", "const",
|
|
|
|
"continue", "decimal", "default", "delegate",
|
|
|
|
"do", "double", "else", "enum",
|
|
|
|
"event", "explicit", "extern", "false",
|
|
|
|
"finally", "fixed", "float", "for",
|
|
|
|
"foreach", "goto", "if", "implicit",
|
|
|
|
"in", "int", "interface", "internal",
|
|
|
|
"is", "lock", "long", "namespace",
|
|
|
|
"new", "null", "object", "operator",
|
|
|
|
"out", "override", "params", "private",
|
|
|
|
"protected", "public", "readonly", "ref",
|
|
|
|
"return", "sbyte", "sealed", "short",
|
|
|
|
"sizeof", "stackalloc", "static", "string",
|
|
|
|
"struct", "switch", "this", "throw",
|
|
|
|
"true", "try", "typeof", "uint",
|
|
|
|
"ulong", "unchecked", "unsafe", "ushort",
|
|
|
|
"using", "virtual", "void", "volatile", "while":
|
|
|
|
return "@" + s
|
|
|
|
|
|
|
|
default:
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-14 08:30:25 +00:00
|
|
|
func isImmutableArrayType(t schema.Type, wrapInput bool) bool {
|
|
|
|
_, isArray := t.(*schema.ArrayType)
|
|
|
|
return isArray && !wrapInput
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
func isValueType(t schema.Type) bool {
|
2021-06-24 16:17:55 +00:00
|
|
|
switch t := t.(type) {
|
|
|
|
case *schema.OptionalType:
|
|
|
|
return isValueType(t.ElementType)
|
|
|
|
case *schema.EnumType:
|
2020-01-21 22:45:48 +00:00
|
|
|
return true
|
|
|
|
default:
|
2021-06-24 16:17:55 +00:00
|
|
|
switch t {
|
|
|
|
case schema.BoolType, schema.IntType, schema.NumberType:
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func namespaceName(namespaces map[string]string, name string) string {
|
|
|
|
if ns, ok := namespaces[name]; ok {
|
|
|
|
return ns
|
|
|
|
}
|
2023-10-20 22:21:58 +00:00
|
|
|
|
|
|
|
// name could be a qualified module name so first split on /
|
|
|
|
parts := strings.Split(name, tokens.QNameDelimiter)
|
|
|
|
for i, part := range parts {
|
|
|
|
names := strings.Split(part, "-")
|
|
|
|
for j, name := range names {
|
|
|
|
names[j] = Title(name)
|
|
|
|
}
|
|
|
|
parts[i] = strings.Join(names, "")
|
2022-04-18 09:03:42 +00:00
|
|
|
}
|
2023-10-20 22:21:58 +00:00
|
|
|
return strings.Join(parts, ".")
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type modContext struct {
|
2022-12-08 14:23:07 +00:00
|
|
|
pkg schema.PackageReference
|
2020-05-19 09:41:06 +00:00
|
|
|
mod string
|
|
|
|
propertyNames map[*schema.Property]string
|
|
|
|
types []*schema.ObjectType
|
2020-11-06 17:01:03 +00:00
|
|
|
enums []*schema.EnumType
|
2020-05-19 09:41:06 +00:00
|
|
|
resources []*schema.Resource
|
|
|
|
functions []*schema.Function
|
|
|
|
typeDetails map[*schema.ObjectType]*typeDetails
|
|
|
|
children []*modContext
|
|
|
|
tool string
|
|
|
|
namespaceName string
|
|
|
|
namespaces map[string]string
|
|
|
|
compatibility string
|
|
|
|
dictionaryConstructors bool
|
2021-10-01 18:33:02 +00:00
|
|
|
|
2021-12-02 01:53:18 +00:00
|
|
|
// If types in the Input namespace are used.
|
|
|
|
fullyQualifiedInputs bool
|
|
|
|
|
2021-10-01 18:33:02 +00:00
|
|
|
// Determine whether to lift single-value method return values
|
|
|
|
liftSingleValueMethodReturns bool
|
2022-01-21 20:58:11 +00:00
|
|
|
|
|
|
|
// The root namespace to use, if any.
|
|
|
|
rootNamespace string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) RootNamespace() string {
|
|
|
|
if mod.rootNamespace != "" {
|
|
|
|
return mod.rootNamespace
|
|
|
|
}
|
|
|
|
return "Pulumi"
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) propertyName(p *schema.Property) string {
|
|
|
|
if n, ok := mod.propertyNames[p]; ok {
|
|
|
|
return n
|
|
|
|
}
|
2020-04-07 00:01:33 +00:00
|
|
|
return Title(p.Name)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) details(t *schema.ObjectType) *typeDetails {
|
|
|
|
details, ok := mod.typeDetails[t]
|
|
|
|
if !ok {
|
|
|
|
details = &typeDetails{}
|
|
|
|
mod.typeDetails[t] = details
|
|
|
|
}
|
|
|
|
return details
|
|
|
|
}
|
|
|
|
|
|
|
|
func tokenToName(tok string) string {
|
|
|
|
// token := pkg : module : member
|
|
|
|
// module := path/to/module
|
|
|
|
|
|
|
|
components := strings.Split(tok, ":")
|
|
|
|
contract.Assertf(len(components) == 3, "malformed token %v", tok)
|
2020-04-07 00:01:33 +00:00
|
|
|
return Title(components[2])
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func resourceName(r *schema.Resource) string {
|
|
|
|
if r.IsProvider {
|
|
|
|
return "Provider"
|
|
|
|
}
|
2023-11-29 16:35:08 +00:00
|
|
|
|
|
|
|
if val1, ok := r.Language["csharp"]; ok {
|
|
|
|
val2, ok := val1.(CSharpResourceInfo)
|
|
|
|
contract.Assertf(ok, "dotnet specific settings for resources should be of type CSharpResourceInfo")
|
|
|
|
return Title(val2.Name)
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
return tokenToName(r.Token)
|
|
|
|
}
|
|
|
|
|
2020-04-28 00:47:01 +00:00
|
|
|
func tokenToFunctionName(tok string) string {
|
|
|
|
return tokenToName(tok)
|
|
|
|
}
|
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
func (mod *modContext) isK8sCompatMode() bool {
|
|
|
|
return mod.compatibility == "kubernetes20"
|
|
|
|
}
|
|
|
|
|
2021-04-16 02:03:28 +00:00
|
|
|
func (mod *modContext) isTFCompatMode() bool {
|
|
|
|
return mod.compatibility == "tfbridge20"
|
|
|
|
}
|
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
func (mod *modContext) tokenToNamespace(tok string, qualifier string) string {
|
2020-01-21 22:45:48 +00:00
|
|
|
components := strings.Split(tok, ":")
|
|
|
|
contract.Assertf(len(components) == 3, "malformed token %v", tok)
|
|
|
|
|
2022-01-21 20:58:11 +00:00
|
|
|
pkg, nsName := mod.RootNamespace()+"."+namespaceName(mod.namespaces, components[0]), mod.pkg.TokenToModule(tok)
|
2020-05-19 09:41:06 +00:00
|
|
|
|
|
|
|
if mod.isK8sCompatMode() {
|
2020-12-16 16:58:42 +00:00
|
|
|
if qualifier != "" {
|
|
|
|
return pkg + ".Types." + qualifier + "." + namespaceName(mod.namespaces, nsName)
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
typ := pkg
|
|
|
|
if nsName != "" {
|
|
|
|
typ += "." + namespaceName(mod.namespaces, nsName)
|
|
|
|
}
|
|
|
|
if qualifier != "" {
|
|
|
|
typ += "." + qualifier
|
|
|
|
}
|
|
|
|
return typ
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2021-04-16 02:03:28 +00:00
|
|
|
func (mod *modContext) typeName(t *schema.ObjectType, state, input, args bool) string {
|
|
|
|
name := tokenToName(t.Token)
|
|
|
|
if state {
|
|
|
|
return name + "GetArgs"
|
|
|
|
}
|
2021-04-16 18:30:25 +00:00
|
|
|
if !mod.isTFCompatMode() && !mod.isK8sCompatMode() {
|
2021-04-16 02:03:28 +00:00
|
|
|
if args {
|
|
|
|
return name + "Args"
|
|
|
|
}
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
2021-10-18 22:18:15 +00:00
|
|
|
case input && args && mod.details(t).usedInFunctionOutputVersionInputs:
|
|
|
|
return name + "InputArgs"
|
2021-04-16 02:03:28 +00:00
|
|
|
case input:
|
|
|
|
return name + "Args"
|
2021-04-19 23:40:39 +00:00
|
|
|
case mod.details(t).plainType:
|
2021-04-16 02:03:28 +00:00
|
|
|
return name + "Result"
|
|
|
|
}
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
func isInputType(t schema.Type) bool {
|
|
|
|
if optional, ok := t.(*schema.OptionalType); ok {
|
|
|
|
t = optional.ElementType
|
|
|
|
}
|
|
|
|
_, isInputType := t.(*schema.InputType)
|
|
|
|
return isInputType
|
|
|
|
}
|
|
|
|
|
|
|
|
func ignoreOptional(t *schema.OptionalType, requireInitializers bool) bool {
|
|
|
|
switch t := t.ElementType.(type) {
|
|
|
|
case *schema.InputType:
|
|
|
|
switch t.ElementType.(type) {
|
|
|
|
case *schema.ArrayType, *schema.MapType:
|
|
|
|
return true
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
case *schema.ArrayType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return !requireInitializers
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-07-22 20:11:33 +00:00
|
|
|
func simplifyInputUnion(union *schema.UnionType) *schema.UnionType {
|
|
|
|
elements := make([]schema.Type, len(union.ElementTypes))
|
|
|
|
for i, et := range union.ElementTypes {
|
|
|
|
if input, ok := et.(*schema.InputType); ok {
|
|
|
|
switch input.ElementType.(type) {
|
|
|
|
case *schema.ArrayType, *schema.MapType:
|
|
|
|
// Instead of just replacing Input<{Array,Map}<T>> with {Array,Map}<T>, replace it with
|
|
|
|
// {Array,Map}<Plain(T)>. This matches the behavior of typeString when presented with an
|
|
|
|
// Input<{Array,Map}<T>>.
|
|
|
|
elements[i] = codegen.PlainType(input.ElementType)
|
|
|
|
default:
|
|
|
|
elements[i] = input.ElementType
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
elements[i] = et
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return &schema.UnionType{
|
|
|
|
ElementTypes: elements,
|
|
|
|
DefaultType: union.DefaultType,
|
|
|
|
Discriminator: union.Discriminator,
|
|
|
|
Mapping: union.Mapping,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
func (mod *modContext) unionTypeString(t *schema.UnionType, qualifier string, input, wrapInput, state, requireInitializers bool) string {
|
2022-11-10 17:49:12 +00:00
|
|
|
elementTypeSet := codegen.StringSet{}
|
2021-06-24 16:17:55 +00:00
|
|
|
var elementTypes []string
|
|
|
|
for _, e := range t.ElementTypes {
|
|
|
|
// If this is an output and a "relaxed" enum, emit the type as the underlying primitive type rather than the union.
|
|
|
|
// Eg. Output<string> rather than Output<Union<EnumType, string>>
|
|
|
|
if typ, ok := e.(*schema.EnumType); ok && !input {
|
|
|
|
return mod.typeString(typ.ElementType, qualifier, input, state, requireInitializers)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
et := mod.typeString(e, qualifier, input, state, false)
|
2022-11-10 17:49:12 +00:00
|
|
|
if !elementTypeSet.Has(et) {
|
|
|
|
elementTypeSet.Add(et)
|
2021-06-24 16:17:55 +00:00
|
|
|
elementTypes = append(elementTypes, et)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch len(elementTypes) {
|
|
|
|
case 1:
|
|
|
|
if wrapInput {
|
|
|
|
return fmt.Sprintf("Input<%s>", elementTypes[0])
|
|
|
|
}
|
|
|
|
return elementTypes[0]
|
|
|
|
case 2:
|
|
|
|
unionT := "Union"
|
|
|
|
if wrapInput {
|
|
|
|
unionT = "InputUnion"
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%s<%s>", unionT, strings.Join(elementTypes, ", "))
|
|
|
|
default:
|
|
|
|
return "object"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) typeString(t schema.Type, qualifier string, input, state, requireInitializers bool) string {
|
|
|
|
switch t := t.(type) {
|
|
|
|
case *schema.OptionalType:
|
|
|
|
elem := mod.typeString(t.ElementType, qualifier, input, state, requireInitializers)
|
|
|
|
if ignoreOptional(t, requireInitializers) {
|
|
|
|
return elem
|
|
|
|
}
|
|
|
|
return elem + "?"
|
|
|
|
case *schema.InputType:
|
|
|
|
inputType := "Input"
|
|
|
|
elem := t.ElementType
|
|
|
|
switch e := t.ElementType.(type) {
|
|
|
|
case *schema.ArrayType:
|
2021-07-07 23:36:27 +00:00
|
|
|
inputType, elem = "InputList", codegen.PlainType(e.ElementType)
|
2021-06-24 16:17:55 +00:00
|
|
|
case *schema.MapType:
|
2021-07-07 23:36:27 +00:00
|
|
|
inputType, elem = "InputMap", codegen.PlainType(e.ElementType)
|
2020-01-21 22:45:48 +00:00
|
|
|
default:
|
2021-06-24 16:17:55 +00:00
|
|
|
if e == schema.JSONType {
|
|
|
|
return "InputJson"
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
if union, ok := elem.(*schema.UnionType); ok {
|
2021-07-22 20:11:33 +00:00
|
|
|
union = simplifyInputUnion(union)
|
2021-06-24 16:17:55 +00:00
|
|
|
if inputType == "Input" {
|
|
|
|
return mod.unionTypeString(union, qualifier, input, true, state, requireInitializers)
|
|
|
|
}
|
|
|
|
elem = union
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%s<%s>", inputType, mod.typeString(elem, qualifier, input, state, requireInitializers))
|
|
|
|
case *schema.EnumType:
|
|
|
|
return fmt.Sprintf("%s.%s", mod.tokenToNamespace(t.Token, ""), tokenToName(t.Token))
|
|
|
|
case *schema.ArrayType:
|
|
|
|
listType := "ImmutableArray"
|
|
|
|
if requireInitializers {
|
|
|
|
listType = "List"
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%v<%v>", listType, mod.typeString(t.ElementType, qualifier, input, state, false))
|
|
|
|
case *schema.MapType:
|
|
|
|
mapType := "ImmutableDictionary"
|
|
|
|
if requireInitializers {
|
|
|
|
mapType = "Dictionary"
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%v<string, %v>", mapType, mod.typeString(t.ElementType, qualifier, input, state, false))
|
2020-01-21 22:45:48 +00:00
|
|
|
case *schema.ObjectType:
|
2020-11-20 20:09:34 +00:00
|
|
|
namingCtx := mod
|
2022-12-08 14:23:07 +00:00
|
|
|
if !codegen.PkgEquals(t.PackageReference, mod.pkg) {
|
2021-01-26 19:02:34 +00:00
|
|
|
// If object type belongs to another package, we apply naming conventions from that package,
|
2020-11-20 20:09:34 +00:00
|
|
|
// including namespace naming and compatibility mode.
|
2022-12-08 14:23:07 +00:00
|
|
|
extPkg := t.PackageReference
|
2020-11-20 20:09:34 +00:00
|
|
|
var info CSharpPackageInfo
|
2022-12-08 14:23:07 +00:00
|
|
|
def, err := extPkg.Definition()
|
2023-02-17 01:23:09 +00:00
|
|
|
contract.AssertNoErrorf(err, "error loading definition for package %q", extPkg.Name())
|
|
|
|
contract.AssertNoErrorf(def.ImportLanguages(map[string]schema.Language{"csharp": Importer}),
|
|
|
|
"error importing csharp for package %q", extPkg.Name())
|
2022-12-08 14:23:07 +00:00
|
|
|
if v, ok := def.Language["csharp"].(CSharpPackageInfo); ok {
|
2020-11-20 20:09:34 +00:00
|
|
|
info = v
|
|
|
|
}
|
|
|
|
namingCtx = &modContext{
|
2021-01-26 19:02:34 +00:00
|
|
|
pkg: extPkg,
|
2020-11-20 20:09:34 +00:00
|
|
|
namespaces: info.Namespaces,
|
2022-01-21 20:58:11 +00:00
|
|
|
rootNamespace: info.GetRootNamespace(),
|
2020-11-20 20:09:34 +00:00
|
|
|
compatibility: info.Compatibility,
|
|
|
|
}
|
|
|
|
}
|
2021-06-24 16:17:55 +00:00
|
|
|
typ := namingCtx.tokenToNamespace(t.Token, qualifier)
|
2020-11-20 20:09:34 +00:00
|
|
|
if (typ == namingCtx.namespaceName && qualifier == "") || typ == namingCtx.namespaceName+"."+qualifier {
|
2020-01-21 22:45:48 +00:00
|
|
|
typ = qualifier
|
|
|
|
}
|
2021-12-02 01:53:18 +00:00
|
|
|
if typ == "Inputs" && mod.fullyQualifiedInputs {
|
2023-12-12 12:19:42 +00:00
|
|
|
typ = mod.namespaceName + ".Inputs"
|
2021-12-02 01:53:18 +00:00
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
if typ != "" {
|
|
|
|
typ += "."
|
|
|
|
}
|
2021-06-24 16:17:55 +00:00
|
|
|
return typ + mod.typeName(t, state, input, t.IsInputShape())
|
2020-10-13 18:33:22 +00:00
|
|
|
case *schema.ResourceType:
|
2020-11-23 20:28:00 +00:00
|
|
|
if strings.HasPrefix(t.Token, "pulumi:providers:") {
|
|
|
|
pkgName := strings.TrimPrefix(t.Token, "pulumi:providers:")
|
2022-01-21 20:58:11 +00:00
|
|
|
return fmt.Sprintf("%s.%s.Provider", mod.RootNamespace(), namespaceName(mod.namespaces, pkgName))
|
2021-06-24 16:17:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
namingCtx := mod
|
2022-12-08 14:23:07 +00:00
|
|
|
if t.Resource != nil && !codegen.PkgEquals(t.Resource.PackageReference, mod.pkg) {
|
2021-06-24 16:17:55 +00:00
|
|
|
// If resource type belongs to another package, we apply naming conventions from that package,
|
|
|
|
// including namespace naming and compatibility mode.
|
2022-12-08 14:23:07 +00:00
|
|
|
extPkg := t.Resource.PackageReference
|
2021-06-24 16:17:55 +00:00
|
|
|
var info CSharpPackageInfo
|
2022-12-08 14:23:07 +00:00
|
|
|
def, err := extPkg.Definition()
|
2023-02-17 01:23:09 +00:00
|
|
|
contract.AssertNoErrorf(err, "error loading definition for package %q", extPkg.Name())
|
|
|
|
contract.AssertNoErrorf(def.ImportLanguages(map[string]schema.Language{"csharp": Importer}),
|
|
|
|
"error importing csharp for package %q", extPkg.Name())
|
2022-12-08 14:23:07 +00:00
|
|
|
if v, ok := def.Language["csharp"].(CSharpPackageInfo); ok {
|
2021-06-24 16:17:55 +00:00
|
|
|
info = v
|
2021-01-26 19:02:34 +00:00
|
|
|
}
|
2021-06-24 16:17:55 +00:00
|
|
|
namingCtx = &modContext{
|
|
|
|
pkg: extPkg,
|
|
|
|
namespaces: info.Namespaces,
|
2022-01-21 20:58:11 +00:00
|
|
|
rootNamespace: info.GetRootNamespace(),
|
2021-06-24 16:17:55 +00:00
|
|
|
compatibility: info.Compatibility,
|
2020-11-23 20:28:00 +00:00
|
|
|
}
|
2020-10-13 18:33:22 +00:00
|
|
|
}
|
2021-06-24 16:17:55 +00:00
|
|
|
typ := namingCtx.tokenToNamespace(t.Token, "")
|
|
|
|
if typ != "" {
|
|
|
|
typ += "."
|
|
|
|
}
|
2023-11-29 16:35:08 +00:00
|
|
|
return typ + resourceName(t.Resource)
|
2020-01-21 22:45:48 +00:00
|
|
|
case *schema.TokenType:
|
|
|
|
// Use the underlying type for now.
|
|
|
|
if t.UnderlyingType != nil {
|
2021-06-24 16:17:55 +00:00
|
|
|
return mod.typeString(t.UnderlyingType, qualifier, input, state, requireInitializers)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
typ := tokenToName(t.Token)
|
2020-05-19 09:41:06 +00:00
|
|
|
if ns := mod.tokenToNamespace(t.Token, qualifier); ns != mod.namespaceName {
|
2020-01-21 22:45:48 +00:00
|
|
|
typ = ns + "." + typ
|
|
|
|
}
|
2021-06-24 16:17:55 +00:00
|
|
|
return typ
|
2020-01-21 22:45:48 +00:00
|
|
|
case *schema.UnionType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return mod.unionTypeString(t, qualifier, input, false, state, requireInitializers)
|
2020-01-21 22:45:48 +00:00
|
|
|
default:
|
|
|
|
switch t {
|
|
|
|
case schema.BoolType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "bool"
|
2020-01-21 22:45:48 +00:00
|
|
|
case schema.IntType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "int"
|
2020-01-21 22:45:48 +00:00
|
|
|
case schema.NumberType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "double"
|
2020-01-21 22:45:48 +00:00
|
|
|
case schema.StringType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "string"
|
2020-01-21 22:45:48 +00:00
|
|
|
case schema.ArchiveType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "Archive"
|
2020-01-21 22:45:48 +00:00
|
|
|
case schema.AssetType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "AssetOrArchive"
|
2020-05-19 09:41:06 +00:00
|
|
|
case schema.JSONType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "System.Text.Json.JsonElement"
|
2020-01-21 22:45:48 +00:00
|
|
|
case schema.AnyType:
|
2021-06-24 16:17:55 +00:00
|
|
|
return "object"
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
panic(fmt.Errorf("unexpected type %T", t))
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var docCommentEscaper = strings.NewReplacer(
|
|
|
|
`&`, "&",
|
|
|
|
`<`, "<",
|
|
|
|
`>`, ">",
|
|
|
|
)
|
|
|
|
|
2021-07-27 23:42:17 +00:00
|
|
|
func printComment(w io.Writer, comment, indent string) {
|
|
|
|
printCommentWithOptions(w, comment, indent, true /*escape*/)
|
|
|
|
}
|
|
|
|
|
|
|
|
func printCommentWithOptions(w io.Writer, comment, indent string, escape bool) {
|
|
|
|
if escape {
|
|
|
|
comment = docCommentEscaper.Replace(comment)
|
|
|
|
}
|
|
|
|
|
|
|
|
lines := strings.Split(comment, "\n")
|
2020-01-21 22:45:48 +00:00
|
|
|
for len(lines) > 0 && lines[len(lines)-1] == "" {
|
|
|
|
lines = lines[:len(lines)-1]
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(lines) > 0 {
|
|
|
|
fmt.Fprintf(w, "%s/// <summary>\n", indent)
|
|
|
|
for _, l := range lines {
|
|
|
|
fmt.Fprintf(w, "%s/// %s\n", indent, l)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "%s/// </summary>\n", indent)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type plainType struct {
|
|
|
|
mod *modContext
|
|
|
|
res *schema.Resource
|
|
|
|
name string
|
|
|
|
comment string
|
2021-07-27 23:42:17 +00:00
|
|
|
unescapeComment bool
|
2020-01-21 22:45:48 +00:00
|
|
|
baseClass string
|
|
|
|
propertyTypeQualifier string
|
|
|
|
properties []*schema.Property
|
2021-04-16 02:03:28 +00:00
|
|
|
args bool
|
2020-01-21 22:45:48 +00:00
|
|
|
state bool
|
2021-10-01 18:33:02 +00:00
|
|
|
internal bool
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
func (pt *plainType) genInputPropertyAttribute(w io.Writer, indent string, prop *schema.Property) {
|
2020-01-21 22:45:48 +00:00
|
|
|
wireName := prop.Name
|
|
|
|
attributeArgs := ""
|
2021-06-24 16:17:55 +00:00
|
|
|
if prop.IsRequired() {
|
2020-01-21 22:45:48 +00:00
|
|
|
attributeArgs = ", required: true"
|
|
|
|
}
|
2020-05-27 08:14:38 +00:00
|
|
|
if pt.res != nil && pt.res.IsProvider {
|
|
|
|
json := true
|
2021-06-24 16:17:55 +00:00
|
|
|
typ := codegen.UnwrapType(prop.Type)
|
|
|
|
if typ == schema.StringType {
|
2020-05-27 08:14:38 +00:00
|
|
|
json = false
|
2021-06-24 16:17:55 +00:00
|
|
|
} else if t, ok := typ.(*schema.TokenType); ok && t.UnderlyingType == schema.StringType {
|
2020-05-27 08:14:38 +00:00
|
|
|
json = false
|
|
|
|
}
|
|
|
|
if json {
|
|
|
|
attributeArgs += ", json: true"
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2021-10-18 22:18:15 +00:00
|
|
|
fmt.Fprintf(w, "%s[Input(\"%s\"%s)]\n", indent, wireName, attributeArgs)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pt *plainType) genInputProperty(w io.Writer, prop *schema.Property, indent string, generateInputAttribute bool) {
|
|
|
|
propertyName := pt.mod.propertyName(prop)
|
|
|
|
propertyType := pt.mod.typeString(prop.Type, pt.propertyTypeQualifier, true, pt.state, false)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
indent = strings.Repeat(indent, 2)
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
// Next generate the input property itself. The way this is generated depends on the type of the property:
|
|
|
|
// complex types like lists and maps need a backing field.
|
2021-05-26 22:00:51 +00:00
|
|
|
needsBackingField := false
|
2021-06-24 16:17:55 +00:00
|
|
|
switch codegen.UnwrapType(prop.Type).(type) {
|
2020-01-21 22:45:48 +00:00
|
|
|
case *schema.ArrayType, *schema.MapType:
|
2021-05-26 22:00:51 +00:00
|
|
|
needsBackingField = true
|
|
|
|
}
|
|
|
|
if prop.Secret {
|
|
|
|
needsBackingField = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next generate the input property itself. The way this is generated depends on the type of the property:
|
|
|
|
// complex types like lists and maps need a backing field. Secret properties also require a backing field.
|
|
|
|
if needsBackingField {
|
2020-01-21 22:45:48 +00:00
|
|
|
backingFieldName := "_" + prop.Name
|
2021-06-24 16:17:55 +00:00
|
|
|
requireInitializers := !pt.args || !isInputType(prop.Type)
|
|
|
|
backingFieldType := pt.mod.typeString(codegen.RequiredType(prop), pt.propertyTypeQualifier, true, pt.state, requireInitializers)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
if generateInputAttribute {
|
|
|
|
pt.genInputPropertyAttribute(w, indent, prop)
|
|
|
|
}
|
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
fmt.Fprintf(w, "%sprivate %s? %s;\n", indent, backingFieldType, backingFieldName)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
if prop.Comment != "" {
|
|
|
|
fmt.Fprintf(w, "\n")
|
2020-11-06 17:01:03 +00:00
|
|
|
printComment(w, prop.Comment, indent)
|
2020-02-06 17:29:06 +00:00
|
|
|
}
|
2020-11-06 17:01:03 +00:00
|
|
|
printObsoleteAttribute(w, prop.DeprecationMessage, indent)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
switch codegen.UnwrapType(prop.Type).(type) {
|
2021-05-26 22:00:51 +00:00
|
|
|
case *schema.ArrayType, *schema.MapType:
|
|
|
|
// Note that we use the backing field type--which is just the property type without any nullable annotation--to
|
|
|
|
// ensure that the user does not see warnings when initializing these properties using object or collection
|
|
|
|
// initializers.
|
|
|
|
fmt.Fprintf(w, "%spublic %s %s\n", indent, backingFieldType, propertyName)
|
|
|
|
fmt.Fprintf(w, "%s{\n", indent)
|
|
|
|
fmt.Fprintf(w, "%s get => %[2]s ?? (%[2]s = new %[3]s());\n", indent, backingFieldName, backingFieldType)
|
|
|
|
default:
|
|
|
|
fmt.Fprintf(w, "%spublic %s? %s\n", indent, backingFieldType, propertyName)
|
|
|
|
fmt.Fprintf(w, "%s{\n", indent)
|
|
|
|
fmt.Fprintf(w, "%s get => %s;\n", indent, backingFieldName)
|
|
|
|
}
|
2022-09-23 19:44:31 +00:00
|
|
|
if prop.Secret && isInputType(prop.Type) {
|
2021-05-26 22:00:51 +00:00
|
|
|
fmt.Fprintf(w, "%s set\n", indent)
|
|
|
|
fmt.Fprintf(w, "%s {\n", indent)
|
|
|
|
// Since we can't directly assign the Output from CreateSecret to the property, use an Output.All or
|
|
|
|
// Output.Tuple to enable the secret flag on the data. (If any input to the All/Tuple is secret, then the
|
|
|
|
// Output will also be secret.)
|
2021-06-24 16:17:55 +00:00
|
|
|
switch t := codegen.UnwrapType(prop.Type).(type) {
|
2021-05-26 22:00:51 +00:00
|
|
|
case *schema.ArrayType:
|
2022-09-26 16:57:05 +00:00
|
|
|
elemType := pt.mod.typeString(codegen.PlainType(t.ElementType), pt.propertyTypeQualifier, true, pt.state, false)
|
|
|
|
fmt.Fprintf(w, "%s var emptySecret = Output.CreateSecret(ImmutableArray.Create<%s>());\n", indent, elemType)
|
2021-05-26 22:00:51 +00:00
|
|
|
fmt.Fprintf(w, "%s %s = Output.All(value, emptySecret).Apply(v => v[0]);\n", indent, backingFieldName)
|
|
|
|
case *schema.MapType:
|
2022-09-26 16:57:05 +00:00
|
|
|
elemType := pt.mod.typeString(codegen.PlainType(t.ElementType), pt.propertyTypeQualifier, true, pt.state, false)
|
|
|
|
fmt.Fprintf(w, "%s var emptySecret = Output.CreateSecret(ImmutableDictionary.Create<string, %s>());\n", indent, elemType)
|
2021-05-26 22:00:51 +00:00
|
|
|
fmt.Fprintf(w, "%s %s = Output.All(value, emptySecret).Apply(v => v[0]);\n", indent, backingFieldName)
|
|
|
|
default:
|
|
|
|
fmt.Fprintf(w, "%s var emptySecret = Output.CreateSecret(0);\n", indent)
|
|
|
|
fmt.Fprintf(w, "%s %s = Output.Tuple<%s?, int>(value, emptySecret).Apply(t => t.Item1);\n", indent, backingFieldName, backingFieldType)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "%s }\n", indent)
|
|
|
|
} else {
|
|
|
|
fmt.Fprintf(w, "%s set => %s = value;\n", indent, backingFieldName)
|
|
|
|
}
|
2020-11-06 17:01:03 +00:00
|
|
|
fmt.Fprintf(w, "%s}\n", indent)
|
2021-05-26 22:00:51 +00:00
|
|
|
} else {
|
2020-01-21 22:45:48 +00:00
|
|
|
initializer := ""
|
2021-06-24 16:17:55 +00:00
|
|
|
if prop.IsRequired() && !isValueType(prop.Type) {
|
2020-01-21 22:45:48 +00:00
|
|
|
initializer = " = null!;"
|
|
|
|
}
|
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
printComment(w, prop.Comment, indent)
|
2021-10-18 22:18:15 +00:00
|
|
|
|
|
|
|
if generateInputAttribute {
|
|
|
|
pt.genInputPropertyAttribute(w, indent, prop)
|
|
|
|
}
|
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
fmt.Fprintf(w, "%spublic %s %s { get; set; }%s\n", indent, propertyType, propertyName, initializer)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
// Set to avoid generating a class with the same name twice.
|
|
|
|
var generatedTypes = codegen.Set{}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
func (pt *plainType) genInputType(w io.Writer, level int) error {
|
2021-10-18 22:18:15 +00:00
|
|
|
return pt.genInputTypeWithFlags(w, level, true /* generateInputAttributes */)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pt *plainType) genInputTypeWithFlags(w io.Writer, level int, generateInputAttributes bool) error {
|
2020-05-19 09:41:06 +00:00
|
|
|
// The way the legacy codegen for kubernetes is structured, inputs for a resource args type and resource args
|
|
|
|
// subtype could become a single class because of the name + namespace clash. We use a set of generated types
|
|
|
|
// to prevent generating classes with equal full names in multiple files. The check should be removed if we
|
|
|
|
// ever change the namespacing in the k8s SDK to the standard one.
|
|
|
|
if pt.mod.isK8sCompatMode() {
|
|
|
|
key := pt.mod.namespaceName + pt.name
|
|
|
|
if generatedTypes.Has(key) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
generatedTypes.Add(key)
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
indent := strings.Repeat(" ", level)
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
sealed := "sealed "
|
|
|
|
if pt.mod.isK8sCompatMode() && (pt.res == nil || !pt.res.IsProvider) {
|
|
|
|
sealed = ""
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Open the class.
|
2021-07-27 23:42:17 +00:00
|
|
|
printCommentWithOptions(w, pt.comment, indent, !pt.unescapeComment)
|
2021-10-18 22:18:15 +00:00
|
|
|
|
|
|
|
var suffix string
|
|
|
|
if pt.baseClass != "" {
|
2023-12-12 12:19:42 +00:00
|
|
|
suffix = " : global::Pulumi." + pt.baseClass
|
2021-10-18 22:18:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "%spublic %sclass %s%s\n", indent, sealed, pt.name, suffix)
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "%s{\n", indent)
|
|
|
|
|
|
|
|
// Declare each input property.
|
|
|
|
for _, p := range pt.properties {
|
2021-10-18 22:18:15 +00:00
|
|
|
pt.genInputProperty(w, p, indent, generateInputAttributes)
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate a constructor that will set default values.
|
|
|
|
fmt.Fprintf(w, "%s public %s()\n", indent, pt.name)
|
|
|
|
fmt.Fprintf(w, "%s {\n", indent)
|
|
|
|
for _, prop := range pt.properties {
|
|
|
|
if prop.DefaultValue != nil {
|
|
|
|
dv, err := pt.mod.getDefaultValue(prop.DefaultValue, prop.Type)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
propertyName := pt.mod.propertyName(prop)
|
|
|
|
fmt.Fprintf(w, "%s %s = %s;\n", indent, propertyName, dv)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "%s }\n", indent)
|
|
|
|
|
2022-07-27 09:23:34 +00:00
|
|
|
// override Empty static property from inherited ResourceArgs
|
|
|
|
// and make it return a concrete args type instead of inherited ResourceArgs
|
|
|
|
fmt.Fprintf(w, "%s public static new %s Empty => new %s();\n", indent, pt.name, pt.name)
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Close the class.
|
|
|
|
fmt.Fprintf(w, "%s}\n", indent)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pt *plainType) genOutputType(w io.Writer, level int) {
|
|
|
|
indent := strings.Repeat(" ", level)
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Open the class and attribute it appropriately.
|
2021-07-27 23:42:17 +00:00
|
|
|
printCommentWithOptions(w, pt.comment, indent, !pt.unescapeComment)
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "%s[OutputType]\n", indent)
|
2021-10-01 18:33:02 +00:00
|
|
|
|
|
|
|
visibility := "public"
|
|
|
|
if pt.internal {
|
|
|
|
visibility = "internal"
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "%s%s sealed class %s\n", indent, visibility, pt.name)
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "%s{\n", indent)
|
|
|
|
|
|
|
|
// Generate each output field.
|
|
|
|
for _, prop := range pt.properties {
|
|
|
|
fieldName := pt.mod.propertyName(prop)
|
2021-06-24 16:17:55 +00:00
|
|
|
typ := prop.Type
|
|
|
|
if !prop.IsRequired() && pt.mod.isK8sCompatMode() {
|
|
|
|
typ = codegen.RequiredType(prop)
|
|
|
|
}
|
|
|
|
fieldType := pt.mod.typeString(typ, pt.propertyTypeQualifier, false, false, false)
|
2020-01-21 22:45:48 +00:00
|
|
|
printComment(w, prop.Comment, indent+" ")
|
|
|
|
fmt.Fprintf(w, "%s public readonly %s %s;\n", indent, fieldType, fieldName)
|
|
|
|
}
|
|
|
|
if len(pt.properties) > 0 {
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate an appropriately-attributed constructor that will set this types' fields.
|
|
|
|
fmt.Fprintf(w, "%s [OutputConstructor]\n", indent)
|
|
|
|
fmt.Fprintf(w, "%s private %s(", indent, pt.name)
|
|
|
|
|
|
|
|
// Generate the constructor parameters.
|
|
|
|
for i, prop := range pt.properties {
|
|
|
|
paramName := csharpIdentifier(prop.Name)
|
2021-06-24 16:17:55 +00:00
|
|
|
typ := prop.Type
|
|
|
|
if !prop.IsRequired() && pt.mod.isK8sCompatMode() {
|
|
|
|
typ = codegen.RequiredType(prop)
|
|
|
|
}
|
|
|
|
paramType := pt.mod.typeString(typ, pt.propertyTypeQualifier, false, false, false)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
terminator := ""
|
|
|
|
if i != len(pt.properties)-1 {
|
|
|
|
terminator = ",\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
paramDef := fmt.Sprintf("%s %s%s", paramType, paramName, terminator)
|
|
|
|
if len(pt.properties) > 1 {
|
|
|
|
paramDef = fmt.Sprintf("\n%s %s", indent, paramDef)
|
|
|
|
}
|
|
|
|
fmt.Fprint(w, paramDef)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, ")\n")
|
|
|
|
|
|
|
|
// Generate the constructor body.
|
|
|
|
fmt.Fprintf(w, "%s {\n", indent)
|
|
|
|
for _, prop := range pt.properties {
|
|
|
|
paramName := csharpIdentifier(prop.Name)
|
|
|
|
fieldName := pt.mod.propertyName(prop)
|
2020-05-01 10:45:47 +00:00
|
|
|
if fieldName == paramName {
|
|
|
|
// Avoid a no-op in case of field and property name collision.
|
|
|
|
fieldName = "this." + fieldName
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "%s %s = %s;\n", indent, fieldName, paramName)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "%s }\n", indent)
|
|
|
|
|
|
|
|
// Close the class.
|
|
|
|
fmt.Fprintf(w, "%s}\n", indent)
|
|
|
|
}
|
|
|
|
|
|
|
|
func primitiveValue(value interface{}) (string, error) {
|
|
|
|
v := reflect.ValueOf(value)
|
|
|
|
if v.Kind() == reflect.Interface {
|
|
|
|
v = v.Elem()
|
|
|
|
}
|
|
|
|
|
turn on the golangci-lint exhaustive linter (#15028)
Turn on the golangci-lint exhaustive linter. This is the first step
towards catching more missing cases during development rather than
in tests, or in production.
This might be best reviewed commit-by-commit, as the first commit turns
on the linter with the `default-signifies-exhaustive: true` option set,
which requires a lot less changes in the current codebase.
I think it's probably worth doing the second commit as well, as that
will get us the real benefits, even though we end up with a little bit
more churn. However it means all the `switch` statements are covered,
which isn't the case after the first commit, since we do have a lot of
`default` statements that just call `assert.Fail`.
Fixes #14601
## Checklist
- [x] I have run `make tidy` to update any new dependencies
- [x] I have run `make lint` to verify my code passes the lint check
- [x] I have formatted my code using `gofumpt`
<!--- Please provide details if the checkbox below is to be left
unchecked. -->
- [ ] I have added tests that prove my fix is effective or that my
feature works
<!---
User-facing changes require a CHANGELOG entry.
-->
- [ ] I have run `make changelog` and committed the
`changelog/pending/<file>` documenting my change
<!--
If the change(s) in this PR is a modification of an existing call to the
Pulumi Cloud,
then the service should honor older versions of the CLI where this
change would not exist.
You must then bump the API version in
/pkg/backend/httpstate/client/api.go, as well as add
it to the service.
-->
- [ ] Yes, there are changes in this PR that warrants bumping the Pulumi
Cloud API version
<!-- @Pulumi employees: If yes, you must submit corresponding changes in
the service repo. -->
2024-01-17 16:50:41 +00:00
|
|
|
//nolint:exhaustive // We only support default values for a subset of types.
|
2020-01-21 22:45:48 +00:00
|
|
|
switch v.Kind() {
|
|
|
|
case reflect.Bool:
|
|
|
|
if v.Bool() {
|
|
|
|
return "true", nil
|
|
|
|
}
|
|
|
|
return "false", nil
|
|
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32:
|
|
|
|
return strconv.FormatInt(v.Int(), 10), nil
|
|
|
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32:
|
|
|
|
return strconv.FormatUint(v.Uint(), 10), nil
|
|
|
|
case reflect.Float32, reflect.Float64:
|
|
|
|
return strconv.FormatFloat(v.Float(), 'f', -1, 64), nil
|
|
|
|
case reflect.String:
|
|
|
|
return fmt.Sprintf("%q", v.String()), nil
|
|
|
|
default:
|
2021-11-13 02:37:17 +00:00
|
|
|
return "", fmt.Errorf("unsupported default value of type %T", value)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) getDefaultValue(dv *schema.DefaultValue, t schema.Type) (string, error) {
|
2021-06-24 16:17:55 +00:00
|
|
|
t = codegen.UnwrapType(t)
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
var val string
|
|
|
|
if dv.Value != nil {
|
2020-12-30 11:46:01 +00:00
|
|
|
switch enum := t.(type) {
|
|
|
|
case *schema.EnumType:
|
|
|
|
enumName := tokenToName(enum.Token)
|
|
|
|
for _, e := range enum.Elements {
|
|
|
|
if e.Value != dv.Value {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
elName := e.Name
|
|
|
|
if elName == "" {
|
|
|
|
elName = fmt.Sprintf("%v", e.Value)
|
|
|
|
}
|
|
|
|
safeName, err := makeSafeEnumName(elName, enumName)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
val = fmt.Sprintf("%s.%s.%s", mod.namespaceName, enumName, safeName)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if val == "" {
|
2021-11-13 02:37:17 +00:00
|
|
|
return "", fmt.Errorf("default value '%v' not found in enum '%s'", dv.Value, enumName)
|
2020-12-30 11:46:01 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
v, err := primitiveValue(dv.Value)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
val = v
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(dv.Environment) != 0 {
|
|
|
|
getType := ""
|
|
|
|
switch t {
|
|
|
|
case schema.BoolType:
|
|
|
|
getType = "Boolean"
|
|
|
|
case schema.IntType:
|
|
|
|
getType = "Int32"
|
|
|
|
case schema.NumberType:
|
|
|
|
getType = "Double"
|
|
|
|
}
|
|
|
|
|
|
|
|
envVars := fmt.Sprintf("%q", dv.Environment[0])
|
|
|
|
for _, e := range dv.Environment[1:] {
|
|
|
|
envVars += fmt.Sprintf(", %q", e)
|
|
|
|
}
|
|
|
|
|
|
|
|
getEnv := fmt.Sprintf("Utilities.GetEnv%s(%s)", getType, envVars)
|
|
|
|
if val != "" {
|
|
|
|
val = fmt.Sprintf("%s ?? %s", getEnv, val)
|
|
|
|
} else {
|
|
|
|
val = getEnv
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return val, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) genResource(w io.Writer, r *schema.Resource) error {
|
|
|
|
// Create a resource module file into which all of this resource's types will go.
|
|
|
|
name := resourceName(r)
|
|
|
|
|
|
|
|
// Open the namespace.
|
|
|
|
fmt.Fprintf(w, "namespace %s\n", mod.namespaceName)
|
|
|
|
fmt.Fprintf(w, "{\n")
|
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
// Write the documentation comment for the resource class
|
2020-06-18 19:32:15 +00:00
|
|
|
printComment(w, codegen.FilterExamples(r.Comment, "csharp"), " ")
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
// Open the class.
|
|
|
|
className := name
|
2020-10-13 18:33:22 +00:00
|
|
|
var baseType string
|
|
|
|
optionsType := "CustomResourceOptions"
|
|
|
|
switch {
|
|
|
|
case r.IsProvider:
|
2022-07-27 09:24:21 +00:00
|
|
|
baseType = "global::Pulumi.ProviderResource"
|
2024-03-04 19:37:54 +00:00
|
|
|
case mod.isK8sCompatMode() && !r.IsComponent:
|
2020-10-13 18:33:22 +00:00
|
|
|
baseType = "KubernetesResource"
|
|
|
|
case r.IsComponent:
|
2022-07-27 09:24:21 +00:00
|
|
|
baseType = "global::Pulumi.ComponentResource"
|
2020-10-13 18:33:22 +00:00
|
|
|
optionsType = "ComponentResourceOptions"
|
|
|
|
default:
|
2022-07-27 09:24:21 +00:00
|
|
|
baseType = "global::Pulumi.CustomResource"
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2020-10-13 18:33:22 +00:00
|
|
|
|
2020-04-24 00:18:29 +00:00
|
|
|
if r.DeprecationMessage != "" {
|
2023-02-23 20:51:11 +00:00
|
|
|
fmt.Fprintf(w, " [Obsolete(@\"%s\")]\n", strings.ReplaceAll(r.DeprecationMessage, `"`, `""`))
|
2020-04-24 00:18:29 +00:00
|
|
|
}
|
2022-12-08 14:23:07 +00:00
|
|
|
fmt.Fprintf(w, " [%sResourceType(\"%s\")]\n", namespaceName(mod.namespaces, mod.pkg.Name()), r.Token)
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, " public partial class %s : %s\n", className, baseType)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
|
2020-07-02 19:30:10 +00:00
|
|
|
var secretProps []string
|
2020-01-21 22:45:48 +00:00
|
|
|
// Emit all output properties.
|
|
|
|
for _, prop := range r.Properties {
|
|
|
|
// Write the property attribute
|
|
|
|
wireName := prop.Name
|
|
|
|
propertyName := mod.propertyName(prop)
|
2021-06-24 16:17:55 +00:00
|
|
|
|
|
|
|
typ := prop.Type
|
|
|
|
if !prop.IsRequired() && mod.isK8sCompatMode() {
|
|
|
|
typ = codegen.RequiredType(prop)
|
|
|
|
}
|
|
|
|
|
|
|
|
propertyType := mod.typeString(typ, "Outputs", false, false, false)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
// Workaround the fact that provider inputs come back as strings.
|
|
|
|
if r.IsProvider && !schema.IsPrimitiveType(prop.Type) {
|
|
|
|
propertyType = "string"
|
2021-06-24 16:17:55 +00:00
|
|
|
if !prop.IsRequired() {
|
2020-01-21 22:45:48 +00:00
|
|
|
propertyType += "?"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-02 19:30:10 +00:00
|
|
|
if prop.Secret {
|
|
|
|
secretProps = append(secretProps, prop.Name)
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
printComment(w, prop.Comment, " ")
|
|
|
|
fmt.Fprintf(w, " [Output(\"%s\")]\n", wireName)
|
|
|
|
fmt.Fprintf(w, " public Output<%s> %s { get; private set; } = null!;\n", propertyType, propertyName)
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
}
|
|
|
|
if len(r.Properties) > 0 {
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Emit the class constructor.
|
2020-05-19 09:41:06 +00:00
|
|
|
argsClassName := className + "Args"
|
|
|
|
if mod.isK8sCompatMode() && !r.IsProvider {
|
|
|
|
argsClassName = fmt.Sprintf("%s.%sArgs", mod.tokenToNamespace(r.Token, "Inputs"), className)
|
|
|
|
}
|
|
|
|
argsType := argsClassName
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
var argsDefault string
|
|
|
|
allOptionalInputs := true
|
2020-05-01 10:45:47 +00:00
|
|
|
hasConstInputs := false
|
2020-01-21 22:45:48 +00:00
|
|
|
for _, prop := range r.InputProperties {
|
2021-06-24 16:17:55 +00:00
|
|
|
allOptionalInputs = allOptionalInputs && !prop.IsRequired()
|
2020-05-01 10:45:47 +00:00
|
|
|
hasConstInputs = hasConstInputs || prop.ConstValue != nil
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2020-05-19 09:41:06 +00:00
|
|
|
if allOptionalInputs || mod.isK8sCompatMode() {
|
2020-01-21 22:45:48 +00:00
|
|
|
// If the number of required input properties was zero, we can make the args object optional.
|
|
|
|
argsDefault = " = null"
|
|
|
|
argsType += "?"
|
|
|
|
}
|
|
|
|
|
|
|
|
tok := r.Token
|
|
|
|
if r.IsProvider {
|
2022-12-08 14:23:07 +00:00
|
|
|
tok = mod.pkg.Name()
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
[dotnet] codegen fix for resources without constant input properties (#15488)
<!---
Thanks so much for your contribution! If this is your first time
contributing, please ensure that you have read the
[CONTRIBUTING](https://github.com/pulumi/pulumi/blob/master/CONTRIBUTING.md)
documentation.
-->
# Description
<!--- Please include a summary of the change and which issue is fixed.
Please also include relevant motivation and context. -->
This PR fixes a bug in the dotnet codegen where the wrong "args" class
name would be used, in the case that the resource has no constant
inputs. This is an edge case because most resources do have at least one
such input.
For example, a new resource definition in p/k produced this output:
```csharp
namespace Pulumi.Kubernetes.Yaml.V2
{
public ConfigGroup(string name, Pulumi.Kubernetes.Types.Inputs.Yaml.V2.ConfigGroupArgs? args = null, CustomResourceOptions? options = null)
: base("kubernetes:yaml/v2:ConfigGroup", name, args ?? new ConfigGroupArgs(), MakeResourceOptions(options, ""), remote: true)
{
}
}
```
Which doesn't compile because `ConfigGroupArgs` is in a separate
namespace.
Should be:
```csharp
public ConfigGroup(string name, Pulumi.Kubernetes.Types.Inputs.Yaml.V2.ConfigGroupArgs? args = null, CustomResourceOptions? options = null)
: base("kubernetes:yaml/v2:ConfigGroup", name, args ?? new Pulumi.Kubernetes.Types.Inputs.Yaml.V2.ConfigGroupArgs(), MakeResourceOptions(options, ""), remote: true)
{
}
```
[Here's
](https://github.com/pulumi/pulumi/pull/15488/files#diff-18b12fabab20d68398aced2890b1ca3073cc32081bb62a022b77a5090c209e3bR45)where
the fix manifests itself in the new test case.
## Testing
A new SDK test case was added to cover the whole `kubernetes20`
compatibility mode, based on a simplified schema from the
pulumi-kubernetes provider.
The schema contains a representative set of resources:
1. `kubernetes:core/v1:ConfigMap` - a non-overlay resource representing
a Kubernetes kind.
2. `kubernetes:core/v1:ConfigMapList` - a Kubernetes list kind
3. `kubernetes:helm.sh/v3:Release` - a non-overlay, non-Kubernetes
resource
4. `kubernetes:yaml:ConfigGroup` - an overlay component resource
An important detail is whether a resource has any input properties that
have a constant value, such as we see with `kind` and `apiVersion`. The
`Release` resource intentionally has no such constant inputs.
## Checklist
- [x] I have run `make tidy` to update any new dependencies
- [x] I have run `make lint` to verify my code passes the lint check
- [x] I have formatted my code using `gofumpt`
<!--- Please provide details if the checkbox below is to be left
unchecked. -->
- [x] I have added tests that prove my fix is effective or that my
feature works
<!---
User-facing changes require a CHANGELOG entry.
-->
- [x] I have run `make changelog` and committed the
`changelog/pending/<file>` documenting my change
<!--
If the change(s) in this PR is a modification of an existing call to the
Pulumi Cloud,
then the service should honor older versions of the CLI where this
change would not exist.
You must then bump the API version in
/pkg/backend/httpstate/client/api.go, as well as add
it to the service.
-->
- [ ] Yes, there are changes in this PR that warrants bumping the Pulumi
Cloud API version
<!-- @Pulumi employees: If yes, you must submit corresponding changes in
the service repo. -->
2024-03-02 06:01:31 +00:00
|
|
|
argsOverride := fmt.Sprintf("args ?? new %s()", argsClassName)
|
2020-05-01 10:45:47 +00:00
|
|
|
if hasConstInputs {
|
|
|
|
argsOverride = "MakeArgs(args)"
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Write a comment prior to the constructor.
|
|
|
|
fmt.Fprintf(w, " /// <summary>\n")
|
|
|
|
fmt.Fprintf(w, " /// Create a %s resource with the given unique name, arguments, and options.\n", className)
|
|
|
|
fmt.Fprintf(w, " /// </summary>\n")
|
|
|
|
fmt.Fprintf(w, " ///\n")
|
|
|
|
fmt.Fprintf(w, " /// <param name=\"name\">The unique name of the resource</param>\n")
|
|
|
|
fmt.Fprintf(w, " /// <param name=\"args\">The arguments used to populate this resource's properties</param>\n")
|
|
|
|
fmt.Fprintf(w, " /// <param name=\"options\">A bag of options that control this resource's behavior</param>\n")
|
|
|
|
|
|
|
|
fmt.Fprintf(w, " public %s(string name, %s args%s, %s? options = null)\n", className, argsType, argsDefault, optionsType)
|
2020-10-13 18:33:22 +00:00
|
|
|
if r.IsComponent {
|
|
|
|
fmt.Fprintf(w, " : base(\"%s\", name, %s, MakeResourceOptions(options, \"\"), remote: true)\n", tok, argsOverride)
|
|
|
|
} else {
|
|
|
|
fmt.Fprintf(w, " : base(\"%s\", name, %s, MakeResourceOptions(options, \"\"))\n", tok, argsOverride)
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
|
2020-10-13 18:33:22 +00:00
|
|
|
// Write a dictionary constructor.
|
|
|
|
if mod.dictionaryConstructors && !r.IsComponent {
|
|
|
|
fmt.Fprintf(w, " internal %s(string name, ImmutableDictionary<string, object?> dictionary, %s? options = null)\n",
|
|
|
|
className, optionsType)
|
|
|
|
if r.IsComponent {
|
|
|
|
fmt.Fprintf(w, " : base(\"%s\", name, new DictionaryResourceArgs(dictionary), MakeResourceOptions(options, \"\"), remote: true)\n", tok)
|
|
|
|
} else {
|
|
|
|
fmt.Fprintf(w, " : base(\"%s\", name, new DictionaryResourceArgs(dictionary), MakeResourceOptions(options, \"\"))\n", tok)
|
|
|
|
}
|
2020-05-19 09:41:06 +00:00
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Write a private constructor for the use of `Get`.
|
2020-10-13 18:33:22 +00:00
|
|
|
if !r.IsProvider && !r.IsComponent {
|
2020-01-21 22:45:48 +00:00
|
|
|
stateParam, stateRef := "", "null"
|
|
|
|
if r.StateInputs != nil {
|
2023-12-12 12:19:42 +00:00
|
|
|
stateParam, stateRef = className+"State? state = null, ", "state"
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
fmt.Fprintf(w, " private %s(string name, Input<string> id, %s%s? options = null)\n", className, stateParam, optionsType)
|
|
|
|
fmt.Fprintf(w, " : base(\"%s\", name, %s, MakeResourceOptions(options, id))\n", tok, stateRef)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
}
|
|
|
|
|
2020-05-01 10:45:47 +00:00
|
|
|
if hasConstInputs {
|
|
|
|
// Write the method that will calculate the resource arguments.
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
fmt.Fprintf(w, " private static %[1]s MakeArgs(%[1]s args)\n", argsType)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
2020-05-19 09:41:06 +00:00
|
|
|
fmt.Fprintf(w, " args ??= new %s();\n", argsClassName)
|
2020-05-01 10:45:47 +00:00
|
|
|
for _, prop := range r.InputProperties {
|
|
|
|
if prop.ConstValue != nil {
|
|
|
|
v, err := primitiveValue(prop.ConstValue)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, " args.%s = %s;\n", mod.propertyName(prop), v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, " return args;\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Write the method that will calculate the resource options.
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
fmt.Fprintf(w, " private static %[1]s MakeResourceOptions(%[1]s? options, Input<string>? id)\n", optionsType)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " var defaultOptions = new %s\n", optionsType)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
2020-07-02 19:30:10 +00:00
|
|
|
fmt.Fprintf(w, " Version = Utilities.Version,\n")
|
2022-12-08 14:23:07 +00:00
|
|
|
def, err := mod.pkg.Definition()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if url := def.PluginDownloadURL; url != "" {
|
2022-01-14 00:10:17 +00:00
|
|
|
fmt.Fprintf(w, " PluginDownloadURL = %q,\n", url)
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-07-02 19:30:10 +00:00
|
|
|
if len(r.Aliases) > 0 {
|
|
|
|
fmt.Fprintf(w, " Aliases =\n")
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
for _, alias := range r.Aliases {
|
|
|
|
fmt.Fprintf(w, " ")
|
2023-11-21 22:40:14 +00:00
|
|
|
if alias.Type != nil {
|
|
|
|
fmt.Fprintf(w, "new global::Pulumi.Alias { Type = \"%v\" },\n", *alias.Type)
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
fmt.Fprintf(w, " },\n")
|
|
|
|
}
|
2020-07-02 19:30:10 +00:00
|
|
|
if len(secretProps) > 0 {
|
|
|
|
fmt.Fprintf(w, " AdditionalSecretOutputs =\n")
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
for _, sp := range secretProps {
|
|
|
|
fmt.Fprintf(w, " ")
|
|
|
|
fmt.Fprintf(w, "%q", sp)
|
|
|
|
fmt.Fprintf(w, ",\n")
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, " },\n")
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2021-09-08 05:23:30 +00:00
|
|
|
replaceOnChangesProps, errList := r.ReplaceOnChanges()
|
|
|
|
for _, err := range errList {
|
|
|
|
cmdutil.Diag().Warningf(&diag.Diag{Message: err.Error()})
|
|
|
|
}
|
|
|
|
if len(replaceOnChangesProps) > 0 {
|
|
|
|
fmt.Fprint(w, " ReplaceOnChanges =\n")
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
for _, n := range schema.PropertyListJoinToString(replaceOnChangesProps,
|
|
|
|
func(s string) string { return s }) {
|
|
|
|
fmt.Fprintf(w, " ")
|
|
|
|
fmt.Fprintf(w, "%q,\n", n)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, " },\n")
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, " };\n")
|
|
|
|
fmt.Fprintf(w, " var merged = %s.Merge(defaultOptions, options);\n", optionsType)
|
|
|
|
fmt.Fprintf(w, " // Override the ID if one was specified for consistency with other language SDKs.\n")
|
|
|
|
fmt.Fprintf(w, " merged.Id = id ?? merged.Id;\n")
|
|
|
|
fmt.Fprintf(w, " return merged;\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
|
2020-10-13 18:33:22 +00:00
|
|
|
// Write the `Get` method for reading instances of this resource unless this is a provider resource or ComponentResource.
|
|
|
|
if !r.IsProvider && !r.IsComponent {
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, " /// <summary>\n")
|
|
|
|
fmt.Fprintf(w, " /// Get an existing %s resource's state with the given name, ID, and optional extra\n", className)
|
|
|
|
fmt.Fprintf(w, " /// properties used to qualify the lookup.\n")
|
|
|
|
fmt.Fprintf(w, " /// </summary>\n")
|
|
|
|
fmt.Fprintf(w, " ///\n")
|
|
|
|
fmt.Fprintf(w, " /// <param name=\"name\">The unique name of the resulting resource.</param>\n")
|
|
|
|
fmt.Fprintf(w, " /// <param name=\"id\">The unique provider ID of the resource to lookup.</param>\n")
|
2020-05-01 10:45:47 +00:00
|
|
|
|
|
|
|
stateParam, stateRef := "", ""
|
|
|
|
if r.StateInputs != nil {
|
2023-12-12 12:19:42 +00:00
|
|
|
stateParam, stateRef = className+"State? state = null, ", "state, "
|
2020-05-01 10:45:47 +00:00
|
|
|
fmt.Fprintf(w, " /// <param name=\"state\">Any extra arguments used during the lookup.</param>\n")
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, " /// <param name=\"options\">A bag of options that control this resource's behavior</param>\n")
|
|
|
|
fmt.Fprintf(w, " public static %s Get(string name, Input<string> id, %s%s? options = null)\n", className, stateParam, optionsType)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " return new %s(name, id, %soptions);\n", className, stateRef)
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
}
|
|
|
|
|
2021-07-27 23:42:17 +00:00
|
|
|
// Generate methods.
|
|
|
|
genMethod := func(method *schema.Method) {
|
|
|
|
methodName := Title(method.Name)
|
|
|
|
fun := method.Function
|
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
var objectReturnType *schema.ObjectType
|
|
|
|
|
|
|
|
if fun.ReturnType != nil {
|
|
|
|
if objectType, ok := fun.ReturnType.(*schema.ObjectType); ok && fun.InlineObjectAsReturnType {
|
|
|
|
objectReturnType = objectType
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
liftReturn := mod.liftSingleValueMethodReturns && objectReturnType != nil && len(objectReturnType.Properties) == 1
|
2021-10-01 18:33:02 +00:00
|
|
|
|
2021-07-27 23:42:17 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
2021-10-01 18:33:02 +00:00
|
|
|
returnType, typeParameter, lift := "void", "", ""
|
2023-01-11 22:17:14 +00:00
|
|
|
if fun.ReturnType != nil {
|
2021-07-27 23:42:17 +00:00
|
|
|
typeParameter = fmt.Sprintf("<%s%sResult>", className, methodName)
|
2023-01-11 22:17:14 +00:00
|
|
|
if liftReturn {
|
2022-09-14 23:19:32 +00:00
|
|
|
returnType = fmt.Sprintf("global::Pulumi.Output<%s>",
|
2023-01-11 22:17:14 +00:00
|
|
|
mod.typeString(objectReturnType.Properties[0].Type, "", false, false, false))
|
2021-10-01 18:33:02 +00:00
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
fieldName := mod.propertyName(objectReturnType.Properties[0])
|
2021-10-01 18:33:02 +00:00
|
|
|
lift = fmt.Sprintf(".Apply(v => v.%s)", fieldName)
|
|
|
|
} else {
|
2023-12-12 12:19:42 +00:00
|
|
|
returnType = "global::Pulumi.Output" + typeParameter
|
2021-10-01 18:33:02 +00:00
|
|
|
}
|
2021-07-27 23:42:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var argsParamDef string
|
|
|
|
argsParamRef := "CallArgs.Empty"
|
|
|
|
if fun.Inputs != nil {
|
|
|
|
var hasArgs bool
|
|
|
|
allOptionalInputs := true
|
|
|
|
for _, arg := range fun.Inputs.InputShape.Properties {
|
|
|
|
if arg.Name == "__self__" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
hasArgs = true
|
|
|
|
allOptionalInputs = allOptionalInputs && !arg.IsRequired()
|
|
|
|
}
|
|
|
|
if hasArgs {
|
|
|
|
var argsDefault, sigil string
|
|
|
|
if allOptionalInputs {
|
|
|
|
// If the number of required input properties was zero, we can make the args object optional.
|
|
|
|
argsDefault, sigil = " = null", "?"
|
|
|
|
}
|
|
|
|
|
|
|
|
argsParamDef = fmt.Sprintf("%s%sArgs%s args%s", className, methodName, sigil, argsDefault)
|
|
|
|
argsParamRef = fmt.Sprintf("args ?? new %s%sArgs()", className, methodName)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Emit the doc comment, if any.
|
|
|
|
printComment(w, fun.Comment, " ")
|
|
|
|
|
|
|
|
if fun.DeprecationMessage != "" {
|
|
|
|
fmt.Fprintf(w, " [Obsolete(@\"%s\")]\n", strings.ReplaceAll(fun.DeprecationMessage, `"`, `""`))
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, " public %s %s(%s)\n", returnType, methodName, argsParamDef)
|
2022-09-14 23:19:32 +00:00
|
|
|
fmt.Fprintf(w, " => global::Pulumi.Deployment.Instance.Call%s(\"%s\", %s, this)%s;\n",
|
2021-10-01 18:33:02 +00:00
|
|
|
typeParameter, fun.Token, argsParamRef, lift)
|
2021-07-27 23:42:17 +00:00
|
|
|
}
|
|
|
|
for _, method := range r.Methods {
|
|
|
|
genMethod(method)
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Close the class.
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
// Arguments are in a different namespace for the Kubernetes SDK.
|
|
|
|
if mod.isK8sCompatMode() && !r.IsProvider {
|
|
|
|
// Close the namespace.
|
|
|
|
fmt.Fprintf(w, "}\n")
|
|
|
|
|
|
|
|
// Open the namespace.
|
|
|
|
fmt.Fprintf(w, "namespace %s\n", mod.tokenToNamespace(r.Token, "Inputs"))
|
|
|
|
fmt.Fprintf(w, "{\n")
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Generate the resource args type.
|
|
|
|
args := &plainType{
|
|
|
|
mod: mod,
|
|
|
|
res: r,
|
|
|
|
name: name + "Args",
|
|
|
|
baseClass: "ResourceArgs",
|
|
|
|
propertyTypeQualifier: "Inputs",
|
|
|
|
properties: r.InputProperties,
|
2021-04-16 02:03:28 +00:00
|
|
|
args: true,
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
if err := args.genInputType(w, 1); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate the `Get` args type, if any.
|
|
|
|
if r.StateInputs != nil {
|
|
|
|
state := &plainType{
|
|
|
|
mod: mod,
|
|
|
|
res: r,
|
|
|
|
name: name + "State",
|
|
|
|
baseClass: "ResourceArgs",
|
|
|
|
propertyTypeQualifier: "Inputs",
|
|
|
|
properties: r.StateInputs.Properties,
|
2021-04-16 02:03:28 +00:00
|
|
|
args: true,
|
2020-01-21 22:45:48 +00:00
|
|
|
state: true,
|
|
|
|
}
|
|
|
|
if err := state.genInputType(w, 1); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-27 23:42:17 +00:00
|
|
|
// Generate method types.
|
|
|
|
genMethodTypes := func(method *schema.Method) error {
|
|
|
|
methodName := Title(method.Name)
|
|
|
|
fun := method.Function
|
|
|
|
|
|
|
|
// Generate args type.
|
|
|
|
var args []*schema.Property
|
|
|
|
if fun.Inputs != nil {
|
|
|
|
// Filter out the __self__ argument from the inputs.
|
2023-06-28 16:02:04 +00:00
|
|
|
args = slice.Prealloc[*schema.Property](len(fun.Inputs.InputShape.Properties) - 1)
|
2021-07-27 23:42:17 +00:00
|
|
|
for _, arg := range fun.Inputs.InputShape.Properties {
|
|
|
|
if arg.Name == "__self__" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
args = append(args, arg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(args) > 0 {
|
|
|
|
comment, escape := fun.Inputs.Comment, true
|
|
|
|
if comment == "" {
|
|
|
|
comment, escape = fmt.Sprintf(
|
|
|
|
"The set of arguments for the <see cref=\"%s.%s\"/> method.", className, methodName), false
|
|
|
|
}
|
|
|
|
argsType := &plainType{
|
|
|
|
mod: mod,
|
|
|
|
comment: comment,
|
|
|
|
unescapeComment: !escape,
|
|
|
|
name: fmt.Sprintf("%s%sArgs", className, methodName),
|
|
|
|
baseClass: "CallArgs",
|
|
|
|
propertyTypeQualifier: "Inputs",
|
|
|
|
properties: args,
|
|
|
|
args: true,
|
|
|
|
}
|
|
|
|
if err := argsType.genInputType(w, 1); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
var objectReturnType *schema.ObjectType
|
|
|
|
if fun.ReturnType != nil {
|
|
|
|
if objectType, ok := fun.ReturnType.(*schema.ObjectType); ok && objectType != nil {
|
|
|
|
objectReturnType = objectType
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-27 23:42:17 +00:00
|
|
|
// Generate result type.
|
2023-01-11 22:17:14 +00:00
|
|
|
if objectReturnType != nil {
|
|
|
|
shouldLiftReturn := mod.liftSingleValueMethodReturns && len(objectReturnType.Properties) == 1
|
2021-10-01 18:33:02 +00:00
|
|
|
|
2021-07-27 23:42:17 +00:00
|
|
|
comment, escape := fun.Inputs.Comment, true
|
|
|
|
if comment == "" {
|
|
|
|
comment, escape = fmt.Sprintf(
|
|
|
|
"The results of the <see cref=\"%s.%s\"/> method.", className, methodName), false
|
|
|
|
}
|
|
|
|
resultType := &plainType{
|
|
|
|
mod: mod,
|
|
|
|
comment: comment,
|
|
|
|
unescapeComment: !escape,
|
|
|
|
name: fmt.Sprintf("%s%sResult", className, methodName),
|
|
|
|
propertyTypeQualifier: "Outputs",
|
2023-01-11 22:17:14 +00:00
|
|
|
properties: objectReturnType.Properties,
|
2021-10-01 18:33:02 +00:00
|
|
|
internal: shouldLiftReturn,
|
2021-07-27 23:42:17 +00:00
|
|
|
}
|
|
|
|
resultType.genOutputType(w, 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
for _, method := range r.Methods {
|
|
|
|
if err := genMethodTypes(method); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Close the namespace.
|
|
|
|
fmt.Fprintf(w, "}\n")
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
func (mod *modContext) genFunctionFileCode(f *schema.Function) (string, error) {
|
|
|
|
buffer := &bytes.Buffer{}
|
2022-01-21 20:58:11 +00:00
|
|
|
importStrings := mod.pulumiImports()
|
2021-12-02 01:53:18 +00:00
|
|
|
|
|
|
|
// True if the function has a non-standard namespace.
|
|
|
|
nonStandardNamespace := mod.namespaceName != mod.tokenToNamespace(f.Token, "")
|
|
|
|
// If so, we need to import our project defined types.
|
|
|
|
if nonStandardNamespace {
|
|
|
|
importStrings = append(importStrings, mod.namespaceName)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need to qualify input types when we are not in the same module as them.
|
|
|
|
if nonStandardNamespace {
|
|
|
|
defer func(current bool) { mod.fullyQualifiedInputs = current }(mod.fullyQualifiedInputs)
|
|
|
|
mod.fullyQualifiedInputs = true
|
|
|
|
}
|
2021-10-18 22:18:15 +00:00
|
|
|
mod.genHeader(buffer, importStrings)
|
|
|
|
if err := mod.genFunction(buffer, f); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return buffer.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func allOptionalInputs(fun *schema.Function) bool {
|
|
|
|
if fun.Inputs != nil {
|
|
|
|
for _, prop := range fun.Inputs.Properties {
|
|
|
|
if prop.IsRequired() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
func typeParamOrEmpty(typeParamName string) string {
|
|
|
|
if typeParamName != "" {
|
|
|
|
return fmt.Sprintf("<%s>", typeParamName)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) functionReturnType(fun *schema.Function) string {
|
|
|
|
className := tokenToFunctionName(fun.Token)
|
|
|
|
if fun.ReturnType != nil {
|
|
|
|
if _, ok := fun.ReturnType.(*schema.ObjectType); ok && fun.InlineObjectAsReturnType {
|
|
|
|
// for object return types, assume a Result type is generated in the same class as it's function
|
|
|
|
// and reference it from here directly
|
2023-12-12 12:19:42 +00:00
|
|
|
return className + "Result"
|
2023-01-11 22:17:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// otherwise, the object type is a reference to an output type
|
|
|
|
return mod.typeString(fun.ReturnType, "Outputs", false, false, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// runtimeInvokeFunction returns the name of the Invoke function to use at runtime
|
|
|
|
// from the SDK for the given provider function. This is necessary because some
|
|
|
|
// functions have simple return types such as number, string, array<string> etc.
|
|
|
|
// and the SDK's Invoke function cannot handle these types since the engine expects
|
|
|
|
// the result of invokes to be a dictionary.
|
|
|
|
//
|
|
|
|
// We use Invoke for functions with object return types and InvokeSingle for everything else.
|
|
|
|
func runtimeInvokeFunction(fun *schema.Function) string {
|
|
|
|
switch fun.ReturnType.(type) {
|
|
|
|
// If the function has no return type, it is a void function.
|
|
|
|
case nil:
|
|
|
|
return "Invoke"
|
|
|
|
// If the function has an object return type, it is a normal invoke function.
|
|
|
|
case *schema.ObjectType:
|
|
|
|
return "Invoke"
|
|
|
|
// If the function has an object return type, it is also a normal invoke function.
|
|
|
|
// because the deserialization can handle it
|
|
|
|
case *schema.MapType:
|
|
|
|
return "Invoke"
|
|
|
|
default:
|
|
|
|
// Anything else needs to be handled by InvokeSingle
|
|
|
|
// which expects an object with a single property to be returned
|
|
|
|
// then unwraps the value from that property
|
|
|
|
return "InvokeSingle"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
func (mod *modContext) genFunction(w io.Writer, fun *schema.Function) error {
|
2020-04-28 00:47:01 +00:00
|
|
|
className := tokenToFunctionName(fun.Token)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
fmt.Fprintf(w, "namespace %s\n", mod.tokenToNamespace(fun.Token, ""))
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "{\n")
|
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
typeParameter := mod.functionReturnType(fun)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
var argsParamDef string
|
|
|
|
argsParamRef := "InvokeArgs.Empty"
|
|
|
|
if fun.Inputs != nil {
|
|
|
|
var argsDefault, sigil string
|
2021-10-18 22:18:15 +00:00
|
|
|
if allOptionalInputs(fun) {
|
2020-01-21 22:45:48 +00:00
|
|
|
// If the number of required input properties was zero, we can make the args object optional.
|
|
|
|
argsDefault, sigil = " = null", "?"
|
|
|
|
}
|
|
|
|
|
2020-04-14 08:30:25 +00:00
|
|
|
argsParamDef = fmt.Sprintf("%sArgs%s args%s, ", className, sigil, argsDefault)
|
|
|
|
argsParamRef = fmt.Sprintf("args ?? new %sArgs()", className)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2020-04-24 00:18:29 +00:00
|
|
|
if fun.DeprecationMessage != "" {
|
2023-02-23 20:51:11 +00:00
|
|
|
fmt.Fprintf(w, " [Obsolete(@\"%s\")]\n", strings.ReplaceAll(fun.DeprecationMessage, `"`, `""`))
|
2020-04-24 00:18:29 +00:00
|
|
|
}
|
2023-01-11 22:17:14 +00:00
|
|
|
|
|
|
|
// Open the class we'll use for data sources.
|
2020-04-14 08:30:25 +00:00
|
|
|
fmt.Fprintf(w, " public static class %s\n", className)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Emit the doc comment, if any.
|
2020-04-14 08:30:25 +00:00
|
|
|
printComment(w, fun.Comment, " ")
|
2023-01-11 22:17:14 +00:00
|
|
|
invokeCall := runtimeInvokeFunction(fun)
|
|
|
|
if !fun.MultiArgumentInputs {
|
|
|
|
// Emit the datasource method.
|
|
|
|
// this is default behavior for all functions.
|
|
|
|
fmt.Fprintf(w, " public static Task%s InvokeAsync(%sInvokeOptions? options = null)\n",
|
|
|
|
typeParamOrEmpty(typeParameter), argsParamDef)
|
|
|
|
// new line and indent
|
|
|
|
fmt.Fprint(w, " ")
|
|
|
|
fmt.Fprintf(w, "=> global::Pulumi.Deployment.Instance.%sAsync%s", invokeCall, typeParamOrEmpty(typeParameter))
|
|
|
|
fmt.Fprintf(w, "(\"%s\", %s, options.WithDefaults());\n", fun.Token, argsParamRef)
|
|
|
|
} else {
|
|
|
|
// multi-argument inputs and output property bag
|
|
|
|
// first generate the function definition
|
|
|
|
fmt.Fprintf(w, " public static async Task%s InvokeAsync(", typeParamOrEmpty(typeParameter))
|
|
|
|
for _, prop := range fun.Inputs.Properties {
|
|
|
|
argumentName := LowerCamelCase(prop.Name)
|
|
|
|
argumentType := mod.typeString(prop.Type, "", false, false, true)
|
|
|
|
paramDeclaration := fmt.Sprintf("%s %s", argumentType, argumentName)
|
|
|
|
if !prop.IsRequired() {
|
|
|
|
paramDeclaration += " = null"
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "%s", paramDeclaration)
|
|
|
|
fmt.Fprint(w, ", ")
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
fmt.Fprint(w, "InvokeOptions? invokeOptions = null)\n")
|
|
|
|
|
|
|
|
funcBodyIndent := func() {
|
|
|
|
fmt.Fprintf(w, " ")
|
|
|
|
}
|
|
|
|
|
|
|
|
// now the function body
|
|
|
|
fmt.Fprint(w, " {\n")
|
|
|
|
// generate a dictionary where each entry is a key-value pair made out of the inputs of the function
|
|
|
|
funcBodyIndent()
|
|
|
|
fmt.Fprint(w, "var builder = ImmutableDictionary.CreateBuilder<string, object?>();\n")
|
|
|
|
for _, prop := range fun.Inputs.Properties {
|
|
|
|
argumentName := LowerCamelCase(prop.Name)
|
|
|
|
funcBodyIndent()
|
|
|
|
fmt.Fprintf(w, "builder[\"%s\"] = %s;\n", prop.Name, argumentName)
|
|
|
|
}
|
|
|
|
|
|
|
|
funcBodyIndent()
|
|
|
|
fmt.Fprint(w, "var args = new global::Pulumi.DictionaryInvokeArgs(builder.ToImmutableDictionary());\n")
|
|
|
|
funcBodyIndent()
|
|
|
|
// full invoke call
|
|
|
|
fmt.Fprint(w, "return await global::Pulumi.Deployment.Instance.")
|
|
|
|
fmt.Fprintf(w, "%sAsync%s", invokeCall, typeParamOrEmpty(typeParameter))
|
|
|
|
fmt.Fprintf(w, "(\"%s\", args, invokeOptions.WithDefaults());\n", fun.Token)
|
|
|
|
fmt.Fprint(w, " }\n")
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
// Emit the Output method if needed.
|
|
|
|
err := mod.genFunctionOutputVersion(w, fun)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Close the class.
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
|
|
|
|
// Emit the args and result types, if any.
|
2023-01-11 22:17:14 +00:00
|
|
|
if fun.Inputs != nil && !fun.MultiArgumentInputs {
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
args := &plainType{
|
|
|
|
mod: mod,
|
2020-04-14 08:30:25 +00:00
|
|
|
name: className + "Args",
|
2020-01-21 22:45:48 +00:00
|
|
|
baseClass: "InvokeArgs",
|
|
|
|
propertyTypeQualifier: "Inputs",
|
|
|
|
properties: fun.Inputs.Properties,
|
|
|
|
}
|
|
|
|
if err := args.genInputType(w, 1); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2021-10-18 22:18:15 +00:00
|
|
|
|
|
|
|
err = mod.genFunctionOutputVersionTypes(w, fun)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
if fun.ReturnType != nil {
|
|
|
|
if objectType, ok := fun.ReturnType.(*schema.ObjectType); ok && fun.InlineObjectAsReturnType {
|
2022-12-16 12:39:03 +00:00
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
res := &plainType{
|
|
|
|
mod: mod,
|
|
|
|
name: className + "Result",
|
|
|
|
propertyTypeQualifier: "Outputs",
|
|
|
|
properties: objectType.Properties,
|
|
|
|
}
|
|
|
|
res.genOutputType(w, 1)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close the namespace.
|
|
|
|
fmt.Fprintf(w, "}\n")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
func functionOutputVersionArgsTypeName(fun *schema.Function) string {
|
|
|
|
className := tokenToFunctionName(fun.Token)
|
2023-12-12 12:19:42 +00:00
|
|
|
return className + "InvokeArgs"
|
2021-10-18 22:18:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Generates `${fn}Output(..)` version lifted to work on
|
|
|
|
// `Input`-wrapped arguments and producing an `Output`-wrapped result.
|
|
|
|
func (mod *modContext) genFunctionOutputVersion(w io.Writer, fun *schema.Function) error {
|
2023-08-08 12:00:30 +00:00
|
|
|
if fun.ReturnType == nil {
|
|
|
|
// no need to generate an output version if the function doesn't return anything
|
2021-10-18 22:18:15 +00:00
|
|
|
return nil
|
|
|
|
}
|
2023-08-08 12:00:30 +00:00
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
var argsDefault, sigil string
|
|
|
|
if allOptionalInputs(fun) {
|
|
|
|
// If the number of required input properties was zero, we can make the args object optional.
|
|
|
|
argsDefault, sigil = " = null", "?"
|
|
|
|
}
|
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
typeParameter := mod.functionReturnType(fun)
|
|
|
|
invokeCall := runtimeInvokeFunction(fun)
|
2021-10-18 22:18:15 +00:00
|
|
|
argsTypeName := functionOutputVersionArgsTypeName(fun)
|
|
|
|
outputArgsParamDef := fmt.Sprintf("%s%s args%s, ", argsTypeName, sigil, argsDefault)
|
|
|
|
outputArgsParamRef := fmt.Sprintf("args ?? new %s()", argsTypeName)
|
|
|
|
|
2023-08-08 12:00:30 +00:00
|
|
|
if fun.Inputs == nil || len(fun.Inputs.Properties) == 0 {
|
|
|
|
outputArgsParamDef = ""
|
|
|
|
outputArgsParamRef = "InvokeArgs.Empty"
|
|
|
|
}
|
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
// Emit the doc comment, if any.
|
|
|
|
printComment(w, fun.Comment, " ")
|
2023-01-11 22:17:14 +00:00
|
|
|
|
|
|
|
if !fun.MultiArgumentInputs {
|
|
|
|
fmt.Fprintf(w, " public static Output%s Invoke(%sInvokeOptions? options = null)\n",
|
|
|
|
typeParamOrEmpty(typeParameter), outputArgsParamDef)
|
|
|
|
fmt.Fprintf(w, " => global::Pulumi.Deployment.Instance.%s%s(\"%s\", %s, options.WithDefaults());\n",
|
|
|
|
invokeCall, typeParamOrEmpty(typeParameter), fun.Token, outputArgsParamRef)
|
|
|
|
} else {
|
|
|
|
fmt.Fprintf(w, " public static Output%s Invoke(", typeParamOrEmpty(typeParameter))
|
|
|
|
for _, prop := range fun.Inputs.Properties {
|
|
|
|
var paramDeclaration string
|
|
|
|
argumentName := LowerCamelCase(prop.Name)
|
|
|
|
propertyType := &schema.InputType{ElementType: prop.Type}
|
|
|
|
argumentType := mod.typeString(propertyType, "", true /* input */, false, true)
|
|
|
|
if prop.IsRequired() {
|
|
|
|
paramDeclaration = fmt.Sprintf("%s %s", argumentType, argumentName)
|
|
|
|
} else {
|
|
|
|
paramDeclaration = fmt.Sprintf("%s? %s = null", argumentType, argumentName)
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, "%s", paramDeclaration)
|
|
|
|
fmt.Fprint(w, ", ")
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprint(w, "InvokeOptions? invokeOptions = null)\n")
|
|
|
|
|
|
|
|
// now the function body
|
|
|
|
fmt.Fprint(w, " {\n")
|
|
|
|
fmt.Fprint(w, " var builder = ImmutableDictionary.CreateBuilder<string, object?>();\n")
|
2023-08-08 12:00:30 +00:00
|
|
|
if fun.Inputs != nil {
|
|
|
|
for _, prop := range fun.Inputs.Properties {
|
|
|
|
argumentName := LowerCamelCase(prop.Name)
|
|
|
|
fmt.Fprintf(w, " builder[\"%s\"] = %s;\n", prop.Name, argumentName)
|
|
|
|
}
|
2023-01-11 22:17:14 +00:00
|
|
|
}
|
|
|
|
fmt.Fprint(w, " var args = new global::Pulumi.DictionaryInvokeArgs(builder.ToImmutableDictionary());\n")
|
|
|
|
fmt.Fprintf(w, " return global::Pulumi.Deployment.Instance.%s%s(\"%s\", args, invokeOptions.WithDefaults());\n",
|
|
|
|
invokeCall, typeParamOrEmpty(typeParameter), fun.Token)
|
|
|
|
fmt.Fprint(w, " }\n")
|
|
|
|
}
|
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate helper type definitions referred to in `genFunctionOutputVersion`.
|
|
|
|
func (mod *modContext) genFunctionOutputVersionTypes(w io.Writer, fun *schema.Function) error {
|
2023-08-08 12:00:30 +00:00
|
|
|
if fun.Inputs == nil || fun.ReturnType == nil || len(fun.Inputs.Properties) == 0 {
|
2021-10-18 22:18:15 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
if !fun.MultiArgumentInputs {
|
|
|
|
applyArgs := &plainType{
|
|
|
|
mod: mod,
|
|
|
|
name: functionOutputVersionArgsTypeName(fun),
|
|
|
|
propertyTypeQualifier: "Inputs",
|
|
|
|
baseClass: "InvokeArgs",
|
|
|
|
properties: fun.Inputs.InputShape.Properties,
|
|
|
|
args: true,
|
|
|
|
}
|
2022-12-16 12:39:03 +00:00
|
|
|
|
2023-01-11 22:17:14 +00:00
|
|
|
if err := applyArgs.genInputTypeWithFlags(w, 1, true /* generateInputAttributes */); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-12-21 01:47:29 +00:00
|
|
|
}
|
2023-01-11 22:17:14 +00:00
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
func (mod *modContext) genEnums(w io.Writer, enums []*schema.EnumType) error {
|
|
|
|
// Open the namespace.
|
|
|
|
fmt.Fprintf(w, "namespace %s\n", mod.namespaceName)
|
|
|
|
fmt.Fprintf(w, "{\n")
|
|
|
|
|
|
|
|
for i, enum := range enums {
|
|
|
|
err := mod.genEnum(w, enum)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if i != len(enums)-1 {
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close the namespace.
|
|
|
|
fmt.Fprintf(w, "}\n")
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func printObsoleteAttribute(w io.Writer, deprecationMessage, indent string) {
|
|
|
|
if deprecationMessage != "" {
|
2023-02-23 20:51:11 +00:00
|
|
|
fmt.Fprintf(w, "%s[Obsolete(@\"%s\")]\n", indent, strings.ReplaceAll(deprecationMessage, `"`, `""`))
|
2020-11-06 17:01:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) genEnum(w io.Writer, enum *schema.EnumType) error {
|
|
|
|
indent := " "
|
|
|
|
enumName := tokenToName(enum.Token)
|
|
|
|
|
2020-11-09 19:33:22 +00:00
|
|
|
// Fix up identifiers for each enum value.
|
|
|
|
for _, e := range enum.Elements {
|
|
|
|
// If the enum doesn't have a name, set the value as the name.
|
|
|
|
if e.Name == "" {
|
|
|
|
e.Name = fmt.Sprintf("%v", e.Value)
|
|
|
|
}
|
|
|
|
|
2020-12-16 17:22:44 +00:00
|
|
|
safeName, err := makeSafeEnumName(e.Name, enumName)
|
2020-11-09 19:33:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
e.Name = safeName
|
|
|
|
}
|
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
// Print documentation comment
|
|
|
|
printComment(w, enum.Comment, indent)
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
underlyingType := mod.typeString(enum.ElementType, "", false, false, false)
|
2020-11-06 17:01:03 +00:00
|
|
|
switch enum.ElementType {
|
|
|
|
case schema.StringType, schema.NumberType:
|
|
|
|
// EnumType attribute
|
|
|
|
fmt.Fprintf(w, "%s[EnumType]\n", indent)
|
|
|
|
|
|
|
|
// Open struct declaration
|
|
|
|
fmt.Fprintf(w, "%[1]spublic readonly struct %[2]s : IEquatable<%[2]s>\n", indent, enumName)
|
|
|
|
fmt.Fprintf(w, "%s{\n", indent)
|
|
|
|
indent := strings.Repeat(indent, 2)
|
|
|
|
fmt.Fprintf(w, "%sprivate readonly %s _value;\n", indent, underlyingType)
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Constructor
|
|
|
|
fmt.Fprintf(w, "%sprivate %s(%s value)\n", indent, enumName, underlyingType)
|
|
|
|
fmt.Fprintf(w, "%s{\n", indent)
|
|
|
|
fmt.Fprintf(w, "%s _value = value", indent)
|
|
|
|
if enum.ElementType == schema.StringType {
|
|
|
|
fmt.Fprintf(w, " ?? throw new ArgumentNullException(nameof(value))")
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, ";\n")
|
|
|
|
fmt.Fprintf(w, "%s}\n", indent)
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Enum values
|
|
|
|
for _, e := range enum.Elements {
|
|
|
|
printComment(w, e.Comment, indent)
|
|
|
|
printObsoleteAttribute(w, e.DeprecationMessage, indent)
|
|
|
|
fmt.Fprintf(w, "%[1]spublic static %[2]s %[3]s { get; } = new %[2]s(", indent, enumName, e.Name)
|
|
|
|
if enum.ElementType == schema.StringType {
|
|
|
|
fmt.Fprintf(w, "%q", e.Value)
|
|
|
|
} else {
|
|
|
|
fmt.Fprintf(w, "%v", e.Value)
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, ");\n")
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Equality and inequality operators
|
|
|
|
fmt.Fprintf(w, "%[1]spublic static bool operator ==(%[2]s left, %[2]s right) => left.Equals(right);\n", indent, enumName)
|
|
|
|
fmt.Fprintf(w, "%[1]spublic static bool operator !=(%[2]s left, %[2]s right) => !left.Equals(right);\n", indent, enumName)
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Explicit conversion operator
|
|
|
|
fmt.Fprintf(w, "%[1]spublic static explicit operator %s(%s value) => value._value;\n", indent, underlyingType, enumName)
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Equals override
|
|
|
|
fmt.Fprintf(w, "%s[EditorBrowsable(EditorBrowsableState.Never)]\n", indent)
|
|
|
|
fmt.Fprintf(w, "%spublic override bool Equals(object? obj) => obj is %s other && Equals(other);\n", indent, enumName)
|
|
|
|
fmt.Fprintf(w, "%spublic bool Equals(%s other) => ", indent, enumName)
|
|
|
|
if enum.ElementType == schema.StringType {
|
|
|
|
fmt.Fprintf(w, "string.Equals(_value, other._value, StringComparison.Ordinal)")
|
|
|
|
} else {
|
|
|
|
fmt.Fprintf(w, "_value == other._value")
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, ";\n")
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// GetHashCode override
|
|
|
|
fmt.Fprintf(w, "%s[EditorBrowsable(EditorBrowsableState.Never)]\n", indent)
|
|
|
|
fmt.Fprintf(w, "%spublic override int GetHashCode() => _value", indent)
|
|
|
|
if enum.ElementType == schema.StringType {
|
|
|
|
fmt.Fprintf(w, "?")
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, ".GetHashCode()")
|
|
|
|
if enum.ElementType == schema.StringType {
|
|
|
|
fmt.Fprintf(w, " ?? 0")
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, ";\n")
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// ToString override
|
|
|
|
fmt.Fprintf(w, "%spublic override string ToString() => _value", indent)
|
|
|
|
if enum.ElementType == schema.NumberType {
|
|
|
|
fmt.Fprintf(w, ".ToString()")
|
|
|
|
}
|
|
|
|
fmt.Fprintf(w, ";\n")
|
|
|
|
case schema.IntType:
|
|
|
|
// Open enum declaration
|
|
|
|
fmt.Fprintf(w, "%spublic enum %s\n", indent, enumName)
|
|
|
|
fmt.Fprintf(w, "%s{\n", indent)
|
|
|
|
for _, e := range enum.Elements {
|
|
|
|
indent := strings.Repeat(indent, 2)
|
|
|
|
printComment(w, e.Comment, indent)
|
|
|
|
printObsoleteAttribute(w, e.DeprecationMessage, indent)
|
|
|
|
fmt.Fprintf(w, "%s%s = %v,\n", indent, e.Name, e.Value)
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
// Issue to implement boolean-based enums: https://github.com/pulumi/pulumi/issues/5652
|
|
|
|
return fmt.Errorf("enums of type %s are not yet implemented for this language", enum.ElementType.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close the declaration
|
|
|
|
fmt.Fprintf(w, "%s}\n", indent)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
func visitObjectTypes(properties []*schema.Property, visitor func(*schema.ObjectType)) {
|
|
|
|
codegen.VisitTypeClosure(properties, func(t schema.Type) {
|
|
|
|
if o, ok := t.(*schema.ObjectType); ok {
|
|
|
|
visitor(o)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2021-04-19 23:40:39 +00:00
|
|
|
})
|
2020-05-01 10:45:47 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
func (mod *modContext) genType(w io.Writer, obj *schema.ObjectType, propertyTypeQualifier string, input, state bool, level int) error {
|
|
|
|
args := obj.IsInputShape()
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
pt := &plainType{
|
|
|
|
mod: mod,
|
2021-04-16 02:03:28 +00:00
|
|
|
name: mod.typeName(obj, state, input, args),
|
2020-01-21 22:45:48 +00:00
|
|
|
comment: obj.Comment,
|
|
|
|
propertyTypeQualifier: propertyTypeQualifier,
|
|
|
|
properties: obj.Properties,
|
|
|
|
state: state,
|
2021-04-16 02:03:28 +00:00
|
|
|
args: args,
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if input {
|
2021-04-16 02:03:28 +00:00
|
|
|
pt.baseClass = "ResourceArgs"
|
2021-04-19 23:40:39 +00:00
|
|
|
if !args && mod.details(obj).plainType {
|
2021-04-16 02:03:28 +00:00
|
|
|
pt.baseClass = "InvokeArgs"
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
return pt.genInputType(w, level)
|
|
|
|
}
|
|
|
|
|
|
|
|
pt.genOutputType(w, level)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-10-13 18:33:22 +00:00
|
|
|
// pulumiImports is a slice of common imports that are used with the genHeader method.
|
2022-01-21 20:58:11 +00:00
|
|
|
func (mod *modContext) pulumiImports() []string {
|
2023-03-03 16:36:39 +00:00
|
|
|
pulumiImports := []string{
|
2022-01-21 20:58:11 +00:00
|
|
|
"System",
|
|
|
|
"System.Collections.Generic",
|
|
|
|
"System.Collections.Immutable",
|
|
|
|
"System.Threading.Tasks",
|
|
|
|
"Pulumi.Serialization",
|
|
|
|
}
|
|
|
|
if mod.RootNamespace() != "Pulumi" {
|
|
|
|
pulumiImports = append(pulumiImports, "Pulumi")
|
|
|
|
}
|
|
|
|
return pulumiImports
|
2020-10-13 18:33:22 +00:00
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
func (mod *modContext) genHeader(w io.Writer, using []string) {
|
|
|
|
fmt.Fprintf(w, "// *** WARNING: this file was generated by %v. ***\n", mod.tool)
|
|
|
|
fmt.Fprintf(w, "// *** Do not edit by hand unless you're certain you know what you are doing! ***\n")
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
for _, u := range using {
|
|
|
|
fmt.Fprintf(w, "using %s;\n", u)
|
|
|
|
}
|
|
|
|
if len(using) > 0 {
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-15 17:59:02 +00:00
|
|
|
func (mod *modContext) getConfigProperty(schemaType schema.Type) (string, string) {
|
2021-06-24 16:17:55 +00:00
|
|
|
schemaType = codegen.UnwrapType(schemaType)
|
|
|
|
|
2024-03-28 17:14:47 +00:00
|
|
|
qualifier := "Types"
|
|
|
|
propertyType := mod.typeString(schemaType, qualifier, false, false, false /*requireInitializers*/)
|
2020-04-15 17:59:02 +00:00
|
|
|
|
|
|
|
var getFunc string
|
|
|
|
nullableSigil := "?"
|
|
|
|
switch schemaType {
|
|
|
|
case schema.StringType:
|
|
|
|
getFunc = "Get"
|
|
|
|
case schema.BoolType:
|
|
|
|
getFunc = "GetBoolean"
|
|
|
|
case schema.IntType:
|
|
|
|
getFunc = "GetInt32"
|
|
|
|
case schema.NumberType:
|
|
|
|
getFunc = "GetDouble"
|
|
|
|
default:
|
|
|
|
switch t := schemaType.(type) {
|
|
|
|
case *schema.TokenType:
|
|
|
|
if t.UnderlyingType != nil {
|
|
|
|
return mod.getConfigProperty(t.UnderlyingType)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getFunc = "GetObject<" + propertyType + ">"
|
|
|
|
if _, ok := schemaType.(*schema.ArrayType); ok {
|
|
|
|
nullableSigil = ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return propertyType + nullableSigil, getFunc
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
func (mod *modContext) genConfig(variables []*schema.Property) (string, error) {
|
|
|
|
w := &bytes.Buffer{}
|
|
|
|
|
2021-07-20 17:22:45 +00:00
|
|
|
mod.genHeader(w, []string{"System", "System.Collections.Immutable"})
|
2020-01-21 22:45:48 +00:00
|
|
|
// Use the root namespace to avoid `Pulumi.Provider.Config.Config.VarName` usage.
|
|
|
|
fmt.Fprintf(w, "namespace %s\n", mod.namespaceName)
|
|
|
|
fmt.Fprintf(w, "{\n")
|
|
|
|
|
|
|
|
// Open the config class.
|
|
|
|
fmt.Fprintf(w, " public static class Config\n")
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
|
2023-09-22 13:41:15 +00:00
|
|
|
fmt.Fprintf(w, " [global::System.Diagnostics.CodeAnalysis.SuppressMessage(\"Microsoft.Design\", \"IDE1006\", Justification = \n")
|
2021-07-20 17:22:45 +00:00
|
|
|
fmt.Fprintf(w, " \"Double underscore prefix used to avoid conflicts with variable names.\")]\n")
|
|
|
|
fmt.Fprintf(w, " private sealed class __Value<T>\n")
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
|
|
|
|
fmt.Fprintf(w, " private readonly Func<T> _getter;\n")
|
|
|
|
fmt.Fprintf(w, " private T _value = default!;\n")
|
|
|
|
fmt.Fprintf(w, " private bool _set;\n")
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
fmt.Fprintf(w, " public __Value(Func<T> getter)\n")
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " _getter = getter;\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
fmt.Fprintf(w, " public T Get() => _set ? _value : _getter();\n")
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
fmt.Fprintf(w, " public void Set(T value)\n")
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " _value = value;\n")
|
|
|
|
fmt.Fprintf(w, " _set = true;\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Create a config bag for the variables to pull from.
|
2022-12-08 14:23:07 +00:00
|
|
|
fmt.Fprintf(w, " private static readonly global::Pulumi.Config __config = new global::Pulumi.Config(\"%v\");\n", mod.pkg.Name())
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Emit an entry for all config variables.
|
|
|
|
for _, p := range variables {
|
2020-04-15 17:59:02 +00:00
|
|
|
propertyType, getFunc := mod.getConfigProperty(p.Type)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-05-01 10:45:47 +00:00
|
|
|
propertyName := mod.propertyName(p)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
initializer := fmt.Sprintf("__config.%s(\"%s\")", getFunc, p.Name)
|
|
|
|
if p.DefaultValue != nil {
|
|
|
|
dv, err := mod.getDefaultValue(p.DefaultValue, p.Type)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
initializer += " ?? " + dv
|
|
|
|
}
|
|
|
|
|
2021-07-20 17:22:45 +00:00
|
|
|
fmt.Fprintf(w, " private static readonly __Value<%[1]s> _%[2]s = new __Value<%[1]s>(() => %[3]s);\n", propertyType, p.Name, initializer)
|
2020-01-21 22:45:48 +00:00
|
|
|
printComment(w, p.Comment, " ")
|
2021-07-20 17:22:45 +00:00
|
|
|
fmt.Fprintf(w, " public static %s %s\n", propertyType, propertyName)
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
fmt.Fprintf(w, " get => _%s.Get();\n", p.Name)
|
|
|
|
fmt.Fprintf(w, " set => _%s.Set(value);\n", p.Name)
|
|
|
|
fmt.Fprintf(w, " }\n")
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
}
|
|
|
|
|
2024-03-28 17:14:47 +00:00
|
|
|
// generate config-friendly object types used in config
|
|
|
|
// regardless of whether they are defined inline or from a shared type
|
|
|
|
var usedObjectTypes []*schema.ObjectType
|
|
|
|
visitObjectTypes(variables, func(objectType *schema.ObjectType) {
|
|
|
|
usedObjectTypes = append(usedObjectTypes, objectType)
|
|
|
|
})
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Emit any nested types.
|
2024-03-28 17:14:47 +00:00
|
|
|
if len(usedObjectTypes) > 0 {
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, " public static class Types\n")
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
|
2024-03-28 17:14:47 +00:00
|
|
|
for _, typ := range usedObjectTypes {
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
|
|
|
|
// Open the class.
|
|
|
|
fmt.Fprintf(w, " public class %s\n", tokenToName(typ.Token))
|
|
|
|
fmt.Fprintf(w, " {\n")
|
|
|
|
|
|
|
|
// Generate each output field.
|
|
|
|
for _, prop := range typ.Properties {
|
|
|
|
name := mod.propertyName(prop)
|
2021-06-24 16:17:55 +00:00
|
|
|
typ := mod.typeString(prop.Type, "Types", false, false, false)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
initializer := ""
|
2021-06-24 16:17:55 +00:00
|
|
|
if !prop.IsRequired() && !isValueType(prop.Type) && !isImmutableArrayType(codegen.UnwrapType(prop.Type), false) {
|
2020-01-21 22:45:48 +00:00
|
|
|
initializer = " = null!;"
|
|
|
|
}
|
|
|
|
|
|
|
|
printComment(w, prop.Comment, " ")
|
|
|
|
fmt.Fprintf(w, " public %s %s { get; set; }%s\n", typ, name, initializer)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close the class.
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close the config class and namespace.
|
|
|
|
fmt.Fprintf(w, " }\n")
|
|
|
|
|
|
|
|
// Close the namespace.
|
|
|
|
fmt.Fprintf(w, "}\n")
|
|
|
|
|
|
|
|
return w.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (mod *modContext) genUtilities() (string, error) {
|
|
|
|
// Strip any 'v' off of the version.
|
|
|
|
w := &bytes.Buffer{}
|
2022-12-08 14:23:07 +00:00
|
|
|
def, err := mod.pkg.Definition()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
err = csharpUtilitiesTemplate.Execute(w, csharpUtilitiesTemplateContext{
|
|
|
|
Name: namespaceName(mod.namespaces, mod.pkg.Name()),
|
2022-01-14 00:10:17 +00:00
|
|
|
Namespace: mod.namespaceName,
|
|
|
|
ClassName: "Utilities",
|
|
|
|
Tool: mod.tool,
|
2022-12-08 14:23:07 +00:00
|
|
|
PluginDownloadURL: def.PluginDownloadURL,
|
2020-01-21 22:45:48 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return w.String(), nil
|
|
|
|
}
|
|
|
|
|
2022-11-09 19:49:59 +00:00
|
|
|
func (mod *modContext) gen(fs codegen.Fs) error {
|
2020-01-21 22:45:48 +00:00
|
|
|
nsComponents := strings.Split(mod.namespaceName, ".")
|
|
|
|
if len(nsComponents) > 0 {
|
|
|
|
// Trim off "Pulumi.Pkg"
|
|
|
|
nsComponents = nsComponents[2:]
|
|
|
|
}
|
|
|
|
|
|
|
|
dir := path.Join(nsComponents...)
|
|
|
|
if mod.mod == "config" {
|
|
|
|
dir = "Config"
|
|
|
|
}
|
|
|
|
|
|
|
|
var files []string
|
|
|
|
for p := range fs {
|
|
|
|
d := path.Dir(p)
|
|
|
|
if d == "." {
|
|
|
|
d = ""
|
|
|
|
}
|
|
|
|
if d == dir {
|
|
|
|
files = append(files, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
addFile := func(name, contents string) {
|
|
|
|
p := path.Join(dir, name)
|
|
|
|
files = append(files, p)
|
2022-11-09 19:49:59 +00:00
|
|
|
fs.Add(p, []byte(contents))
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure that the target module directory contains a README.md file.
|
2022-12-08 14:23:07 +00:00
|
|
|
readme := mod.pkg.Description()
|
2020-01-21 22:45:48 +00:00
|
|
|
if readme != "" && readme[len(readme)-1] != '\n' {
|
|
|
|
readme += "\n"
|
|
|
|
}
|
2022-11-09 19:49:59 +00:00
|
|
|
fs.Add(filepath.Join(dir, "README.md"), []byte(readme))
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
// Utilities, config
|
|
|
|
switch mod.mod {
|
|
|
|
case "":
|
|
|
|
utilities, err := mod.genUtilities()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-11-09 19:49:59 +00:00
|
|
|
fs.Add("Utilities.cs", []byte(utilities))
|
2020-01-21 22:45:48 +00:00
|
|
|
case "config":
|
2022-12-08 14:23:07 +00:00
|
|
|
config, err := mod.pkg.Config()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(config) > 0 {
|
|
|
|
config, err := mod.genConfig(config)
|
2020-01-21 22:45:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
addFile("Config.cs", config)
|
2020-04-14 08:30:25 +00:00
|
|
|
return nil
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resources
|
|
|
|
for _, r := range mod.resources {
|
2021-11-12 00:00:03 +00:00
|
|
|
if r.IsOverlay {
|
|
|
|
// This resource code is generated by the provider, so no further action is required.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
buffer := &bytes.Buffer{}
|
2022-01-21 20:58:11 +00:00
|
|
|
importStrings := mod.pulumiImports()
|
2020-10-13 18:33:22 +00:00
|
|
|
mod.genHeader(buffer, importStrings)
|
2020-01-21 22:45:48 +00:00
|
|
|
|
|
|
|
if err := mod.genResource(buffer, r); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
addFile(resourceName(r)+".cs", buffer.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Functions
|
|
|
|
for _, f := range mod.functions {
|
2021-11-12 00:00:03 +00:00
|
|
|
if f.IsOverlay {
|
|
|
|
// This function code is generated by the provider, so no further action is required.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2021-10-18 22:18:15 +00:00
|
|
|
code, err := mod.genFunctionFileCode(f)
|
|
|
|
if err != nil {
|
2020-01-21 22:45:48 +00:00
|
|
|
return err
|
|
|
|
}
|
2021-10-18 22:18:15 +00:00
|
|
|
addFile(tokenToName(f.Token)+".cs", code)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Nested types
|
|
|
|
for _, t := range mod.types {
|
2021-11-12 00:00:03 +00:00
|
|
|
if t.IsOverlay {
|
|
|
|
// This type is generated by the provider, so no further action is required.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
if mod.details(t).inputType {
|
|
|
|
buffer := &bytes.Buffer{}
|
2022-01-21 20:58:11 +00:00
|
|
|
mod.genHeader(buffer, mod.pulumiImports())
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
fmt.Fprintf(buffer, "namespace %s\n", mod.tokenToNamespace(t.Token, "Inputs"))
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(buffer, "{\n")
|
2021-06-24 16:17:55 +00:00
|
|
|
|
|
|
|
if err := mod.genType(buffer, t, "Inputs", true, false, 1); err != nil {
|
|
|
|
return err
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2021-06-24 16:17:55 +00:00
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(buffer, "}\n")
|
|
|
|
|
2021-06-24 16:17:55 +00:00
|
|
|
name := tokenToName(t.Token)
|
|
|
|
if t.IsInputShape() {
|
|
|
|
name += "Args"
|
|
|
|
}
|
|
|
|
addFile(path.Join("Inputs", name+".cs"), buffer.String())
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
if mod.details(t).stateType {
|
|
|
|
buffer := &bytes.Buffer{}
|
2022-01-21 20:58:11 +00:00
|
|
|
mod.genHeader(buffer, mod.pulumiImports())
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
fmt.Fprintf(buffer, "namespace %s\n", mod.tokenToNamespace(t.Token, "Inputs"))
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(buffer, "{\n")
|
2021-06-24 16:17:55 +00:00
|
|
|
if err := mod.genType(buffer, t, "Inputs", true, true, 1); err != nil {
|
2020-01-21 22:45:48 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Fprintf(buffer, "}\n")
|
|
|
|
addFile(path.Join("Inputs", tokenToName(t.Token)+"GetArgs.cs"), buffer.String())
|
|
|
|
}
|
|
|
|
if mod.details(t).outputType {
|
|
|
|
buffer := &bytes.Buffer{}
|
2022-01-21 20:58:11 +00:00
|
|
|
mod.genHeader(buffer, mod.pulumiImports())
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-05-19 09:41:06 +00:00
|
|
|
fmt.Fprintf(buffer, "namespace %s\n", mod.tokenToNamespace(t.Token, "Outputs"))
|
2020-01-21 22:45:48 +00:00
|
|
|
fmt.Fprintf(buffer, "{\n")
|
2021-06-24 16:17:55 +00:00
|
|
|
if err := mod.genType(buffer, t, "Outputs", false, false, 1); err != nil {
|
2020-01-21 22:45:48 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Fprintf(buffer, "}\n")
|
|
|
|
|
|
|
|
suffix := ""
|
2021-04-19 23:40:39 +00:00
|
|
|
if (mod.isTFCompatMode() || mod.isK8sCompatMode()) && mod.details(t).plainType {
|
2020-01-21 22:45:48 +00:00
|
|
|
suffix = "Result"
|
|
|
|
}
|
|
|
|
addFile(path.Join("Outputs", tokenToName(t.Token)+suffix+".cs"), buffer.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-06 17:01:03 +00:00
|
|
|
// Enums
|
|
|
|
if len(mod.enums) > 0 {
|
|
|
|
buffer := &bytes.Buffer{}
|
|
|
|
mod.genHeader(buffer, []string{"System", "System.ComponentModel", "Pulumi"})
|
|
|
|
|
|
|
|
if err := mod.genEnums(buffer, mod.enums); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
addFile("Enums.cs", buffer.String())
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// genPackageMetadata generates all the non-code metadata required by a Pulumi package.
|
2021-10-18 22:18:15 +00:00
|
|
|
func genPackageMetadata(pkg *schema.Package,
|
|
|
|
assemblyName string,
|
|
|
|
packageReferences map[string]string,
|
|
|
|
projectReferences []string,
|
2023-03-03 16:36:39 +00:00
|
|
|
files codegen.Fs,
|
2024-04-19 10:21:09 +00:00
|
|
|
localDependencies map[string]string,
|
2023-03-03 16:36:39 +00:00
|
|
|
) error {
|
2023-03-27 14:02:03 +00:00
|
|
|
version := ""
|
|
|
|
lang, ok := pkg.Language["csharp"].(CSharpPackageInfo)
|
|
|
|
if pkg.Version != nil && ok && lang.RespectSchemaVersion {
|
|
|
|
version = pkg.Version.String()
|
|
|
|
files.Add("version.txt", []byte(version))
|
|
|
|
}
|
2024-04-22 06:37:50 +00:00
|
|
|
if pkg.SupportPack {
|
|
|
|
if pkg.Version == nil {
|
|
|
|
return errors.New("package version is required")
|
|
|
|
}
|
|
|
|
version = pkg.Version.String()
|
|
|
|
files.Add("version.txt", []byte(version))
|
|
|
|
}
|
2023-03-27 14:02:03 +00:00
|
|
|
|
2024-04-19 10:21:09 +00:00
|
|
|
projectFile, err := genProjectFile(pkg, assemblyName, packageReferences, projectReferences, version, localDependencies)
|
2020-01-21 22:45:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
logo, err := getLogo(pkg)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-02-03 16:07:13 +00:00
|
|
|
|
|
|
|
pulumiPlugin := &plugin.PulumiPluginJSON{
|
2021-12-02 20:47:33 +00:00
|
|
|
Resource: true,
|
|
|
|
Name: pkg.Name,
|
|
|
|
Server: pkg.PluginDownloadURL,
|
2023-03-27 14:02:03 +00:00
|
|
|
Version: version,
|
2022-02-03 16:07:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
plugin, err := (pulumiPlugin).JSON()
|
2021-12-02 20:47:33 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2022-11-09 19:49:59 +00:00
|
|
|
files.Add(assemblyName+".csproj", projectFile)
|
|
|
|
files.Add("logo.png", logo)
|
|
|
|
files.Add("pulumi-plugin.json", plugin)
|
2020-01-21 22:45:48 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// genProjectFile emits a C# project file into the configured output directory.
|
2021-10-18 22:18:15 +00:00
|
|
|
func genProjectFile(pkg *schema.Package,
|
|
|
|
assemblyName string,
|
|
|
|
packageReferences map[string]string,
|
2023-03-03 16:36:39 +00:00
|
|
|
projectReferences []string,
|
2023-03-27 14:02:03 +00:00
|
|
|
version string,
|
2024-04-19 10:21:09 +00:00
|
|
|
localDependencies map[string]string,
|
2023-03-03 16:36:39 +00:00
|
|
|
) ([]byte, error) {
|
2022-11-09 21:53:13 +00:00
|
|
|
if packageReferences == nil {
|
|
|
|
packageReferences = map[string]string{}
|
|
|
|
}
|
2022-11-14 14:10:00 +00:00
|
|
|
|
2024-04-22 06:37:50 +00:00
|
|
|
// Find all the local dependency folders
|
|
|
|
folders := mapset.NewSet[string]()
|
|
|
|
for _, dep := range localDependencies {
|
|
|
|
folders.Add(path.Dir(dep))
|
|
|
|
}
|
|
|
|
restoreSources := ""
|
|
|
|
if len(folders.ToSlice()) > 0 {
|
|
|
|
restoreSources = strings.Join(folders.ToSlice(), ";")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the Pulumi package reference
|
|
|
|
for _, path := range localDependencies {
|
|
|
|
filename := filepath.Base(path)
|
|
|
|
pkg, rest, _ := strings.Cut(filename, ".")
|
|
|
|
version, _ := strings.CutSuffix(rest, ".nupkg")
|
|
|
|
packageReferences[pkg] = version
|
|
|
|
}
|
|
|
|
|
2022-11-14 14:10:00 +00:00
|
|
|
// if we don't have a package reference to Pulumi SDK from nuget
|
|
|
|
// we need to add it, unless we are referencing a local Pulumi SDK project via a project reference
|
2022-11-09 21:53:13 +00:00
|
|
|
if _, ok := packageReferences["Pulumi"]; !ok {
|
2022-11-14 14:10:00 +00:00
|
|
|
referencedLocalPulumiProject := false
|
|
|
|
for _, projectReference := range projectReferences {
|
|
|
|
if strings.HasSuffix(projectReference, "Pulumi.csproj") {
|
|
|
|
referencedLocalPulumiProject = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// only add a package reference to Pulumi if we're not referencing a local Pulumi project
|
|
|
|
// which we usually do when testing schemas locally
|
|
|
|
if !referencedLocalPulumiProject {
|
2023-03-02 16:34:28 +00:00
|
|
|
packageReferences["Pulumi"] = "[3.54.1.0,4)"
|
2022-11-14 14:10:00 +00:00
|
|
|
}
|
2022-11-09 21:53:13 +00:00
|
|
|
}
|
2022-11-14 14:10:00 +00:00
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
w := &bytes.Buffer{}
|
|
|
|
err := csharpProjectFileTemplate.Execute(w, csharpProjectFileTemplateContext{
|
|
|
|
XMLDoc: fmt.Sprintf(`.\%s.xml`, assemblyName),
|
|
|
|
Package: pkg,
|
|
|
|
PackageReferences: packageReferences,
|
2021-10-18 22:18:15 +00:00
|
|
|
ProjectReferences: projectReferences,
|
2023-03-27 14:02:03 +00:00
|
|
|
Version: version,
|
2024-04-22 06:37:50 +00:00
|
|
|
RestoreSources: restoreSources,
|
2020-01-21 22:45:48 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return w.Bytes(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// emitLogo downloads an image and saves it as logo.png into the configured output directory.
|
|
|
|
func getLogo(pkg *schema.Package) ([]byte, error) {
|
|
|
|
url := pkg.LogoURL
|
|
|
|
if url == "" {
|
|
|
|
// Default to a generic Pulumi logo from the parent repository.
|
2021-04-21 19:26:00 +00:00
|
|
|
url = "https://raw.githubusercontent.com/pulumi/pulumi/dbc96206bec722b7791a22ff50e895ab7c0abdc0/sdk/dotnet/pulumi_logo_64x64.png"
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the data.
|
2023-01-06 00:07:45 +00:00
|
|
|
//nolint:gosec
|
2020-01-21 22:45:48 +00:00
|
|
|
resp, err := http.Get(url)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer contract.IgnoreClose(resp.Body)
|
|
|
|
|
2023-01-06 22:39:16 +00:00
|
|
|
return io.ReadAll(resp.Body)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
2020-04-20 23:36:05 +00:00
|
|
|
func computePropertyNames(props []*schema.Property, names map[*schema.Property]string) {
|
2020-01-21 22:45:48 +00:00
|
|
|
for _, p := range props {
|
2020-04-20 23:36:05 +00:00
|
|
|
if info, ok := p.Language["csharp"].(CSharpPropertyInfo); ok && info.Name != "" {
|
|
|
|
names[p] = info.Name
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-30 22:19:21 +00:00
|
|
|
// LanguageResource is derived from the schema and can be used by downstream codegen.
|
|
|
|
type LanguageResource struct {
|
|
|
|
*schema.Resource
|
|
|
|
|
|
|
|
Name string // The resource name (e.g. Deployment)
|
|
|
|
Package string // The package name (e.g. Apps.V1)
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
|
2020-11-20 20:09:34 +00:00
|
|
|
func generateModuleContextMap(tool string, pkg *schema.Package) (map[string]*modContext, *CSharpPackageInfo, error) {
|
|
|
|
// Decode .NET-specific info for each package as we discover them.
|
|
|
|
infos := map[*schema.Package]*CSharpPackageInfo{}
|
2023-03-03 16:36:39 +00:00
|
|
|
getPackageInfo := func(p schema.PackageReference) *CSharpPackageInfo {
|
2022-12-08 14:23:07 +00:00
|
|
|
def, err := p.Definition()
|
2023-02-17 01:23:09 +00:00
|
|
|
contract.AssertNoErrorf(err, "error loading definition for package %v", p.Name())
|
2022-12-08 14:23:07 +00:00
|
|
|
info, ok := infos[def]
|
2020-11-20 20:09:34 +00:00
|
|
|
if !ok {
|
2022-12-08 14:23:07 +00:00
|
|
|
err := def.ImportLanguages(map[string]schema.Language{"csharp": Importer})
|
2023-02-17 01:23:09 +00:00
|
|
|
contract.AssertNoErrorf(err, "error importing csharp language info for package %q", p.Name())
|
2020-11-20 20:09:34 +00:00
|
|
|
csharpInfo, _ := pkg.Language["csharp"].(CSharpPackageInfo)
|
|
|
|
info = &csharpInfo
|
2022-12-08 14:23:07 +00:00
|
|
|
infos[def] = info
|
2020-11-20 20:09:34 +00:00
|
|
|
}
|
|
|
|
return info
|
|
|
|
}
|
2022-12-08 14:23:07 +00:00
|
|
|
infos[pkg] = getPackageInfo(pkg.Reference())
|
2020-11-20 20:09:34 +00:00
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
propertyNames := map[*schema.Property]string{}
|
2020-05-01 10:45:47 +00:00
|
|
|
computePropertyNames(pkg.Config, propertyNames)
|
|
|
|
computePropertyNames(pkg.Provider.InputProperties, propertyNames)
|
2020-01-21 22:45:48 +00:00
|
|
|
for _, r := range pkg.Resources {
|
2021-11-12 00:00:03 +00:00
|
|
|
if r.IsOverlay {
|
|
|
|
// This resource code is generated by the provider, so no further action is required.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-04-20 23:36:05 +00:00
|
|
|
computePropertyNames(r.Properties, propertyNames)
|
|
|
|
computePropertyNames(r.InputProperties, propertyNames)
|
2020-01-21 22:45:48 +00:00
|
|
|
if r.StateInputs != nil {
|
2020-04-20 23:36:05 +00:00
|
|
|
computePropertyNames(r.StateInputs.Properties, propertyNames)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, f := range pkg.Functions {
|
2021-11-12 00:00:03 +00:00
|
|
|
if f.IsOverlay {
|
|
|
|
// This function code is generated by the provider, so no further action is required.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
if f.Inputs != nil {
|
2020-04-20 23:36:05 +00:00
|
|
|
computePropertyNames(f.Inputs.Properties, propertyNames)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2023-01-11 22:17:14 +00:00
|
|
|
if f.ReturnType != nil {
|
|
|
|
if objectType, ok := f.ReturnType.(*schema.ObjectType); ok && objectType != nil {
|
|
|
|
computePropertyNames(objectType.Properties, propertyNames)
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
2023-01-11 22:17:14 +00:00
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
for _, t := range pkg.Types {
|
|
|
|
if obj, ok := t.(*schema.ObjectType); ok {
|
2020-04-20 23:36:05 +00:00
|
|
|
computePropertyNames(obj.Properties, propertyNames)
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// group resources, types, and functions into Go packages
|
|
|
|
modules := map[string]*modContext{}
|
|
|
|
details := map[*schema.ObjectType]*typeDetails{}
|
|
|
|
|
2022-12-08 14:23:07 +00:00
|
|
|
var getMod func(modName string, p schema.PackageReference) *modContext
|
|
|
|
getMod = func(modName string, p schema.PackageReference) *modContext {
|
2020-01-21 22:45:48 +00:00
|
|
|
mod, ok := modules[modName]
|
|
|
|
if !ok {
|
2020-11-20 20:09:34 +00:00
|
|
|
info := getPackageInfo(p)
|
2022-01-21 20:58:11 +00:00
|
|
|
ns := info.GetRootNamespace() + "." + namespaceName(info.Namespaces, pkg.Name)
|
2020-01-21 22:45:48 +00:00
|
|
|
if modName != "" {
|
|
|
|
ns += "." + namespaceName(info.Namespaces, modName)
|
|
|
|
}
|
|
|
|
mod = &modContext{
|
2021-10-01 18:33:02 +00:00
|
|
|
pkg: p,
|
|
|
|
mod: modName,
|
|
|
|
tool: tool,
|
|
|
|
namespaceName: ns,
|
|
|
|
namespaces: info.Namespaces,
|
2022-01-21 20:58:11 +00:00
|
|
|
rootNamespace: info.GetRootNamespace(),
|
2021-10-01 18:33:02 +00:00
|
|
|
typeDetails: details,
|
|
|
|
propertyNames: propertyNames,
|
|
|
|
compatibility: info.Compatibility,
|
|
|
|
dictionaryConstructors: info.DictionaryConstructors,
|
|
|
|
liftSingleValueMethodReturns: info.LiftSingleValueMethodReturns,
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if modName != "" {
|
|
|
|
parentName := path.Dir(modName)
|
2020-06-03 01:15:21 +00:00
|
|
|
if parentName == "." {
|
|
|
|
parentName = ""
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2020-11-20 20:09:34 +00:00
|
|
|
parent := getMod(parentName, p)
|
2020-01-21 22:45:48 +00:00
|
|
|
parent.children = append(parent.children, mod)
|
|
|
|
}
|
|
|
|
|
2021-01-26 19:02:34 +00:00
|
|
|
// Save the module only if it's for the current package.
|
|
|
|
// This way, modules for external packages are not saved.
|
2022-12-08 14:23:07 +00:00
|
|
|
if codegen.PkgEquals(p, pkg.Reference()) {
|
2021-01-26 19:02:34 +00:00
|
|
|
modules[modName] = mod
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
return mod
|
|
|
|
}
|
|
|
|
|
2022-12-08 14:23:07 +00:00
|
|
|
getModFromToken := func(token string, p schema.PackageReference) *modContext {
|
2020-11-20 20:09:34 +00:00
|
|
|
return getMod(p.TokenToModule(token), p)
|
2020-06-03 01:15:21 +00:00
|
|
|
}
|
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Create the config module if necessary.
|
|
|
|
if len(pkg.Config) > 0 {
|
2022-12-08 14:23:07 +00:00
|
|
|
cfg := getMod("config", pkg.Reference())
|
2022-01-21 20:58:11 +00:00
|
|
|
cfg.namespaceName = fmt.Sprintf("%s.%s", cfg.RootNamespace(), namespaceName(infos[pkg].Namespaces, pkg.Name))
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find input and output types referenced by resources.
|
|
|
|
scanResource := func(r *schema.Resource) {
|
2022-12-08 14:23:07 +00:00
|
|
|
mod := getModFromToken(r.Token, pkg.Reference())
|
2020-01-21 22:45:48 +00:00
|
|
|
mod.resources = append(mod.resources, r)
|
2021-06-24 16:17:55 +00:00
|
|
|
visitObjectTypes(r.Properties, func(t *schema.ObjectType) {
|
2022-12-08 14:23:07 +00:00
|
|
|
getModFromToken(t.Token, t.PackageReference).details(t).outputType = true
|
2021-04-19 23:40:39 +00:00
|
|
|
})
|
2021-06-24 16:17:55 +00:00
|
|
|
visitObjectTypes(r.InputProperties, func(t *schema.ObjectType) {
|
2022-12-08 14:23:07 +00:00
|
|
|
getModFromToken(t.Token, t.PackageReference).details(t).inputType = true
|
2021-04-19 23:40:39 +00:00
|
|
|
})
|
2020-01-21 22:45:48 +00:00
|
|
|
if r.StateInputs != nil {
|
2021-06-24 16:17:55 +00:00
|
|
|
visitObjectTypes(r.StateInputs.Properties, func(t *schema.ObjectType) {
|
2022-12-08 14:23:07 +00:00
|
|
|
getModFromToken(t.Token, t.PackageReference).details(t).inputType = true
|
|
|
|
getModFromToken(t.Token, t.PackageReference).details(t).stateType = true
|
2020-01-21 22:45:48 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
scanResource(pkg.Provider)
|
|
|
|
for _, r := range pkg.Resources {
|
|
|
|
scanResource(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find input and output types referenced by functions.
|
|
|
|
for _, f := range pkg.Functions {
|
2021-11-12 00:00:03 +00:00
|
|
|
if f.IsOverlay {
|
|
|
|
// This function code is generated by the provider, so no further action is required.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2022-12-08 14:23:07 +00:00
|
|
|
mod := getModFromToken(f.Token, pkg.Reference())
|
2021-07-27 23:42:17 +00:00
|
|
|
if !f.IsMethod {
|
|
|
|
mod.functions = append(mod.functions, f)
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
if f.Inputs != nil {
|
2021-06-24 16:17:55 +00:00
|
|
|
visitObjectTypes(f.Inputs.Properties, func(t *schema.ObjectType) {
|
2022-12-08 14:23:07 +00:00
|
|
|
details := getModFromToken(t.Token, t.PackageReference).details(t)
|
2021-04-16 02:03:28 +00:00
|
|
|
details.inputType = true
|
2021-04-19 23:40:39 +00:00
|
|
|
details.plainType = true
|
2020-01-21 22:45:48 +00:00
|
|
|
})
|
2021-10-18 22:18:15 +00:00
|
|
|
if f.NeedsOutputVersion() {
|
|
|
|
visitObjectTypes(f.Inputs.InputShape.Properties, func(t *schema.ObjectType) {
|
2022-12-08 14:23:07 +00:00
|
|
|
details := getModFromToken(t.Token, t.PackageReference).details(t)
|
2021-10-18 22:18:15 +00:00
|
|
|
details.inputType = true
|
|
|
|
details.usedInFunctionOutputVersionInputs = true
|
|
|
|
})
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
2023-01-11 22:17:14 +00:00
|
|
|
if f.ReturnType != nil {
|
|
|
|
// special case where the return type is defined inline with the function
|
|
|
|
if objectType, ok := f.ReturnType.(*schema.ObjectType); ok && f.InlineObjectAsReturnType {
|
|
|
|
visitObjectTypes(objectType.Properties, func(t *schema.ObjectType) {
|
|
|
|
details := getModFromToken(t.Token, t.PackageReference).details(t)
|
|
|
|
details.outputType = true
|
|
|
|
details.plainType = true
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
// otherwise, the return type is a reference to a type defined elsewhere
|
|
|
|
codegen.VisitType(f.ReturnType, func(schemaType schema.Type) {
|
|
|
|
if t, ok := schemaType.(*schema.ObjectType); ok {
|
|
|
|
details := getModFromToken(t.Token, t.PackageReference).details(t)
|
|
|
|
details.outputType = true
|
|
|
|
details.plainType = true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find nested types.
|
|
|
|
for _, t := range pkg.Types {
|
2020-11-06 17:01:03 +00:00
|
|
|
switch typ := t.(type) {
|
|
|
|
case *schema.ObjectType:
|
2022-12-08 14:23:07 +00:00
|
|
|
mod := getModFromToken(typ.Token, pkg.Reference())
|
2020-11-06 17:01:03 +00:00
|
|
|
mod.types = append(mod.types, typ)
|
|
|
|
case *schema.EnumType:
|
2021-11-16 23:53:28 +00:00
|
|
|
if !typ.IsOverlay {
|
2022-12-08 14:23:07 +00:00
|
|
|
mod := getModFromToken(typ.Token, pkg.Reference())
|
2021-11-16 23:53:28 +00:00
|
|
|
mod.enums = append(mod.enums, typ)
|
|
|
|
}
|
2020-11-06 17:01:03 +00:00
|
|
|
default:
|
|
|
|
continue
|
2020-01-21 22:45:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-20 20:09:34 +00:00
|
|
|
return modules, infos[pkg], nil
|
2020-06-30 22:19:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// LanguageResources returns a map of resources that can be used by downstream codegen. The map
|
|
|
|
// key is the resource schema token.
|
|
|
|
func LanguageResources(tool string, pkg *schema.Package) (map[string]LanguageResource, error) {
|
2020-11-20 20:09:34 +00:00
|
|
|
modules, info, err := generateModuleContextMap(tool, pkg)
|
2020-06-30 22:19:21 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
resources := map[string]LanguageResource{}
|
|
|
|
for modName, mod := range modules {
|
|
|
|
if modName == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, r := range mod.resources {
|
2021-11-12 00:00:03 +00:00
|
|
|
if r.IsOverlay {
|
|
|
|
// This resource code is generated by the provider, so no further action is required.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-06-30 22:19:21 +00:00
|
|
|
lr := LanguageResource{
|
|
|
|
Resource: r,
|
|
|
|
Package: namespaceName(info.Namespaces, modName),
|
2023-11-29 16:35:08 +00:00
|
|
|
Name: resourceName(r),
|
2020-06-30 22:19:21 +00:00
|
|
|
}
|
|
|
|
resources[r.Token] = lr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return resources, nil
|
|
|
|
}
|
|
|
|
|
2024-04-19 10:21:09 +00:00
|
|
|
func GeneratePackage(
|
|
|
|
tool string, pkg *schema.Package, extraFiles map[string][]byte, localDependencies map[string]string,
|
|
|
|
) (map[string][]byte, error) {
|
2020-11-20 20:09:34 +00:00
|
|
|
modules, info, err := generateModuleContextMap(tool, pkg)
|
2020-06-30 22:19:21 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-01-21 20:58:11 +00:00
|
|
|
assemblyName := info.GetRootNamespace() + "." + namespaceName(info.Namespaces, pkg.Name)
|
2020-06-30 22:19:21 +00:00
|
|
|
|
2020-01-21 22:45:48 +00:00
|
|
|
// Generate each module.
|
2022-11-09 19:49:59 +00:00
|
|
|
files := codegen.Fs{}
|
2020-01-21 22:45:48 +00:00
|
|
|
for p, f := range extraFiles {
|
2022-11-09 19:49:59 +00:00
|
|
|
files.Add(p, f)
|
2021-09-23 02:39:07 +00:00
|
|
|
}
|
2020-01-21 22:45:48 +00:00
|
|
|
for _, mod := range modules {
|
|
|
|
if err := mod.gen(files); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-30 22:19:21 +00:00
|
|
|
// Finally emit the package metadata.
|
2021-10-18 22:18:15 +00:00
|
|
|
if err := genPackageMetadata(pkg,
|
|
|
|
assemblyName,
|
|
|
|
info.PackageReferences,
|
|
|
|
info.ProjectReferences,
|
2024-04-19 10:21:09 +00:00
|
|
|
files,
|
|
|
|
localDependencies); err != nil {
|
2020-01-21 22:45:48 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return files, nil
|
|
|
|
}
|