mirror of https://github.com/pulumi/pulumi.git
874 lines
26 KiB
Go
874 lines
26 KiB
Go
// Copyright 2016-2020, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package importer
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"math"
|
|
"strings"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen"
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/hcl2/model"
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/hcl2/syntax"
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/schema"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy/providers"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
"github.com/zclconf/go-cty/cty"
|
|
)
|
|
|
|
// Null represents Pulumi HCL2's `null` variable.
|
|
var Null = &model.Variable{
|
|
Name: "null",
|
|
VariableType: model.NoneType,
|
|
}
|
|
|
|
type PathedLiteralValue struct {
|
|
Root string
|
|
Value string
|
|
ExpressionReference *model.ScopeTraversalExpression
|
|
}
|
|
|
|
// ImportState tracks the state of an import process.
|
|
type ImportState struct {
|
|
Names NameTable
|
|
PathedLiteralValues []PathedLiteralValue
|
|
|
|
// A snapshot of the resources in the Pulumi program that new resources are being imported to. This is used to resolve
|
|
// references to packages providers.
|
|
Snapshot []*resource.State
|
|
}
|
|
|
|
// filterReferences filters out self-references from the import state so that if a resource has a property
|
|
// that happens to have the same value as the ID of that resource, it doesn't create a self-reference.
|
|
func filterReferences(resourceName string, importState ImportState) ImportState {
|
|
pathedLiteralValues := make([]PathedLiteralValue, 0)
|
|
for _, pathedLiteralValue := range importState.PathedLiteralValues {
|
|
if pathedLiteralValue.Root != resourceName {
|
|
// if the pathedLiteralValue is not a self-reference, add it to the list
|
|
pathedLiteralValues = append(pathedLiteralValues, pathedLiteralValue)
|
|
}
|
|
}
|
|
|
|
withoutDuplicates := removeDuplicatePathedValues(pathedLiteralValues)
|
|
|
|
return ImportState{
|
|
Names: importState.Names,
|
|
PathedLiteralValues: withoutDuplicates,
|
|
Snapshot: importState.Snapshot,
|
|
}
|
|
}
|
|
|
|
// GenerateHCL2Definition generates a Pulumi HCL2 definition for a given resource.
|
|
func GenerateHCL2Definition(
|
|
loader schema.Loader,
|
|
state *resource.State,
|
|
importState ImportState,
|
|
) (*model.Block, *schema.PackageDescriptor, error) {
|
|
// First up, we'll need to load the appropriate package for this resource. We'll do this by grabbing the resource's
|
|
// provider reference and looking up that provider resource in the current program snapshot. From there, we can build
|
|
// a package descriptor and load the package and its schema.
|
|
providerRef, err := providers.ParseReference(state.Provider)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("parse resource provider reference: %w", err)
|
|
}
|
|
|
|
var provider *resource.State
|
|
for _, s := range importState.Snapshot {
|
|
if s.URN == providerRef.URN() && s.ID == providerRef.ID() {
|
|
provider = s
|
|
break
|
|
}
|
|
}
|
|
if provider == nil {
|
|
return nil, nil, fmt.Errorf("provider %v not found in snapshot", providerRef)
|
|
}
|
|
|
|
packageName := state.Type.Package()
|
|
pluginName, err := providers.GetProviderName(packageName, provider.Inputs)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("get provider name: %w", err)
|
|
}
|
|
pluginVersion, err := providers.GetProviderVersion(provider.Inputs)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("get provider version: %w", err)
|
|
}
|
|
downloadURL, err := providers.GetProviderDownloadURL(provider.Inputs)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("get provider version: %w", err)
|
|
}
|
|
var parameterization *schema.ParameterizationDescriptor
|
|
parameters, err := providers.GetProviderParameterization(packageName, provider.Inputs)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("get provider parameterization: %w", err)
|
|
}
|
|
if parameters != nil {
|
|
parameterization = &schema.ParameterizationDescriptor{
|
|
Name: string(parameters.Name),
|
|
Version: parameters.Version,
|
|
Value: parameters.Value,
|
|
}
|
|
}
|
|
|
|
pkgDesc := &schema.PackageDescriptor{
|
|
Name: string(pluginName),
|
|
Version: pluginVersion,
|
|
DownloadURL: downloadURL,
|
|
Parameterization: parameterization,
|
|
}
|
|
|
|
pkg, err := schema.LoadPackageReferenceV2(context.TODO(), loader, pkgDesc)
|
|
if err != nil {
|
|
// If the version loaded does not match the version requested, we'll continue anyway. This matches behaviour prior
|
|
// to the introduction of parameterized packages (which confer stricter version checks), and allows some existing
|
|
// workflows, such as attaching already running instances of providers for debugging, without encountering mismatch
|
|
// errors.
|
|
//
|
|
// https://github.com/pulumi/pulumi/issues/18271 tracks the issue of whether we can do a bit better here (e.g.
|
|
// allowing semver-compatible version differences but not any others).
|
|
var versionMismatchErr *schema.PackageReferenceVersionMismatchError
|
|
if !errors.As(err, &versionMismatchErr) {
|
|
return nil, nil, fmt.Errorf("loading package '%v': %w", pkgDesc, err)
|
|
}
|
|
}
|
|
|
|
// With the package loaded, we can get the full resource schema and use that to generate an appropriate HCL2
|
|
// definition.
|
|
r, ok, err := pkg.Resources().Get(string(state.Type))
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("loading resource '%v': %w", state.Type, err)
|
|
}
|
|
if !ok {
|
|
return nil, nil, fmt.Errorf("unknown resource type '%v'", r)
|
|
}
|
|
|
|
var items []model.BodyItem
|
|
name := sanitizeName(state.URN.Name())
|
|
// Check if _this_ urn is in the name table, if so we need to set logicalName and use the mapped name for
|
|
// the resource block.
|
|
if mappedName, ok := importState.Names[state.URN]; ok {
|
|
items = append(items, &model.Attribute{
|
|
Name: "__logicalName",
|
|
Value: &model.TemplateExpression{
|
|
Parts: []model.Expression{
|
|
&model.LiteralValueExpression{
|
|
Value: cty.StringVal(state.URN.Name()),
|
|
},
|
|
},
|
|
},
|
|
})
|
|
name = sanitizeName(mappedName)
|
|
}
|
|
|
|
// keep track of a set of added references to avoid adding the same reference to the dependsOn list
|
|
// when the resource is already implicitly referenced via its properties
|
|
addedReferences := make(map[string]bool)
|
|
onReferenceFound := func(rootName string) {
|
|
addedReferences[rootName] = true
|
|
}
|
|
|
|
importStateContext := filterReferences(name, importState)
|
|
for _, p := range r.InputProperties {
|
|
input := state.Inputs[resource.PropertyKey(p.Name)]
|
|
x, err := generatePropertyValue(p, input, importStateContext, onReferenceFound)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
if x != nil {
|
|
items = append(items, &model.Attribute{
|
|
Name: p.Name,
|
|
Value: x,
|
|
})
|
|
}
|
|
}
|
|
|
|
resourceOptions, err := makeResourceOptions(state, importState.Names, addedReferences)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
if resourceOptions != nil {
|
|
items = append(items, resourceOptions)
|
|
}
|
|
|
|
typ := string(state.URN.Type())
|
|
return &model.Block{
|
|
Tokens: syntax.NewBlockTokens("resource", name, typ),
|
|
Type: "resource",
|
|
Labels: []string{name, typ},
|
|
Body: &model.Body{
|
|
Items: items,
|
|
},
|
|
}, pkgDesc, nil
|
|
}
|
|
|
|
func newVariableReference(name string) model.Expression {
|
|
return model.VariableReference(&model.Variable{
|
|
Name: name,
|
|
VariableType: model.DynamicType,
|
|
})
|
|
}
|
|
|
|
func appendResourceOption(block *model.Block, name string, value model.Expression) *model.Block {
|
|
if block == nil {
|
|
block = &model.Block{
|
|
Tokens: syntax.NewBlockTokens("options"),
|
|
Type: "options",
|
|
Body: &model.Body{},
|
|
}
|
|
}
|
|
block.Body.Items = append(block.Body.Items, &model.Attribute{
|
|
Tokens: syntax.NewAttributeTokens(name),
|
|
Name: name,
|
|
Value: value,
|
|
})
|
|
return block
|
|
}
|
|
|
|
func makeResourceOptions(state *resource.State, names NameTable, addedRefs map[string]bool) (*model.Block, error) {
|
|
var resourceOptions *model.Block
|
|
if state.Parent != "" && state.Parent.QualifiedType() != resource.RootStackType {
|
|
name, ok := names[state.Parent]
|
|
if !ok {
|
|
return nil, fmt.Errorf("no name for parent %v", state.Parent)
|
|
}
|
|
resourceOptions = appendResourceOption(resourceOptions, "parent", newVariableReference(name))
|
|
}
|
|
if state.Provider != "" {
|
|
ref, err := providers.ParseReference(state.Provider)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("invalid provider reference %v: %w", state.Provider, err)
|
|
}
|
|
if !providers.IsDefaultProvider(ref.URN()) {
|
|
name, ok := names[ref.URN()]
|
|
if !ok {
|
|
return nil, fmt.Errorf("no name for provider %v", state.Provider)
|
|
}
|
|
resourceOptions = appendResourceOption(resourceOptions, "provider", newVariableReference(name))
|
|
}
|
|
}
|
|
if len(state.Dependencies) != 0 {
|
|
deps := make([]model.Expression, 0)
|
|
for _, d := range state.Dependencies {
|
|
name, ok := names[d]
|
|
if !ok {
|
|
return nil, fmt.Errorf("no name for resource %v", d)
|
|
}
|
|
// implicitly referenced resource via their properties do not need to be added to the dependsOn list
|
|
// for example if you have a property bucket: exampleBucket.id then exampleBucket doesn't need to
|
|
// be explicitly added to the dependsOn list
|
|
if _, alreadyImplicitlyReferenced := addedRefs[name]; !alreadyImplicitlyReferenced {
|
|
deps = append(deps, newVariableReference(name))
|
|
}
|
|
}
|
|
|
|
resourceOptions = appendResourceOption(resourceOptions, "dependsOn", &model.TupleConsExpression{
|
|
Tokens: syntax.NewTupleConsTokens(len(deps)),
|
|
Expressions: deps,
|
|
})
|
|
}
|
|
if state.Protect {
|
|
resourceOptions = appendResourceOption(resourceOptions, "protect", &model.LiteralValueExpression{
|
|
Tokens: syntax.NewLiteralValueTokens(cty.True),
|
|
Value: cty.True,
|
|
})
|
|
}
|
|
return resourceOptions, nil
|
|
}
|
|
|
|
// typeRank orders types by their simplicity.
|
|
func typeRank(t schema.Type) int {
|
|
switch t {
|
|
case schema.BoolType:
|
|
return 1
|
|
case schema.IntType:
|
|
return 2
|
|
case schema.NumberType:
|
|
return 3
|
|
case schema.StringType:
|
|
return 4
|
|
case schema.AssetType:
|
|
return 5
|
|
case schema.ArchiveType:
|
|
return 6
|
|
case schema.JSONType:
|
|
return 7
|
|
case schema.AnyType:
|
|
return 13
|
|
default:
|
|
switch t := t.(type) {
|
|
case *schema.TokenType:
|
|
return 8
|
|
case *schema.ArrayType:
|
|
return 9
|
|
case *schema.MapType:
|
|
return 10
|
|
case *schema.ObjectType:
|
|
return 11
|
|
case *schema.UnionType:
|
|
return 12
|
|
case *schema.InputType:
|
|
return typeRank(t.ElementType)
|
|
case *schema.OptionalType:
|
|
return typeRank(t.ElementType)
|
|
default:
|
|
return int(math.MaxInt32)
|
|
}
|
|
}
|
|
}
|
|
|
|
// simplerType returns true if T is simpler than U.
|
|
//
|
|
// The first-order ranking is:
|
|
//
|
|
// bool < int < number < string < archive < asset < json < token < array < map < object < union < any
|
|
//
|
|
// Additional rules apply to composite types of the same kind:
|
|
// - array(T) is simpler than array(U) if T is simpler than U
|
|
// - map(T) is simpler than map(U) if T is simpler than U
|
|
// - object({ ... }) is simpler than object({ ... }) if the former has a greater number of required properties that
|
|
// are simpler than the latter's required properties
|
|
// - union(...) is simpler than union(...) if the former's simplest element type is simpler than the latter's simplest
|
|
// element type
|
|
func simplerType(t, u schema.Type) bool {
|
|
tRank, uRank := typeRank(t), typeRank(u)
|
|
if tRank < uRank {
|
|
return true
|
|
} else if tRank > uRank {
|
|
return false
|
|
}
|
|
|
|
t, u = codegen.UnwrapType(t), codegen.UnwrapType(u)
|
|
|
|
// At this point we know that t and u have the same concrete type.
|
|
switch t := t.(type) {
|
|
case *schema.TokenType:
|
|
u := u.(*schema.TokenType)
|
|
if t.UnderlyingType != nil && u.UnderlyingType != nil {
|
|
return simplerType(t.UnderlyingType, u.UnderlyingType)
|
|
}
|
|
return false
|
|
case *schema.ArrayType:
|
|
return simplerType(t.ElementType, u.(*schema.ArrayType).ElementType)
|
|
case *schema.MapType:
|
|
return simplerType(t.ElementType, u.(*schema.MapType).ElementType)
|
|
case *schema.ObjectType:
|
|
// Count how many of T's required properties are simpler than U's required properties and vice versa.
|
|
uu := u.(*schema.ObjectType)
|
|
tscore, nt, uscore := 0, 0, 0
|
|
for _, p := range t.Properties {
|
|
if p.IsRequired() {
|
|
nt++
|
|
for _, q := range uu.Properties {
|
|
if q.IsRequired() {
|
|
if simplerType(p.Type, q.Type) {
|
|
tscore++
|
|
}
|
|
if simplerType(q.Type, p.Type) {
|
|
uscore++
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// If the number of T's required properties that are simpler that U's required properties exceeds the number
|
|
// of U's required properties that are simpler than T's required properties, T is simpler.
|
|
if tscore > uscore {
|
|
return true
|
|
}
|
|
if tscore < uscore {
|
|
return false
|
|
}
|
|
|
|
// If the above counts are equal, T is simpler if it has fewer required properties.
|
|
nu := 0
|
|
for _, q := range uu.Properties {
|
|
if q.IsRequired() {
|
|
nu++
|
|
}
|
|
}
|
|
|
|
return nt < nu
|
|
case *schema.UnionType:
|
|
// Pick whichever has the simplest element type.
|
|
var simplestElementType schema.Type
|
|
for _, u := range u.(*schema.UnionType).ElementTypes {
|
|
if simplestElementType == nil || simplerType(u, simplestElementType) {
|
|
simplestElementType = u
|
|
}
|
|
}
|
|
for _, t := range t.ElementTypes {
|
|
if simplestElementType == nil || simplerType(t, simplestElementType) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
// zeroValue constructs a zero value of the given type.
|
|
func zeroValue(t schema.Type) model.Expression {
|
|
emptyImportState := ImportState{}
|
|
onReferenceAdded := func(string) {}
|
|
switch t := t.(type) {
|
|
case *schema.OptionalType:
|
|
return model.VariableReference(Null)
|
|
case *schema.InputType:
|
|
return zeroValue(t.ElementType)
|
|
case *schema.MapType:
|
|
return &model.ObjectConsExpression{}
|
|
case *schema.ArrayType:
|
|
return &model.TupleConsExpression{}
|
|
case *schema.UnionType:
|
|
// If there is a default type, create a value of that type.
|
|
if t.DefaultType != nil {
|
|
return zeroValue(t.DefaultType)
|
|
}
|
|
// Otherwise, pick the simplest type in the list.
|
|
var simplestType schema.Type
|
|
for _, t := range t.ElementTypes {
|
|
if simplestType == nil || simplerType(t, simplestType) {
|
|
simplestType = t
|
|
}
|
|
}
|
|
return zeroValue(simplestType)
|
|
case *schema.ObjectType:
|
|
var items []model.ObjectConsItem
|
|
for _, p := range t.Properties {
|
|
if p.IsRequired() {
|
|
items = append(items, model.ObjectConsItem{
|
|
Key: &model.LiteralValueExpression{
|
|
Value: cty.StringVal(p.Name),
|
|
},
|
|
Value: zeroValue(p.Type),
|
|
})
|
|
}
|
|
}
|
|
return &model.ObjectConsExpression{Items: items}
|
|
case *schema.TokenType:
|
|
if t.UnderlyingType != nil {
|
|
return zeroValue(t.UnderlyingType)
|
|
}
|
|
return model.VariableReference(Null)
|
|
}
|
|
switch t {
|
|
case schema.BoolType:
|
|
x, err := generateValue(t, resource.NewBoolProperty(false), emptyImportState, onReferenceAdded)
|
|
contract.IgnoreError(err)
|
|
return x
|
|
case schema.IntType, schema.NumberType:
|
|
x, err := generateValue(t, resource.NewNumberProperty(0), emptyImportState, onReferenceAdded)
|
|
contract.IgnoreError(err)
|
|
return x
|
|
case schema.StringType:
|
|
x, err := generateValue(t, resource.NewStringProperty(""), emptyImportState, onReferenceAdded)
|
|
contract.IgnoreError(err)
|
|
return x
|
|
case schema.ArchiveType, schema.AssetType:
|
|
return model.VariableReference(Null)
|
|
case schema.JSONType, schema.AnyType:
|
|
return &model.ObjectConsExpression{}
|
|
default:
|
|
contract.Failf("unexpected schema type %v", t)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// generatePropertyValue generates the value for the given property. If the value is absent and the property is
|
|
// required, a zero value for the property's type is generated. If the value is absent and the property is not
|
|
// required, no value is generated (i.e. this function returns nil).
|
|
func generatePropertyValue(
|
|
property *schema.Property,
|
|
value resource.PropertyValue,
|
|
importState ImportState,
|
|
onReferenceFound func(string),
|
|
) (model.Expression, error) {
|
|
if !value.HasValue() {
|
|
if !property.IsRequired() {
|
|
return nil, nil
|
|
}
|
|
return zeroValue(property.Type), nil
|
|
}
|
|
|
|
return generateValue(property.Type, value, importState, onReferenceFound)
|
|
}
|
|
|
|
// valueStructurallyTypedAs returns true if the given value is structurally typed as the given schema type.
|
|
func valueStructurallyTypedAs(value resource.PropertyValue, schemaType schema.Type) bool {
|
|
if union, ok := schemaType.(*schema.UnionType); ok {
|
|
schemaType = reduceUnionType(union, value)
|
|
}
|
|
|
|
switch {
|
|
case value.IsObject():
|
|
switch arg := schemaType.(type) {
|
|
case *schema.ObjectType:
|
|
schemaProperties := make(map[string]schema.Type)
|
|
for _, schemaProperty := range arg.Properties {
|
|
schemaProperties[schemaProperty.Name] = schemaProperty.Type
|
|
}
|
|
|
|
objectProperties := value.ObjectValue()
|
|
// check that each property is present in the schema and that the value is structurally typed as well
|
|
for propertyKey, propertyValue := range objectProperties {
|
|
propertyValueSchema, ok := arg.Property(string(propertyKey))
|
|
if !ok {
|
|
// unknown property
|
|
return false
|
|
}
|
|
|
|
if !valueStructurallyTypedAs(propertyValue, propertyValueSchema.Type) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
// check that all required properties from the schema are present in the object properties
|
|
for _, schemaProperty := range arg.Properties {
|
|
if schemaProperty.IsRequired() {
|
|
if _, ok := objectProperties[resource.PropertyKey(schemaProperty.Name)]; !ok {
|
|
// the required property was not present in the object
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
// all properties are present and structurally typed
|
|
return true
|
|
|
|
case *schema.UnionType:
|
|
// make sure that at least of the union element types is structurally typed
|
|
for _, unionElement := range arg.ElementTypes {
|
|
if valueStructurallyTypedAs(value, unionElement) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
case value.IsString():
|
|
// basic case
|
|
if schemaType == schema.StringType {
|
|
return true
|
|
}
|
|
|
|
// for unions: check that at least of one of the element types is also a string
|
|
// collapsing unions of unions as necessary recursively
|
|
if union, ok := schemaType.(*schema.UnionType); ok {
|
|
for _, elementType := range union.ElementTypes {
|
|
if valueStructurallyTypedAs(value, elementType) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
case value.IsBool():
|
|
// basic case
|
|
if schemaType == schema.BoolType {
|
|
return true
|
|
}
|
|
|
|
// for unions: check that at least of one of the element types is also a bool
|
|
// collapsing unions of unions as necessary recursively
|
|
if union, ok := schemaType.(*schema.UnionType); ok {
|
|
for _, elementType := range union.ElementTypes {
|
|
if valueStructurallyTypedAs(value, elementType) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
case value.IsNumber():
|
|
// basic case
|
|
if schemaType == schema.NumberType || schemaType == schema.IntType {
|
|
return true
|
|
}
|
|
|
|
// for unions: check that at least of one of the element types is also a number
|
|
// collapsing unions of unions as necessary recursively
|
|
if union, ok := schemaType.(*schema.UnionType); ok {
|
|
for _, elementType := range union.ElementTypes {
|
|
if valueStructurallyTypedAs(value, elementType) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
case value.IsArray():
|
|
// basic case: check that each element in the array is structurally typed as the element type of the schenma array
|
|
switch arg := schemaType.(type) {
|
|
case *schema.ArrayType:
|
|
for _, element := range value.ArrayValue() {
|
|
if !valueStructurallyTypedAs(element, arg.ElementType) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
// all elements are structurally typed
|
|
return true
|
|
case *schema.UnionType:
|
|
// make sure that at least of the union element types is structurally typed
|
|
for _, unionElement := range arg.ElementTypes {
|
|
if valueStructurallyTypedAs(value, unionElement) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// reduceUnionType reduces the given union type to a simpler type that potentially matches the value.
|
|
// When the value type is primitive, choose the first element type of the union elements that is of the same type.
|
|
// When the value is an object, use the discriminator to choose the element type.
|
|
func reduceUnionType(schemaUnion *schema.UnionType, value resource.PropertyValue) schema.Type {
|
|
switch {
|
|
case value.IsObject():
|
|
// return the first element type that matches structurally fits the value
|
|
findBestFitType := func() schema.Type {
|
|
for _, t := range schemaUnion.ElementTypes {
|
|
if valueStructurallyTypedAs(value, t) {
|
|
return t
|
|
}
|
|
}
|
|
|
|
// if we still couldn't find a type that fits the value
|
|
return nil
|
|
}
|
|
|
|
// If the value is an object, use the discriminator to choose the element type.
|
|
if schemaUnion.Discriminator == "" {
|
|
return findBestFitType()
|
|
}
|
|
|
|
obj := value.ObjectValue()
|
|
discriminatorValue, ok := obj[resource.PropertyKey(schemaUnion.Discriminator)]
|
|
if !ok {
|
|
// discriminator property is not present
|
|
// return the first type that fits the value
|
|
return findBestFitType()
|
|
}
|
|
|
|
if !discriminatorValue.IsString() {
|
|
// discriminator property value is not a string,
|
|
// so we can't select a type from the union mapping
|
|
return findBestFitType()
|
|
}
|
|
|
|
correspondingTypeToken, ok := schemaUnion.Mapping[discriminatorValue.StringValue()]
|
|
if !ok {
|
|
// discriminator property value is not a key in the union mapping,
|
|
return findBestFitType()
|
|
}
|
|
|
|
for _, elementType := range schemaUnion.ElementTypes {
|
|
// found the type token
|
|
// match it against the element type which should be an object
|
|
elementTypeObject, ok := codegen.UnwrapType(elementType).(*schema.ObjectType)
|
|
if ok {
|
|
elementTypeToken, parseError := tokens.ParseTypeToken(elementTypeObject.Token)
|
|
if parseError != nil {
|
|
continue
|
|
}
|
|
|
|
foundTypeToken, parseError := tokens.ParseTypeToken(correspondingTypeToken)
|
|
|
|
if parseError != nil {
|
|
continue
|
|
}
|
|
|
|
typeName := string(elementTypeToken.Name())
|
|
foundTypeName := string(foundTypeToken.Name())
|
|
if typeName == foundTypeName {
|
|
return elementTypeObject
|
|
}
|
|
}
|
|
}
|
|
|
|
default:
|
|
for _, t := range schemaUnion.ElementTypes {
|
|
if unionType, ok := t.(*schema.UnionType); ok {
|
|
t = reduceUnionType(unionType, value)
|
|
}
|
|
|
|
if valueStructurallyTypedAs(value, t) {
|
|
return t
|
|
}
|
|
}
|
|
}
|
|
|
|
// anything else, we don't know
|
|
return nil
|
|
}
|
|
|
|
// generateValue generates a value from the given property value. The given type may or may not match the shape of the
|
|
// given value.
|
|
func generateValue(
|
|
typ schema.Type,
|
|
value resource.PropertyValue,
|
|
importState ImportState,
|
|
onReferenceFound func(string),
|
|
) (model.Expression, error) {
|
|
typ = codegen.UnwrapType(typ)
|
|
|
|
if unionType, ok := typ.(*schema.UnionType); ok {
|
|
typ = reduceUnionType(unionType, value)
|
|
}
|
|
|
|
switch {
|
|
case value.IsArchive():
|
|
return nil, errors.New("NYI: archives")
|
|
case value.IsArray():
|
|
elementType := schema.AnyType
|
|
if typ, ok := typ.(*schema.ArrayType); ok {
|
|
elementType = typ.ElementType
|
|
}
|
|
|
|
arr := value.ArrayValue()
|
|
exprs := make([]model.Expression, len(arr))
|
|
for i, v := range arr {
|
|
x, err := generateValue(elementType, v, importState, onReferenceFound)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
exprs[i] = x
|
|
}
|
|
return &model.TupleConsExpression{
|
|
Tokens: syntax.NewTupleConsTokens(len(exprs)),
|
|
Expressions: exprs,
|
|
}, nil
|
|
case value.IsAsset():
|
|
return nil, errors.New("NYI: assets")
|
|
case value.IsBool():
|
|
return &model.LiteralValueExpression{
|
|
Value: cty.BoolVal(value.BoolValue()),
|
|
}, nil
|
|
case value.IsComputed() || value.IsOutput():
|
|
return nil, errors.New("cannot define computed values")
|
|
case value.IsNull():
|
|
return model.VariableReference(Null), nil
|
|
case value.IsNumber():
|
|
return &model.LiteralValueExpression{
|
|
Value: cty.NumberFloatVal(value.NumberValue()),
|
|
}, nil
|
|
case value.IsObject():
|
|
obj := value.ObjectValue()
|
|
items := slice.Prealloc[model.ObjectConsItem](len(obj))
|
|
|
|
switch arg := typ.(type) {
|
|
case *schema.ObjectType:
|
|
for _, p := range arg.Properties {
|
|
x, err := generatePropertyValue(p, obj[resource.PropertyKey(p.Name)], importState, onReferenceFound)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if x != nil {
|
|
items = append(items, model.ObjectConsItem{
|
|
Key: &model.LiteralValueExpression{
|
|
Value: cty.StringVal(p.Name),
|
|
},
|
|
Value: x,
|
|
})
|
|
}
|
|
}
|
|
|
|
default:
|
|
elementType := schema.AnyType
|
|
if mapType, ok := typ.(*schema.MapType); ok {
|
|
elementType = mapType.ElementType
|
|
}
|
|
|
|
for _, k := range obj.StableKeys() {
|
|
// Ignore internal properties.
|
|
if strings.HasPrefix(string(k), "__") {
|
|
continue
|
|
}
|
|
|
|
x, err := generateValue(elementType, obj[k], importState, onReferenceFound)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Always quote the key in case it includes invalid identifier characters (like '/' or ':')
|
|
propKey := fmt.Sprintf("%q", string(k))
|
|
|
|
items = append(items, model.ObjectConsItem{
|
|
Key: &model.LiteralValueExpression{
|
|
Value: cty.StringVal(propKey),
|
|
},
|
|
Value: x,
|
|
})
|
|
}
|
|
}
|
|
return &model.ObjectConsExpression{
|
|
Tokens: syntax.NewObjectConsTokens(len(items)),
|
|
Items: items,
|
|
}, nil
|
|
case value.IsSecret():
|
|
arg, err := generateValue(typ, value.SecretValue().Element, importState, onReferenceFound)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &model.FunctionCallExpression{
|
|
Name: "secret",
|
|
Signature: model.StaticFunctionSignature{
|
|
Parameters: []model.Parameter{{
|
|
Name: "value",
|
|
Type: arg.Type(),
|
|
}},
|
|
ReturnType: model.NewOutputType(arg.Type()),
|
|
},
|
|
Args: []model.Expression{arg},
|
|
}, nil
|
|
case value.IsString():
|
|
x := &model.TemplateExpression{
|
|
Parts: []model.Expression{
|
|
&model.LiteralValueExpression{
|
|
Value: cty.StringVal(value.StringValue()),
|
|
},
|
|
},
|
|
}
|
|
switch typ {
|
|
case schema.ArchiveType:
|
|
return &model.FunctionCallExpression{
|
|
Name: "fileArchive",
|
|
Args: []model.Expression{x},
|
|
}, nil
|
|
case schema.AssetType:
|
|
return &model.FunctionCallExpression{
|
|
Name: "fileAsset",
|
|
Args: []model.Expression{x},
|
|
}, nil
|
|
default:
|
|
for _, pathedValue := range importState.PathedLiteralValues {
|
|
if pathedValue.Value == value.StringValue() {
|
|
onReferenceFound(pathedValue.Root)
|
|
return pathedValue.ExpressionReference, nil
|
|
}
|
|
}
|
|
|
|
return x, nil
|
|
}
|
|
default:
|
|
contract.Failf("unexpected property value %v", value)
|
|
return nil, nil
|
|
}
|
|
}
|