2020-03-18 17:28:57 +01:00
|
|
|
// 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 model
|
|
|
|
|
|
|
|
type typeTransform int
|
|
|
|
|
|
|
|
var (
|
|
|
|
makeIdentity = typeTransform(0)
|
|
|
|
makePromise = typeTransform(1)
|
|
|
|
makeOutput = typeTransform(2)
|
|
|
|
)
|
|
|
|
|
|
|
|
func (f typeTransform) do(t Type) Type {
|
|
|
|
switch f {
|
|
|
|
case makePromise:
|
|
|
|
return NewPromiseType(t)
|
|
|
|
case makeOutput:
|
|
|
|
return NewOutputType(t)
|
|
|
|
default:
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveEventuals(t Type, resolveOutputs bool) (Type, typeTransform) {
|
2020-09-19 02:29:21 +02:00
|
|
|
return resolveEventualsImpl(t, resolveOutputs, map[Type]Type{})
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveEventualsImpl(t Type, resolveOutputs bool, seen map[Type]Type) (Type, typeTransform) {
|
2020-03-18 17:28:57 +01:00
|
|
|
switch t := t.(type) {
|
|
|
|
case *OutputType:
|
|
|
|
if resolveOutputs {
|
|
|
|
return t.ElementType, makeOutput
|
|
|
|
}
|
|
|
|
return t, makeIdentity
|
|
|
|
case *PromiseType:
|
2020-09-19 02:29:21 +02:00
|
|
|
element, transform := resolveEventualsImpl(t.ElementType, resolveOutputs, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
if makePromise > transform {
|
|
|
|
transform = makePromise
|
|
|
|
}
|
|
|
|
return element, transform
|
|
|
|
case *MapType:
|
2020-09-19 02:29:21 +02:00
|
|
|
resolved, transform := resolveEventualsImpl(t.ElementType, resolveOutputs, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
return NewMapType(resolved), transform
|
|
|
|
case *ListType:
|
2020-09-19 02:29:21 +02:00
|
|
|
resolved, transform := resolveEventualsImpl(t.ElementType, resolveOutputs, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
return NewListType(resolved), transform
|
|
|
|
case *SetType:
|
2020-09-19 02:29:21 +02:00
|
|
|
resolved, transform := resolveEventualsImpl(t.ElementType, resolveOutputs, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
return NewSetType(resolved), transform
|
|
|
|
case *UnionType:
|
|
|
|
transform := makeIdentity
|
|
|
|
elementTypes := make([]Type, len(t.ElementTypes))
|
|
|
|
for i, t := range t.ElementTypes {
|
2020-09-19 02:29:21 +02:00
|
|
|
element, elementTransform := resolveEventualsImpl(t, resolveOutputs, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
if elementTransform > transform {
|
|
|
|
transform = elementTransform
|
|
|
|
}
|
|
|
|
elementTypes[i] = element
|
|
|
|
}
|
2021-07-16 18:56:26 +02:00
|
|
|
return NewUnionTypeAnnotated(elementTypes, t.Annotations...), transform
|
2020-03-18 17:28:57 +01:00
|
|
|
case *ObjectType:
|
|
|
|
transform := makeIdentity
|
2020-09-19 02:29:21 +02:00
|
|
|
if already, ok := seen[t]; ok {
|
|
|
|
return already, transform
|
|
|
|
}
|
2020-03-18 17:28:57 +01:00
|
|
|
properties := map[string]Type{}
|
2020-09-19 02:29:21 +02:00
|
|
|
objType := NewObjectType(properties, t.Annotations...)
|
|
|
|
seen[t] = objType
|
2020-03-18 17:28:57 +01:00
|
|
|
for k, t := range t.Properties {
|
2020-09-19 02:29:21 +02:00
|
|
|
property, propertyTransform := resolveEventualsImpl(t, resolveOutputs, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
if propertyTransform > transform {
|
|
|
|
transform = propertyTransform
|
|
|
|
}
|
|
|
|
properties[k] = property
|
|
|
|
}
|
2020-09-19 02:29:21 +02:00
|
|
|
return objType, transform
|
2020-03-18 17:28:57 +01:00
|
|
|
case *TupleType:
|
|
|
|
transform := makeIdentity
|
|
|
|
elements := make([]Type, len(t.ElementTypes))
|
|
|
|
for i, t := range t.ElementTypes {
|
2020-09-19 02:29:21 +02:00
|
|
|
element, elementTransform := resolveEventualsImpl(t, resolveOutputs, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
if elementTransform > transform {
|
|
|
|
transform = elementTransform
|
|
|
|
}
|
|
|
|
elements[i] = element
|
|
|
|
}
|
|
|
|
return NewTupleType(elements...), transform
|
|
|
|
default:
|
|
|
|
return t, makeIdentity
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResolveOutputs recursively replaces all output(T) and promise(T) types in the input type with their element type.
|
|
|
|
func ResolveOutputs(t Type) Type {
|
2020-05-22 11:47:34 +02:00
|
|
|
containsOutputs, containsPromises := ContainsEventuals(t)
|
|
|
|
if !containsOutputs && !containsPromises {
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
2020-03-18 17:28:57 +01:00
|
|
|
resolved, _ := resolveEventuals(t, true)
|
|
|
|
return resolved
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResolvePromises recursively replaces all promise(T) types in the input type with their element type.
|
|
|
|
func ResolvePromises(t Type) Type {
|
2020-05-22 11:47:34 +02:00
|
|
|
if !ContainsPromises(t) {
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
2020-03-18 17:28:57 +01:00
|
|
|
resolved, _ := resolveEventuals(t, false)
|
|
|
|
return resolved
|
|
|
|
}
|
|
|
|
|
2020-05-22 11:47:34 +02:00
|
|
|
// ContainsEventuals returns true if the input type contains output or promise types.
|
|
|
|
func ContainsEventuals(t Type) (containsOutputs, containsPromises bool) {
|
2020-09-19 02:29:21 +02:00
|
|
|
return containsEventualsImpl(t, map[Type]struct{}{})
|
|
|
|
}
|
|
|
|
|
|
|
|
func containsEventualsImpl(t Type, seen map[Type]struct{}) (containsOutputs, containsPromises bool) {
|
|
|
|
if _, ok := seen[t]; ok {
|
|
|
|
return false, false
|
|
|
|
}
|
|
|
|
seen[t] = struct{}{}
|
|
|
|
|
2020-05-22 11:47:34 +02:00
|
|
|
switch t := t.(type) {
|
|
|
|
case *OutputType:
|
|
|
|
return true, false
|
|
|
|
case *PromiseType:
|
|
|
|
return ContainsOutputs(t.ElementType), true
|
|
|
|
case *MapType:
|
2020-09-19 02:29:21 +02:00
|
|
|
return containsEventualsImpl(t.ElementType, seen)
|
2020-05-22 11:47:34 +02:00
|
|
|
case *ListType:
|
2020-09-19 02:29:21 +02:00
|
|
|
return containsEventualsImpl(t.ElementType, seen)
|
2020-05-22 11:47:34 +02:00
|
|
|
case *SetType:
|
2020-09-19 02:29:21 +02:00
|
|
|
return containsEventualsImpl(t.ElementType, seen)
|
2020-05-22 11:47:34 +02:00
|
|
|
case *UnionType:
|
|
|
|
for _, t := range t.ElementTypes {
|
2020-09-19 02:29:21 +02:00
|
|
|
outputs, promises := containsEventualsImpl(t, seen)
|
2020-05-22 11:47:34 +02:00
|
|
|
containsOutputs = outputs || containsOutputs
|
|
|
|
containsPromises = promises || containsPromises
|
|
|
|
}
|
|
|
|
return
|
|
|
|
case *ObjectType:
|
|
|
|
for _, t := range t.Properties {
|
2020-09-19 02:29:21 +02:00
|
|
|
outputs, promises := containsEventualsImpl(t, seen)
|
2020-05-22 11:47:34 +02:00
|
|
|
containsOutputs = outputs || containsOutputs
|
|
|
|
containsPromises = promises || containsPromises
|
|
|
|
}
|
|
|
|
return
|
|
|
|
case *TupleType:
|
|
|
|
for _, t := range t.ElementTypes {
|
2020-09-19 02:29:21 +02:00
|
|
|
outputs, promises := containsEventualsImpl(t, seen)
|
2020-05-22 11:47:34 +02:00
|
|
|
containsOutputs = outputs || containsOutputs
|
|
|
|
containsPromises = promises || containsPromises
|
|
|
|
}
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
return false, false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ContainsOutputs returns true if the input type contains output types.
|
|
|
|
func ContainsOutputs(t Type) bool {
|
|
|
|
switch t := t.(type) {
|
|
|
|
case *OutputType:
|
|
|
|
return true
|
|
|
|
case *PromiseType:
|
|
|
|
return ContainsOutputs(t.ElementType)
|
|
|
|
case *MapType:
|
|
|
|
return ContainsOutputs(t.ElementType)
|
|
|
|
case *ListType:
|
|
|
|
return ContainsOutputs(t.ElementType)
|
|
|
|
case *SetType:
|
|
|
|
return ContainsOutputs(t.ElementType)
|
|
|
|
case *UnionType:
|
|
|
|
for _, t := range t.ElementTypes {
|
|
|
|
if ContainsOutputs(t) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
case *ObjectType:
|
|
|
|
for _, t := range t.Properties {
|
|
|
|
if ContainsOutputs(t) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
case *TupleType:
|
|
|
|
for _, t := range t.ElementTypes {
|
|
|
|
if ContainsOutputs(t) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ContainsPromises returns true if the input type contains promise types.
|
|
|
|
func ContainsPromises(t Type) bool {
|
|
|
|
switch t := t.(type) {
|
|
|
|
case *PromiseType:
|
|
|
|
return true
|
|
|
|
case *MapType:
|
|
|
|
return ContainsPromises(t.ElementType)
|
|
|
|
case *ListType:
|
|
|
|
return ContainsPromises(t.ElementType)
|
|
|
|
case *SetType:
|
|
|
|
return ContainsPromises(t.ElementType)
|
|
|
|
case *UnionType:
|
|
|
|
for _, t := range t.ElementTypes {
|
|
|
|
if ContainsPromises(t) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
case *ObjectType:
|
|
|
|
for _, t := range t.Properties {
|
|
|
|
if ContainsPromises(t) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
case *TupleType:
|
|
|
|
for _, t := range t.ElementTypes {
|
|
|
|
if ContainsPromises(t) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-18 17:28:57 +01:00
|
|
|
func liftOperationType(resultType Type, arguments ...Expression) Type {
|
|
|
|
var transform typeTransform
|
|
|
|
for _, arg := range arguments {
|
|
|
|
_, t := resolveEventuals(arg.Type(), true)
|
|
|
|
if t > transform {
|
|
|
|
transform = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return transform.do(resultType)
|
|
|
|
}
|
|
|
|
|
|
|
|
// InputType returns the result of replacing each type in T with union(T, output(T)).
|
|
|
|
func InputType(t Type) Type {
|
2020-09-19 02:29:21 +02:00
|
|
|
return inputTypeImpl(t, map[Type]Type{})
|
|
|
|
}
|
|
|
|
func inputTypeImpl(t Type, seen map[Type]Type) Type {
|
2020-03-18 17:28:57 +01:00
|
|
|
if t == DynamicType || t == NoneType {
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
|
|
|
var src Type
|
|
|
|
switch t := t.(type) {
|
|
|
|
case *OutputType:
|
|
|
|
return t
|
|
|
|
case *PromiseType:
|
2020-09-19 02:29:21 +02:00
|
|
|
src = NewPromiseType(inputTypeImpl(t.ElementType, seen))
|
2020-03-18 17:28:57 +01:00
|
|
|
case *MapType:
|
2020-09-19 02:29:21 +02:00
|
|
|
src = NewMapType(inputTypeImpl(t.ElementType, seen))
|
2020-03-18 17:28:57 +01:00
|
|
|
case *ListType:
|
2020-09-19 02:29:21 +02:00
|
|
|
src = NewListType(inputTypeImpl(t.ElementType, seen))
|
2020-03-18 17:28:57 +01:00
|
|
|
case *UnionType:
|
|
|
|
elementTypes := make([]Type, len(t.ElementTypes))
|
|
|
|
for i, t := range t.ElementTypes {
|
2020-09-19 02:29:21 +02:00
|
|
|
elementTypes[i] = inputTypeImpl(t, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
}
|
2021-04-16 14:04:42 +02:00
|
|
|
src = NewUnionTypeAnnotated(elementTypes, t.Annotations...)
|
2020-03-18 17:28:57 +01:00
|
|
|
case *ObjectType:
|
2021-07-16 18:56:26 +02:00
|
|
|
if already, ok := seen[t]; ok {
|
|
|
|
return already
|
|
|
|
}
|
|
|
|
|
2020-03-18 17:28:57 +01:00
|
|
|
properties := map[string]Type{}
|
2020-09-19 02:29:21 +02:00
|
|
|
src = NewObjectType(properties, t.Annotations...)
|
|
|
|
seen[t] = src
|
2020-03-18 17:28:57 +01:00
|
|
|
for k, t := range t.Properties {
|
2020-09-19 02:29:21 +02:00
|
|
|
properties[k] = inputTypeImpl(t, seen)
|
2020-03-18 17:28:57 +01:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
src = t
|
|
|
|
}
|
|
|
|
|
2020-05-22 11:47:34 +02:00
|
|
|
return NewUnionType(src, NewOutputType(src))
|
2020-03-18 17:28:57 +01:00
|
|
|
}
|