mirror of https://github.com/pulumi/pulumi.git
2654 lines
75 KiB
Go
2654 lines
75 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 model
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"math/big"
|
|
"strconv"
|
|
|
|
"github.com/hashicorp/hcl/v2"
|
|
"github.com/hashicorp/hcl/v2/hclsyntax"
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/hcl2/syntax"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
"github.com/zclconf/go-cty/cty"
|
|
"github.com/zclconf/go-cty/cty/convert"
|
|
)
|
|
|
|
// Expression represents a semantically-analyzed HCL2 expression.
|
|
type Expression interface {
|
|
printable
|
|
|
|
// SyntaxNode returns the hclsyntax.Node associated with the expression.
|
|
SyntaxNode() hclsyntax.Node
|
|
// NodeTokens returns the syntax.Tokens associated with the expression.
|
|
NodeTokens() syntax.NodeTokens
|
|
|
|
// SetLeadingTrivia sets the leading trivia associated with the expression.
|
|
SetLeadingTrivia(syntax.TriviaList)
|
|
// SetTrailingTrivia sets the trailing trivia associated with the expression.
|
|
SetTrailingTrivia(syntax.TriviaList)
|
|
|
|
// Type returns the type of the expression.
|
|
Type() Type
|
|
// Typecheck recomputes the type of the expression, optionally typechecking its operands first.
|
|
Typecheck(typecheckOperands bool) hcl.Diagnostics
|
|
|
|
// Evaluate evaluates the expression.
|
|
Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics)
|
|
|
|
isExpression()
|
|
}
|
|
|
|
func identToken(token syntax.Token, ident string) syntax.Token {
|
|
if string(token.Raw.Bytes) != ident {
|
|
token.Raw.Bytes = []byte(ident)
|
|
}
|
|
return token
|
|
}
|
|
|
|
func exprHasLeadingTrivia(parens syntax.Parentheses, first interface{}) bool {
|
|
if parens.Any() {
|
|
return true
|
|
}
|
|
switch first := first.(type) {
|
|
case Expression:
|
|
return first.HasLeadingTrivia()
|
|
case bool:
|
|
return first
|
|
default:
|
|
contract.Failf("unexpected value of type %T for first", first)
|
|
return false
|
|
}
|
|
}
|
|
|
|
func exprHasTrailingTrivia(parens syntax.Parentheses, last interface{}) bool {
|
|
if parens.Any() {
|
|
return true
|
|
}
|
|
switch last := last.(type) {
|
|
case Expression:
|
|
return last.HasTrailingTrivia()
|
|
case bool:
|
|
return last
|
|
default:
|
|
contract.Failf("unexpected value of type %T for last", last)
|
|
return false
|
|
}
|
|
}
|
|
|
|
func getExprLeadingTrivia(parens syntax.Parentheses, first interface{}) syntax.TriviaList {
|
|
if parens.Any() {
|
|
return parens.GetLeadingTrivia()
|
|
}
|
|
switch first := first.(type) {
|
|
case Expression:
|
|
return first.GetLeadingTrivia()
|
|
case syntax.Token:
|
|
return first.LeadingTrivia
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func setExprLeadingTrivia(parens syntax.Parentheses, first interface{}, trivia syntax.TriviaList) {
|
|
if parens.Any() {
|
|
parens.SetLeadingTrivia(trivia)
|
|
return
|
|
}
|
|
switch first := first.(type) {
|
|
case Expression:
|
|
first.SetLeadingTrivia(trivia)
|
|
case *syntax.Token:
|
|
first.LeadingTrivia = trivia
|
|
}
|
|
}
|
|
|
|
func getExprTrailingTrivia(parens syntax.Parentheses, last interface{}) syntax.TriviaList {
|
|
if parens.Any() {
|
|
return parens.GetTrailingTrivia()
|
|
}
|
|
switch last := last.(type) {
|
|
case Expression:
|
|
return last.GetTrailingTrivia()
|
|
case syntax.Token:
|
|
return last.TrailingTrivia
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func setExprTrailingTrivia(parens syntax.Parentheses, last interface{}, trivia syntax.TriviaList) {
|
|
if parens.Any() {
|
|
parens.SetTrailingTrivia(trivia)
|
|
return
|
|
}
|
|
switch last := last.(type) {
|
|
case Expression:
|
|
last.SetTrailingTrivia(trivia)
|
|
case *syntax.Token:
|
|
last.TrailingTrivia = trivia
|
|
}
|
|
}
|
|
|
|
type syntaxExpr struct {
|
|
hclsyntax.LiteralValueExpr
|
|
|
|
expr Expression
|
|
}
|
|
|
|
func (x syntaxExpr) Value(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
return x.expr.Evaluate(context)
|
|
}
|
|
|
|
type hclExpression struct {
|
|
x Expression
|
|
}
|
|
|
|
func HCLExpression(x Expression) hcl.Expression {
|
|
return hclExpression{x: x}
|
|
}
|
|
|
|
func (x hclExpression) Value(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
return x.x.Evaluate(context)
|
|
}
|
|
|
|
func (x hclExpression) Variables() []hcl.Traversal {
|
|
var variables []hcl.Traversal
|
|
scope := NewRootScope(syntax.None)
|
|
_, diags := VisitExpression(x.x, func(n Expression) (Expression, hcl.Diagnostics) {
|
|
switch n := n.(type) {
|
|
case *AnonymousFunctionExpression:
|
|
scope = scope.Push(syntax.None)
|
|
for _, p := range n.Parameters {
|
|
scope.Define(p.Name, p)
|
|
}
|
|
case *ForExpression:
|
|
scope = scope.Push(syntax.None)
|
|
if n.KeyVariable != nil {
|
|
scope.Define(n.KeyVariable.Name, n.KeyVariable)
|
|
}
|
|
scope.Define(n.ValueVariable.Name, n.ValueVariable)
|
|
}
|
|
return n, nil
|
|
}, func(n Expression) (Expression, hcl.Diagnostics) {
|
|
switch n := n.(type) {
|
|
case *AnonymousFunctionExpression, *ForExpression:
|
|
scope = scope.Pop()
|
|
case *ScopeTraversalExpression:
|
|
if _, isSplatVariable := n.Parts[0].(*SplatVariable); !isSplatVariable {
|
|
if _, defined := scope.BindReference(n.RootName); !defined {
|
|
variables = append(variables, n.Traversal.SimpleSplit().Abs)
|
|
}
|
|
}
|
|
}
|
|
return n, nil
|
|
})
|
|
contract.Assertf(len(diags) == 0, "unexpected diagnostics: %v", diags)
|
|
return variables
|
|
}
|
|
|
|
func (x hclExpression) Range() hcl.Range {
|
|
if syntax := x.x.SyntaxNode(); syntax != nil {
|
|
return syntax.Range()
|
|
}
|
|
return hcl.Range{}
|
|
}
|
|
|
|
func (x hclExpression) StartRange() hcl.Range {
|
|
if syntax := x.x.SyntaxNode(); syntax != nil {
|
|
return syntax.(hcl.Expression).StartRange()
|
|
}
|
|
return hcl.Range{}
|
|
}
|
|
|
|
func operatorPrecedence(op *hclsyntax.Operation) int {
|
|
switch op {
|
|
case hclsyntax.OpLogicalOr:
|
|
return 1
|
|
case hclsyntax.OpLogicalAnd:
|
|
return 2
|
|
case hclsyntax.OpEqual, hclsyntax.OpNotEqual:
|
|
return 3
|
|
case hclsyntax.OpGreaterThan, hclsyntax.OpGreaterThanOrEqual, hclsyntax.OpLessThan, hclsyntax.OpLessThanOrEqual:
|
|
return 4
|
|
case hclsyntax.OpAdd, hclsyntax.OpSubtract:
|
|
return 5
|
|
case hclsyntax.OpMultiply, hclsyntax.OpDivide, hclsyntax.OpModulo:
|
|
return 6
|
|
case hclsyntax.OpNegate, hclsyntax.OpLogicalNot:
|
|
return 7
|
|
default:
|
|
return 8
|
|
}
|
|
}
|
|
|
|
// AnonymousFunctionExpression represents a semantically-analyzed anonymous function expression.
|
|
//
|
|
// These expressions are not the result of semantically analyzing syntax nodes. Instead, they may be synthesized by
|
|
// transforms over the IR for a program (e.g. the Apply transform).
|
|
type AnonymousFunctionExpression struct {
|
|
// The signature for the anonymous function.
|
|
Signature StaticFunctionSignature
|
|
// The parameter definitions for the anonymous function.
|
|
Parameters []*Variable
|
|
|
|
// The body of the anonymous function.
|
|
Body Expression
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the body of the anonymous function.
|
|
func (x *AnonymousFunctionExpression) SyntaxNode() hclsyntax.Node {
|
|
return x.Body.SyntaxNode()
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the body of the anonymous function.
|
|
func (x *AnonymousFunctionExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Body.NodeTokens()
|
|
}
|
|
|
|
// Type returns the type of the anonymous function expression.
|
|
//
|
|
// TODO: currently this returns the any type. Instead, it should return a function type.
|
|
func (x *AnonymousFunctionExpression) Type() Type {
|
|
return DynamicType
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
bodyDiags := x.Body.Typecheck(true)
|
|
diagnostics = append(diagnostics, bodyDiags...)
|
|
}
|
|
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
return cty.NilVal, hcl.Diagnostics{cannotEvaluateAnonymousFunctionExpressions()}
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) HasLeadingTrivia() bool {
|
|
return x.Body.HasLeadingTrivia()
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) HasTrailingTrivia() bool {
|
|
return x.Body.HasTrailingTrivia()
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return x.Body.GetLeadingTrivia()
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
x.Body.SetLeadingTrivia(t)
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return x.Body.GetTrailingTrivia()
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
x.Body.SetTrailingTrivia(t)
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *AnonymousFunctionExpression) print(w io.Writer, p *printer) {
|
|
// Print a call to eval.
|
|
p.fprintf(w, "eval(")
|
|
|
|
// Print the parameter names.
|
|
for _, v := range x.Parameters {
|
|
p.fprintf(w, "%v, ", v.Name)
|
|
}
|
|
|
|
// Print the body and closing paren.
|
|
p.fprintf(w, "%v)", x.Body)
|
|
}
|
|
|
|
func (*AnonymousFunctionExpression) isExpression() {}
|
|
|
|
// BinaryOpExpression represents a semantically-analyzed binary operation.
|
|
type BinaryOpExpression struct {
|
|
// The syntax node associated with the binary operation.
|
|
Syntax *hclsyntax.BinaryOpExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.BinaryOpTokens
|
|
|
|
// The left-hand operand of the operation.
|
|
LeftOperand Expression
|
|
// The operation.
|
|
Operation *hclsyntax.Operation
|
|
// The right-hand operand of the operation.
|
|
RightOperand Expression
|
|
|
|
leftType Type
|
|
rightType Type
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the binary operation.
|
|
func (x *BinaryOpExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the binary operation.
|
|
func (x *BinaryOpExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// LeftOperandType returns the desired type for the left operand of the binary operation.
|
|
func (x *BinaryOpExpression) LeftOperandType() Type {
|
|
return x.leftType
|
|
}
|
|
|
|
// RightOperandType returns the desired type for the right operand of the binary operation.
|
|
func (x *BinaryOpExpression) RightOperandType() Type {
|
|
return x.rightType
|
|
}
|
|
|
|
// Type returns the type of the binary operation.
|
|
func (x *BinaryOpExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *BinaryOpExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
var rng hcl.Range
|
|
if x.Syntax != nil {
|
|
rng = x.Syntax.Range()
|
|
}
|
|
|
|
if typecheckOperands {
|
|
leftDiags := x.LeftOperand.Typecheck(true)
|
|
diagnostics = append(diagnostics, leftDiags...)
|
|
|
|
rightDiags := x.RightOperand.Typecheck(true)
|
|
diagnostics = append(diagnostics, rightDiags...)
|
|
}
|
|
|
|
// If either operand is a bigInt lift the whole expression to bigInt
|
|
big := false
|
|
if x.LeftOperand.Type().AssignableFrom(BigIntegerType) ||
|
|
x.RightOperand.Type().AssignableFrom(BigIntegerType) {
|
|
big = true
|
|
}
|
|
|
|
// Compute the signature for the operator and typecheck the arguments.
|
|
signature := getOperationSignature(x.Operation, big)
|
|
contract.Assertf(len(signature.Parameters) == 2,
|
|
"expected binary operator signature to have two parameters, got %v", len(signature.Parameters))
|
|
|
|
x.leftType = signature.Parameters[0].Type
|
|
x.rightType = signature.Parameters[1].Type
|
|
|
|
typecheckDiags := typecheckArgs(rng, signature, x.LeftOperand, x.RightOperand)
|
|
diagnostics = append(diagnostics, typecheckDiags...)
|
|
|
|
x.exprType = liftOperationType(signature.ReturnType, x.LeftOperand, x.RightOperand)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *BinaryOpExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.BinaryOpExpr{
|
|
LHS: &syntaxExpr{expr: x.LeftOperand},
|
|
Op: x.Operation,
|
|
RHS: &syntaxExpr{expr: x.RightOperand},
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *BinaryOpExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.LeftOperand)
|
|
}
|
|
|
|
func (x *BinaryOpExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.RightOperand)
|
|
}
|
|
|
|
func (x *BinaryOpExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.LeftOperand)
|
|
}
|
|
|
|
func (x *BinaryOpExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
setExprLeadingTrivia(x.Tokens.GetParentheses(), x.LeftOperand, t)
|
|
}
|
|
|
|
func (x *BinaryOpExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), x.RightOperand)
|
|
}
|
|
|
|
func (x *BinaryOpExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewBinaryOpTokens(x.Operation)
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.GetParentheses(), x.RightOperand, t)
|
|
}
|
|
|
|
func (x *BinaryOpExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *BinaryOpExpression) print(w io.Writer, p *printer) {
|
|
precedence := operatorPrecedence(x.Operation)
|
|
p.fprintf(w, "%[2](%.[1]*[3]v% [4]v% .[1]*[5]o%[6])",
|
|
precedence,
|
|
x.Tokens.GetParentheses(),
|
|
x.LeftOperand, x.Tokens.GetOperator(x.Operation), x.RightOperand,
|
|
x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*BinaryOpExpression) isExpression() {}
|
|
|
|
// ConditionalExpression represents a semantically-analzed conditional expression (i.e.
|
|
// <condition> '?' <true> ':' <false>).
|
|
type ConditionalExpression struct {
|
|
// The syntax node associated with the conditional expression.
|
|
Syntax *hclsyntax.ConditionalExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens syntax.NodeTokens
|
|
|
|
// The condition.
|
|
Condition Expression
|
|
// The result of the expression if the condition evaluates to true.
|
|
TrueResult Expression
|
|
// The result of the expression if the condition evaluates to false.
|
|
FalseResult Expression
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the conditional expression.
|
|
func (x *ConditionalExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the conditional expression.
|
|
func (x *ConditionalExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the conditional expression.
|
|
func (x *ConditionalExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *ConditionalExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
conditionDiags := x.Condition.Typecheck(true)
|
|
diagnostics = append(diagnostics, conditionDiags...)
|
|
|
|
trueDiags := x.TrueResult.Typecheck(true)
|
|
diagnostics = append(diagnostics, trueDiags...)
|
|
|
|
falseDiags := x.FalseResult.Typecheck(true)
|
|
diagnostics = append(diagnostics, falseDiags...)
|
|
}
|
|
|
|
// Compute the type of the result.
|
|
resultType, _ := UnifyTypes(x.TrueResult.Type(), x.FalseResult.Type())
|
|
|
|
// Typecheck the condition expression.
|
|
if InputType(BoolType).ConversionFrom(x.Condition.Type()) == NoConversion {
|
|
diagnostics = append(diagnostics, ExprNotConvertible(InputType(BoolType), x.Condition))
|
|
}
|
|
|
|
x.exprType = liftOperationType(resultType, x.Condition)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *ConditionalExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.ConditionalExpr{
|
|
Condition: &syntaxExpr{expr: x.Condition},
|
|
TrueResult: &syntaxExpr{expr: x.TrueResult},
|
|
FalseResult: &syntaxExpr{expr: x.FalseResult},
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *ConditionalExpression) HasLeadingTrivia() bool {
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ConditionalTokens:
|
|
if tokens.Parentheses.Any() {
|
|
return true
|
|
}
|
|
case *syntax.TemplateConditionalTokens:
|
|
return len(tokens.OpenIf.LeadingTrivia) != 0
|
|
}
|
|
return x.Condition.HasLeadingTrivia()
|
|
}
|
|
|
|
func (x *ConditionalExpression) HasTrailingTrivia() bool {
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ConditionalTokens:
|
|
if tokens.Parentheses.Any() {
|
|
return true
|
|
}
|
|
case *syntax.TemplateConditionalTokens:
|
|
return len(tokens.CloseEndif.TrailingTrivia) != 0
|
|
}
|
|
return x.FalseResult.HasTrailingTrivia()
|
|
}
|
|
|
|
func (x *ConditionalExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ConditionalTokens:
|
|
if tokens.Parentheses.Any() {
|
|
return tokens.Parentheses.GetLeadingTrivia()
|
|
}
|
|
case *syntax.TemplateConditionalTokens:
|
|
return tokens.OpenIf.LeadingTrivia
|
|
}
|
|
return x.Condition.GetLeadingTrivia()
|
|
}
|
|
|
|
func (x *ConditionalExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ConditionalTokens:
|
|
if tokens.Parentheses.Any() {
|
|
tokens.Parentheses.SetLeadingTrivia(t)
|
|
return
|
|
}
|
|
case *syntax.TemplateConditionalTokens:
|
|
tokens.OpenIf.LeadingTrivia = t
|
|
return
|
|
}
|
|
x.Condition.SetLeadingTrivia(t)
|
|
}
|
|
|
|
func (x *ConditionalExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ConditionalTokens:
|
|
if tokens.Parentheses.Any() {
|
|
return tokens.Parentheses.GetTrailingTrivia()
|
|
}
|
|
case *syntax.TemplateConditionalTokens:
|
|
return tokens.CloseEndif.TrailingTrivia
|
|
}
|
|
return x.FalseResult.GetTrailingTrivia()
|
|
}
|
|
|
|
func (x *ConditionalExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewConditionalTokens()
|
|
}
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ConditionalTokens:
|
|
if tokens.Parentheses.Any() {
|
|
tokens.Parentheses.SetTrailingTrivia(t)
|
|
return
|
|
}
|
|
case *syntax.TemplateConditionalTokens:
|
|
tokens.CloseEndif.TrailingTrivia = t
|
|
return
|
|
}
|
|
x.FalseResult.SetTrailingTrivia(t)
|
|
}
|
|
|
|
func (x *ConditionalExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *ConditionalExpression) print(w io.Writer, p *printer) {
|
|
tokens := x.Tokens
|
|
if tokens == nil {
|
|
tokens = syntax.NewConditionalTokens()
|
|
}
|
|
|
|
switch tokens := tokens.(type) {
|
|
case *syntax.ConditionalTokens:
|
|
p.fprintf(w, "%(%v% v% v% v% v%)",
|
|
tokens.Parentheses,
|
|
x.Condition, tokens.QuestionMark, x.TrueResult, tokens.Colon, x.FalseResult,
|
|
tokens.Parentheses)
|
|
case *syntax.TemplateConditionalTokens:
|
|
p.fprintf(w, "%v%v% v%v%v", tokens.OpenIf, tokens.If, x.Condition, tokens.CloseIf, x.TrueResult)
|
|
if tokens.Else != nil {
|
|
p.fprintf(w, "%v%v%v%v", tokens.OpenElse, tokens.Else, tokens.CloseElse, x.FalseResult)
|
|
}
|
|
p.fprintf(w, "%v%v%v", tokens.OpenEndif, tokens.Endif, tokens.CloseEndif)
|
|
}
|
|
}
|
|
|
|
func (*ConditionalExpression) isExpression() {}
|
|
|
|
// ErrorExpression represents an expression that could not be bound due to an error.
|
|
type ErrorExpression struct {
|
|
// The syntax node associated with the error, if any.
|
|
Syntax hclsyntax.Node
|
|
// The tokens associated with the error.
|
|
Tokens syntax.NodeTokens
|
|
// The message associated with the error.
|
|
Message string
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the error expression.
|
|
func (x *ErrorExpression) SyntaxNode() hclsyntax.Node {
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the error expression.
|
|
func (x *ErrorExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the error expression.
|
|
func (x *ErrorExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *ErrorExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
return nil
|
|
}
|
|
|
|
func (x *ErrorExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
return cty.DynamicVal, hcl.Diagnostics{{
|
|
Severity: hcl.DiagError,
|
|
Summary: x.Message,
|
|
}}
|
|
}
|
|
|
|
func (x *ErrorExpression) HasLeadingTrivia() bool {
|
|
return false
|
|
}
|
|
|
|
func (x *ErrorExpression) HasTrailingTrivia() bool {
|
|
return false
|
|
}
|
|
|
|
func (x *ErrorExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return nil
|
|
}
|
|
|
|
func (x *ErrorExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
}
|
|
|
|
func (x *ErrorExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return nil
|
|
}
|
|
|
|
func (x *ErrorExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
}
|
|
|
|
func (x *ErrorExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *ErrorExpression) print(w io.Writer, p *printer) {
|
|
p.fprintf(w, "error(%q)", x.Message)
|
|
}
|
|
|
|
func (*ErrorExpression) isExpression() {}
|
|
|
|
// ForExpression represents a semantically-analyzed for expression.
|
|
type ForExpression struct {
|
|
fmt.Formatter
|
|
|
|
// The syntax node associated with the for expression.
|
|
Syntax *hclsyntax.ForExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens syntax.NodeTokens
|
|
|
|
// The key variable, if any.
|
|
KeyVariable *Variable
|
|
// The value variable.
|
|
ValueVariable *Variable
|
|
|
|
// The collection being iterated.
|
|
Collection Expression
|
|
// The expression that generates the keys of the result, if any. If this field is non-nil, the result is a map.
|
|
Key Expression
|
|
// The expression that generates the values of the result.
|
|
Value Expression
|
|
// The condition that filters the items of the result, if any.
|
|
Condition Expression
|
|
|
|
// True if the value expression is being grouped.
|
|
Group bool
|
|
|
|
// Whether the collection type should be strictly type-checked
|
|
// When true, unsupported collection types will result in an error
|
|
// otherwise a warning will be emitted
|
|
StrictCollectionTypechecking bool
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the for expression.
|
|
func (x *ForExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the for expression.
|
|
func (x *ForExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the for expression.
|
|
func (x *ForExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *ForExpression) typecheck(typecheckCollection, typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
var rng hcl.Range
|
|
if x.Syntax != nil {
|
|
rng = x.Syntax.CollExpr.Range()
|
|
}
|
|
|
|
if typecheckOperands {
|
|
collectionDiags := x.Collection.Typecheck(true)
|
|
diagnostics = append(diagnostics, collectionDiags...)
|
|
}
|
|
|
|
if typecheckCollection {
|
|
keyType, valueType, kvDiags := GetCollectionTypes(x.Collection.Type(), rng, x.StrictCollectionTypechecking)
|
|
diagnostics = append(diagnostics, kvDiags...)
|
|
|
|
if x.KeyVariable != nil {
|
|
x.KeyVariable.VariableType = keyType
|
|
}
|
|
x.ValueVariable.VariableType = valueType
|
|
}
|
|
|
|
if typecheckOperands {
|
|
if x.Key != nil {
|
|
keyDiags := x.Key.Typecheck(true)
|
|
diagnostics = append(diagnostics, keyDiags...)
|
|
}
|
|
|
|
valueDiags := x.Value.Typecheck(true)
|
|
diagnostics = append(diagnostics, valueDiags...)
|
|
|
|
if x.Condition != nil {
|
|
conditionDiags := x.Condition.Typecheck(true)
|
|
diagnostics = append(diagnostics, conditionDiags...)
|
|
}
|
|
}
|
|
|
|
if x.Key != nil {
|
|
// A key expression is only present when producing a map. Key types must therefore be strings.
|
|
if !InputType(StringType).ConversionFrom(x.Key.Type()).Exists() {
|
|
diagnostics = append(diagnostics, ExprNotConvertible(InputType(StringType), x.Key))
|
|
}
|
|
}
|
|
|
|
if x.Condition != nil {
|
|
if !InputType(BoolType).ConversionFrom(x.Condition.Type()).Exists() {
|
|
diagnostics = append(diagnostics, ExprNotConvertible(InputType(BoolType), x.Condition))
|
|
}
|
|
}
|
|
|
|
// If there is a key expression, we are producing a map. Otherwise, we are producing an list. In either case, wrap
|
|
// the result type in the same set of eventuals and optionals present in the collection type.
|
|
var resultType Type
|
|
if x.Key != nil {
|
|
valueType := x.Value.Type()
|
|
if x.Group {
|
|
valueType = NewListType(valueType)
|
|
}
|
|
resultType = wrapIterableResultType(x.Collection.Type(), NewMapType(valueType))
|
|
} else {
|
|
resultType = wrapIterableResultType(x.Collection.Type(), NewListType(x.Value.Type()))
|
|
}
|
|
|
|
// If either the key expression or the condition expression is eventual, the result is eventual: each of these
|
|
// values is required to determine which items are present in the result.
|
|
var liftArgs []Expression
|
|
if x.Key != nil {
|
|
liftArgs = append(liftArgs, x.Key)
|
|
}
|
|
if x.Condition != nil {
|
|
liftArgs = append(liftArgs, x.Condition)
|
|
}
|
|
|
|
x.exprType = liftOperationType(resultType, liftArgs...)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *ForExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
return x.typecheck(true, typecheckOperands)
|
|
}
|
|
|
|
func (x *ForExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.ForExpr{
|
|
ValVar: x.ValueVariable.Name,
|
|
CollExpr: &syntaxExpr{expr: x.Collection},
|
|
ValExpr: &syntaxExpr{expr: x.Value},
|
|
Group: x.Group,
|
|
}
|
|
if x.KeyVariable != nil {
|
|
syntax.KeyVar = x.KeyVariable.Name
|
|
}
|
|
if x.Key != nil {
|
|
syntax.KeyExpr = &syntaxExpr{expr: x.Key}
|
|
}
|
|
if x.Condition != nil {
|
|
syntax.CondExpr = &syntaxExpr{expr: x.Condition}
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *ForExpression) HasLeadingTrivia() bool {
|
|
return x.Tokens != nil
|
|
}
|
|
|
|
func (x *ForExpression) HasTrailingTrivia() bool {
|
|
return x.Tokens != nil
|
|
}
|
|
|
|
func (x *ForExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ForTokens:
|
|
return getExprLeadingTrivia(tokens.Parentheses, tokens.Open)
|
|
case *syntax.TemplateForTokens:
|
|
return tokens.OpenFor.LeadingTrivia
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (x *ForExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
keyVariable := ""
|
|
if x.KeyVariable != nil {
|
|
keyVariable = x.KeyVariable.Name
|
|
}
|
|
x.Tokens = syntax.NewForTokens(keyVariable, x.ValueVariable.Name, x.Key != nil, x.Group, x.Condition != nil)
|
|
}
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ForTokens:
|
|
setExprLeadingTrivia(tokens.Parentheses, &tokens.Open, t)
|
|
case *syntax.TemplateForTokens:
|
|
tokens.OpenFor.LeadingTrivia = t
|
|
}
|
|
}
|
|
|
|
func (x *ForExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ForTokens:
|
|
return getExprTrailingTrivia(tokens.Parentheses, tokens.Close)
|
|
case *syntax.TemplateForTokens:
|
|
return tokens.CloseEndfor.TrailingTrivia
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (x *ForExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
keyVariable := ""
|
|
if x.KeyVariable != nil {
|
|
keyVariable = x.KeyVariable.Name
|
|
}
|
|
x.Tokens = syntax.NewForTokens(keyVariable, x.ValueVariable.Name, x.Key != nil, x.Group, x.Condition != nil)
|
|
}
|
|
switch tokens := x.Tokens.(type) {
|
|
case *syntax.ForTokens:
|
|
setExprTrailingTrivia(tokens.Parentheses, &tokens.Close, t)
|
|
case *syntax.TemplateForTokens:
|
|
tokens.CloseEndfor.TrailingTrivia = t
|
|
}
|
|
}
|
|
|
|
func (x *ForExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *ForExpression) print(w io.Writer, p *printer) {
|
|
tokens := x.Tokens
|
|
if tokens == nil {
|
|
keyVariable := ""
|
|
if x.KeyVariable != nil {
|
|
keyVariable = x.KeyVariable.Name
|
|
}
|
|
syntax.NewForTokens(keyVariable, x.ValueVariable.Name, x.Key != nil, x.Group, x.Condition != nil)
|
|
}
|
|
|
|
switch tokens := tokens.(type) {
|
|
case *syntax.ForTokens:
|
|
// Print the opening rune and the for token.
|
|
p.fprintf(w, "%(%v%v", tokens.Parentheses, tokens.Open, tokens.For)
|
|
|
|
// Print the key variable, if any.
|
|
if x.KeyVariable != nil {
|
|
keyToken := tokens.Key
|
|
if x.KeyVariable != nil && keyToken == nil {
|
|
keyToken = &syntax.Token{Raw: hclsyntax.Token{Type: hclsyntax.TokenIdent}}
|
|
}
|
|
|
|
key := identToken(*keyToken, x.KeyVariable.Name)
|
|
p.fprintf(w, "% v%v", key, tokens.Comma)
|
|
}
|
|
|
|
// Print the value variable, the in token, the collection expression, and the colon.
|
|
value := identToken(tokens.Value, x.ValueVariable.Name)
|
|
p.fprintf(w, "% v% v% v%v", value, tokens.In, x.Collection, tokens.Colon)
|
|
|
|
// Print the key expression and arrow token, if any.
|
|
if x.Key != nil {
|
|
p.fprintf(w, "% v% v", x.Key, tokens.Arrow)
|
|
}
|
|
|
|
// Print the value expression.
|
|
p.fprintf(w, "% v", x.Value)
|
|
|
|
// Print the group token, if any.
|
|
if x.Group {
|
|
p.fprintf(w, "%v", tokens.Group)
|
|
}
|
|
|
|
// Print the if token and the condition, if any.
|
|
if x.Condition != nil {
|
|
p.fprintf(w, "% v% v", tokens.If, x.Condition)
|
|
}
|
|
|
|
// Print the closing rune.
|
|
p.fprintf(w, "%v%)", tokens.Close, tokens.Parentheses)
|
|
case *syntax.TemplateForTokens:
|
|
// Print the opening sequence.
|
|
p.fprintf(w, "%v%v", tokens.OpenFor, tokens.For)
|
|
|
|
// Print the key variable, if any.
|
|
if x.KeyVariable != nil {
|
|
keyToken := tokens.Key
|
|
if x.KeyVariable != nil && keyToken == nil {
|
|
keyToken = &syntax.Token{Raw: hclsyntax.Token{Type: hclsyntax.TokenIdent}}
|
|
}
|
|
|
|
key := identToken(*keyToken, x.KeyVariable.Name)
|
|
p.fprintf(w, "% v%v", key, tokens.Comma)
|
|
}
|
|
|
|
// Print the value variable, the in token, the collection expression, the control sequence terminator, the
|
|
// value expression, and the closing sequence.
|
|
p.fprintf(w, "% v% v% v%v%v%v%v%v",
|
|
identToken(tokens.Value, x.ValueVariable.Name), tokens.In, x.Collection, tokens.CloseFor,
|
|
x.Value,
|
|
tokens.OpenEndfor, tokens.Endfor, tokens.CloseEndfor)
|
|
}
|
|
}
|
|
|
|
func (*ForExpression) isExpression() {}
|
|
|
|
// FunctionCallExpression represents a semantically-analyzed function call expression.
|
|
type FunctionCallExpression struct {
|
|
// The syntax node associated with the function call expression.
|
|
Syntax *hclsyntax.FunctionCallExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.FunctionCallTokens
|
|
|
|
// The name of the called function.
|
|
Name string
|
|
// The signature of the called function.
|
|
Signature StaticFunctionSignature
|
|
// The arguments to the function call.
|
|
Args []Expression
|
|
// ExpandFinal indicates that the final argument should be a tuple, list, or set whose elements will be passed as
|
|
// individual arguments to the function.
|
|
ExpandFinal bool
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the function call expression.
|
|
func (x *FunctionCallExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the function call expression.
|
|
func (x *FunctionCallExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the function call expression.
|
|
func (x *FunctionCallExpression) Type() Type {
|
|
return x.Signature.ReturnType
|
|
}
|
|
|
|
func (x *FunctionCallExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
for _, arg := range x.Args {
|
|
argDiagnostics := arg.Typecheck(true)
|
|
diagnostics = append(diagnostics, argDiagnostics...)
|
|
}
|
|
}
|
|
|
|
var rng hcl.Range
|
|
if x.Syntax != nil {
|
|
rng = x.Syntax.Range()
|
|
}
|
|
|
|
// Typecheck the function's arguments.
|
|
typecheckDiags := typecheckArgs(rng, x.Signature, x.Args...)
|
|
diagnostics = append(diagnostics, typecheckDiags...)
|
|
|
|
// Unless the function is already automatically using an
|
|
// Output-returning version, modify the signature to account
|
|
// for automatic lifting to Promise or Output.
|
|
_, isOutput := x.Signature.ReturnType.(*OutputType)
|
|
if !isOutput {
|
|
x.Signature.ReturnType = liftOperationType(x.Signature.ReturnType, x.Args...)
|
|
}
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *FunctionCallExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.FunctionCallExpr{
|
|
Name: x.Name,
|
|
Args: make([]hclsyntax.Expression, len(x.Args)),
|
|
ExpandFinal: x.ExpandFinal,
|
|
}
|
|
for i, arg := range x.Args {
|
|
syntax.Args[i] = &syntaxExpr{expr: arg}
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *FunctionCallExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *FunctionCallExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *FunctionCallExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetName(x.Name))
|
|
}
|
|
|
|
func (x *FunctionCallExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewFunctionCallTokens(x.Name, len(x.Args))
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, &x.Tokens.Name, t)
|
|
}
|
|
|
|
func (x *FunctionCallExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetCloseParen())
|
|
}
|
|
|
|
func (x *FunctionCallExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewFunctionCallTokens(x.Name, len(x.Args))
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.Parentheses, &x.Tokens.CloseParen, t)
|
|
}
|
|
|
|
func (x *FunctionCallExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *FunctionCallExpression) print(w io.Writer, p *printer) {
|
|
// Print the name and opening parenthesis.
|
|
p.fprintf(w, "%(%v%v", x.Tokens.GetParentheses(), x.Tokens.GetName(x.Name), x.Tokens.GetOpenParen())
|
|
|
|
// Print each argument and its comma.
|
|
commas := x.Tokens.GetCommas(len(x.Args))
|
|
for i, arg := range x.Args {
|
|
if i == 0 {
|
|
p.fprintf(w, "%v", arg)
|
|
} else {
|
|
p.fprintf(w, "% v", arg)
|
|
}
|
|
|
|
if i < len(x.Args)-1 {
|
|
var comma syntax.Token
|
|
if i < len(commas) {
|
|
comma = commas[i]
|
|
}
|
|
p.fprintf(w, "%v", comma)
|
|
}
|
|
}
|
|
|
|
// If there were commas left over, print the trivia for each.
|
|
if len(x.Args) > 0 && len(x.Args)-1 <= len(commas) {
|
|
for _, comma := range commas[len(x.Args)-1:] {
|
|
p.fprintf(w, "%v", comma.AllTrivia().CollapseWhitespace())
|
|
}
|
|
}
|
|
|
|
// Print the closing parenthesis.
|
|
p.fprintf(w, "%v%)", x.Tokens.GetCloseParen(), x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*FunctionCallExpression) isExpression() {}
|
|
|
|
// IndexExpression represents a semantically-analyzed index expression.
|
|
type IndexExpression struct {
|
|
// The syntax node associated with the index expression.
|
|
Syntax *hclsyntax.IndexExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.IndexTokens
|
|
|
|
// The collection being indexed.
|
|
Collection Expression
|
|
// The index key.
|
|
Key Expression
|
|
// Whether the collection type should be strictly type-checked
|
|
// When true, unsupported collection types will result in an error
|
|
// otherwise a warning will be emitted
|
|
StrictCollectionTypechecking bool
|
|
|
|
keyType Type
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the index expression.
|
|
func (x *IndexExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the index expression.
|
|
func (x *IndexExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// KeyType returns the expected type of the index expression's key.
|
|
func (x *IndexExpression) KeyType() Type {
|
|
return x.keyType
|
|
}
|
|
|
|
// Type returns the type of the index expression.
|
|
func (x *IndexExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *IndexExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
collectionDiags := x.Collection.Typecheck(true)
|
|
diagnostics = append(diagnostics, collectionDiags...)
|
|
|
|
keyDiags := x.Key.Typecheck(true)
|
|
diagnostics = append(diagnostics, keyDiags...)
|
|
}
|
|
|
|
var rng hcl.Range
|
|
if x.Syntax != nil {
|
|
rng = x.Syntax.Collection.Range()
|
|
}
|
|
|
|
keyType, valueType, kvDiags := GetCollectionTypes(x.Collection.Type(), rng, x.StrictCollectionTypechecking)
|
|
diagnostics = append(diagnostics, kvDiags...)
|
|
x.keyType = keyType
|
|
|
|
if lit, ok := x.Key.(*LiteralValueExpression); ok {
|
|
traverser := hcl.TraverseIndex{
|
|
Key: lit.Value,
|
|
}
|
|
valueType, traverseDiags := x.Collection.Type().Traverse(traverser)
|
|
if len(traverseDiags) == 0 {
|
|
x.exprType = valueType.(Type)
|
|
return diagnostics
|
|
}
|
|
}
|
|
|
|
if !InputType(keyType).ConversionFrom(x.Key.Type()).Exists() {
|
|
diagnostics = append(diagnostics, ExprNotConvertible(InputType(keyType), x.Key))
|
|
}
|
|
|
|
resultType := wrapIterableResultType(x.Collection.Type(), valueType)
|
|
x.exprType = liftOperationType(resultType, x.Key)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *IndexExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.IndexExpr{
|
|
Collection: &syntaxExpr{expr: x.Collection},
|
|
Key: &syntaxExpr{expr: x.Key},
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *IndexExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Collection)
|
|
}
|
|
|
|
func (x *IndexExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *IndexExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetOpenBracket())
|
|
}
|
|
|
|
func (x *IndexExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewIndexTokens()
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, x.Collection, t)
|
|
}
|
|
|
|
func (x *IndexExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetCloseBracket())
|
|
}
|
|
|
|
func (x *IndexExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewIndexTokens()
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.Parentheses, &x.Tokens.CloseBracket, t)
|
|
}
|
|
|
|
func (x *IndexExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *IndexExpression) print(w io.Writer, p *printer) {
|
|
p.fprintf(w, "%(%v%v%v%v%)",
|
|
x.Tokens.GetParentheses(),
|
|
x.Collection, x.Tokens.GetOpenBracket(), x.Key, x.Tokens.GetCloseBracket(),
|
|
x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*IndexExpression) isExpression() {}
|
|
|
|
func literalText(value cty.Value, rawBytes []byte, escaped, quoted bool) string {
|
|
if len(rawBytes) > 0 {
|
|
parsed, diags := hclsyntax.ParseExpression(rawBytes, "", hcl.Pos{})
|
|
if !diags.HasErrors() {
|
|
if lit, ok := parsed.(*hclsyntax.LiteralValueExpr); ok && lit.Val.RawEquals(value) {
|
|
return string(rawBytes)
|
|
}
|
|
}
|
|
}
|
|
|
|
switch value.Type() {
|
|
case cty.Bool:
|
|
if value.True() {
|
|
return "true"
|
|
}
|
|
return "false"
|
|
case cty.Number:
|
|
bf := value.AsBigFloat()
|
|
i, acc := bf.Int64()
|
|
if acc == big.Exact {
|
|
return strconv.FormatInt(i, 10)
|
|
}
|
|
d, _ := bf.Float64()
|
|
return fmt.Sprintf("%g", d)
|
|
case cty.String:
|
|
if !escaped {
|
|
return value.AsString()
|
|
}
|
|
s := escapeString(value.AsString())
|
|
if quoted {
|
|
return fmt.Sprintf(`"%s"`, s)
|
|
}
|
|
return s
|
|
default:
|
|
panic(fmt.Errorf("unexpected literal type %v", value.Type().FriendlyName()))
|
|
}
|
|
}
|
|
|
|
func escapeString(s string) string {
|
|
// escape special characters
|
|
s = strconv.Quote(s)
|
|
s = s[1 : len(s)-1] // Remove surrounding double quote (`"`)
|
|
|
|
// Escape `${`
|
|
runes := []rune(s)
|
|
out := slice.Prealloc[rune](len(runes))
|
|
for i, r := range runes {
|
|
next := func() rune {
|
|
if i >= len(runes)-1 {
|
|
return 0
|
|
}
|
|
return runes[i+1]
|
|
}
|
|
if r == '$' && next() == '{' {
|
|
out = append(out, '$')
|
|
} else if r == '%' && next() == '{' {
|
|
out = append(out, '%')
|
|
}
|
|
out = append(out, r)
|
|
}
|
|
return string(out)
|
|
}
|
|
|
|
// LiteralValueExpression represents a semantically-analyzed literal value expression.
|
|
type LiteralValueExpression struct {
|
|
// The syntax node associated with the literal value expression.
|
|
Syntax *hclsyntax.LiteralValueExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.LiteralValueTokens
|
|
|
|
// The value of the expression.
|
|
Value cty.Value
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the literal value expression.
|
|
func (x *LiteralValueExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the literal value expression.
|
|
func (x *LiteralValueExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the literal value expression.
|
|
func (x *LiteralValueExpression) Type() Type {
|
|
if x.exprType == nil {
|
|
_ = x.Typecheck(false)
|
|
}
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *LiteralValueExpression) Typecheck(_ bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
typ := NoneType
|
|
if !x.Value.IsNull() {
|
|
typ = ctyTypeToType(x.Value.Type(), false, false)
|
|
}
|
|
|
|
switch {
|
|
case typ == NumberType || typ == IntType:
|
|
// If it's an int, we might want to represent it as a big int if it's too big for a float
|
|
f := x.Value.AsBigFloat()
|
|
if _, acc := f.Int(nil); acc == big.Exact {
|
|
// This is an integer 'i', but it might roundtrip through float and if it does we should leave it as number.
|
|
// But otherwise say this is a BigInteger type to maintain precision.
|
|
if _, acc := f.Float64(); acc != big.Exact {
|
|
typ = BigIntegerType
|
|
}
|
|
}
|
|
// Not an int, must be a float.
|
|
typ = NewConstType(typ, x.Value)
|
|
case typ == NoneType || typ == StringType || typ == BoolType:
|
|
typ = NewConstType(typ, x.Value)
|
|
default:
|
|
var rng hcl.Range
|
|
if x.Syntax != nil {
|
|
rng = x.Syntax.Range()
|
|
}
|
|
typ, diagnostics = DynamicType, hcl.Diagnostics{unsupportedLiteralValue(x.Value, rng)}
|
|
}
|
|
|
|
x.exprType = typ
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *LiteralValueExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.LiteralValueExpr{
|
|
Val: x.Value,
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *LiteralValueExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *LiteralValueExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *LiteralValueExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
if v := x.Tokens.GetValue(x.Value); len(v) > 0 {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), v[0])
|
|
}
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), nil)
|
|
}
|
|
|
|
func (x *LiteralValueExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewLiteralValueTokens(x.Value)
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, &x.Tokens.Value[0], t)
|
|
}
|
|
|
|
func (x *LiteralValueExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
if v := x.Tokens.GetValue(x.Value); len(v) > 0 {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), v[len(v)-1])
|
|
}
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), nil)
|
|
}
|
|
|
|
func (x *LiteralValueExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewLiteralValueTokens(x.Value)
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.Parentheses, &x.Tokens.Value[len(x.Tokens.Value)-1], t)
|
|
}
|
|
|
|
func (x *LiteralValueExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *LiteralValueExpression) printLit(w io.Writer, p *printer, escaped bool) {
|
|
// Literals are... odd. They may be composed of multiple tokens, but those tokens should never contain interior
|
|
// trivia.
|
|
|
|
var leading, trailing syntax.TriviaList
|
|
var rawBytes []byte
|
|
if toks := x.Tokens.GetValue(x.Value); len(toks) > 0 {
|
|
leading, trailing = toks[0].LeadingTrivia, toks[len(toks)-1].TrailingTrivia
|
|
|
|
for _, t := range toks {
|
|
rawBytes = append(rawBytes, t.Raw.Bytes...)
|
|
}
|
|
}
|
|
|
|
p.fprintf(w, "%(%v%v%v%)",
|
|
x.Tokens.GetParentheses(),
|
|
leading, literalText(x.Value, rawBytes, escaped, false), trailing,
|
|
x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (x *LiteralValueExpression) print(w io.Writer, p *printer) {
|
|
x.printLit(w, p, false)
|
|
}
|
|
|
|
func (*LiteralValueExpression) isExpression() {}
|
|
|
|
// ObjectConsItem records a key-value pair that is part of object construction expression.
|
|
type ObjectConsItem struct {
|
|
// The key.
|
|
Key Expression
|
|
// The value.
|
|
Value Expression
|
|
}
|
|
|
|
// ObjectConsExpression represents a semantically-analyzed object construction expression.
|
|
type ObjectConsExpression struct {
|
|
// The syntax node associated with the object construction expression.
|
|
Syntax *hclsyntax.ObjectConsExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.ObjectConsTokens
|
|
|
|
// The items that comprise the object construction expression.
|
|
Items []ObjectConsItem
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the object construction expression.
|
|
func (x *ObjectConsExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the object construction expression.
|
|
func (x *ObjectConsExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the object construction expression.
|
|
func (x *ObjectConsExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *ObjectConsExpression) WithType(updateType func(Type) *ObjectConsExpression) *ObjectConsExpression {
|
|
return updateType(x.exprType)
|
|
}
|
|
|
|
func (x *ObjectConsExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
keys := slice.Prealloc[Expression](len(x.Items))
|
|
for _, item := range x.Items {
|
|
if typecheckOperands {
|
|
keyDiags := item.Key.Typecheck(true)
|
|
diagnostics = append(diagnostics, keyDiags...)
|
|
|
|
valDiags := item.Value.Typecheck(true)
|
|
diagnostics = append(diagnostics, valDiags...)
|
|
}
|
|
|
|
keys = append(keys, item.Key)
|
|
if !InputType(StringType).ConversionFrom(item.Key.Type()).Exists() {
|
|
diagnostics = append(diagnostics, objectKeysMustBeStrings(item.Key))
|
|
}
|
|
}
|
|
|
|
// Attempt to build an object type out of the result. If there are any attribute names that come from variables,
|
|
// type the result as map(unify(propertyTypes)).
|
|
properties, isMapType, types := map[string]Type{}, false, []Type{}
|
|
for _, item := range x.Items {
|
|
types = append(types, item.Value.Type())
|
|
|
|
key := item.Key
|
|
if template, ok := key.(*TemplateExpression); ok && len(template.Parts) == 1 {
|
|
key = template.Parts[0]
|
|
}
|
|
|
|
keyLit, ok := key.(*LiteralValueExpression)
|
|
if ok {
|
|
key, err := convert.Convert(keyLit.Value, cty.String)
|
|
if err == nil {
|
|
properties[key.AsString()] = item.Value.Type()
|
|
continue
|
|
}
|
|
}
|
|
isMapType = true
|
|
}
|
|
var typ Type
|
|
if isMapType {
|
|
elementType, _ := UnifyTypes(types...)
|
|
typ = NewMapType(elementType)
|
|
} else {
|
|
typ = NewObjectType(properties)
|
|
}
|
|
|
|
x.exprType = liftOperationType(typ, keys...)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *ObjectConsExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.ObjectConsExpr{
|
|
Items: make([]hclsyntax.ObjectConsItem, len(x.Items)),
|
|
}
|
|
for i, item := range x.Items {
|
|
syntax.Items[i] = hclsyntax.ObjectConsItem{
|
|
KeyExpr: &syntaxExpr{expr: item.Key},
|
|
ValueExpr: &syntaxExpr{expr: item.Value},
|
|
}
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *ObjectConsExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *ObjectConsExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *ObjectConsExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetOpenBrace(len(x.Items)))
|
|
}
|
|
|
|
func (x *ObjectConsExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewObjectConsTokens(len(x.Items))
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, &x.Tokens.OpenBrace, t)
|
|
}
|
|
|
|
func (x *ObjectConsExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetCloseBrace())
|
|
}
|
|
|
|
func (x *ObjectConsExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewObjectConsTokens(len(x.Items))
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.Parentheses, &x.Tokens.CloseBrace, t)
|
|
}
|
|
|
|
func (x *ObjectConsExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *ObjectConsExpression) print(w io.Writer, p *printer) {
|
|
// Print the opening brace.
|
|
p.fprintf(w, "%(%v", x.Tokens.GetParentheses(), x.Tokens.GetOpenBrace(len(x.Items)))
|
|
|
|
// Print the items.
|
|
isMultiLine, trailingNewline := false, false
|
|
p.indented(func() {
|
|
items := x.Tokens.GetItems(len(x.Items))
|
|
for i, item := range x.Items {
|
|
tokens := syntax.NewObjectConsItemTokens(i == len(x.Items)-1)
|
|
if i < len(items) {
|
|
tokens = items[i]
|
|
}
|
|
|
|
if item.Key.HasLeadingTrivia() {
|
|
if _, i := item.Key.GetLeadingTrivia().Index("\n"); i != -1 {
|
|
isMultiLine = true
|
|
}
|
|
} else if len(items) > 1 {
|
|
isMultiLine = true
|
|
p.fprintf(w, "\n%s", p.indent)
|
|
}
|
|
p.fprintf(w, "%v% v% v", item.Key, tokens.Equals, item.Value)
|
|
|
|
if tokens.Comma != nil {
|
|
p.fprintf(w, "%v", tokens.Comma)
|
|
}
|
|
|
|
if isMultiLine && i == len(items)-1 {
|
|
trailingTrivia := item.Value.GetTrailingTrivia()
|
|
if tokens.Comma != nil {
|
|
trailingTrivia = tokens.Comma.TrailingTrivia
|
|
}
|
|
trailingNewline = trailingTrivia.EndsOnNewLine()
|
|
}
|
|
}
|
|
|
|
if len(x.Items) < len(items) {
|
|
for _, item := range items[len(x.Items):] {
|
|
p.fprintf(w, "%v", item.Equals.AllTrivia().CollapseWhitespace())
|
|
if item.Comma != nil {
|
|
p.fprintf(w, "%v", item.Comma.AllTrivia().CollapseWhitespace())
|
|
}
|
|
}
|
|
}
|
|
})
|
|
|
|
if x.Tokens != nil {
|
|
pre := ""
|
|
if isMultiLine && !trailingNewline {
|
|
pre = "\n" + p.indent
|
|
}
|
|
p.fprintf(w, "%s%v%)", pre, x.Tokens.CloseBrace, x.Tokens.Parentheses)
|
|
} else {
|
|
p.fprintf(w, "\n%s}", p.indent)
|
|
}
|
|
}
|
|
|
|
func (*ObjectConsExpression) isExpression() {}
|
|
|
|
func getTraverserTrivia(tokens syntax.TraverserTokens) (syntax.TriviaList, syntax.TriviaList) {
|
|
var leading, trailing syntax.TriviaList
|
|
switch tokens := tokens.(type) {
|
|
case *syntax.DotTraverserTokens:
|
|
leading = getExprLeadingTrivia(tokens.Parentheses, tokens.Dot)
|
|
trailing = getExprTrailingTrivia(tokens.Parentheses, tokens.Index)
|
|
case *syntax.BracketTraverserTokens:
|
|
leading = getExprLeadingTrivia(tokens.Parentheses, tokens.OpenBracket)
|
|
trailing = getExprTrailingTrivia(tokens.Parentheses, tokens.CloseBracket)
|
|
}
|
|
return leading, trailing
|
|
}
|
|
|
|
func setTraverserTrailingTrivia(tokens syntax.TraverserTokens, t syntax.TriviaList) {
|
|
switch tokens := tokens.(type) {
|
|
case *syntax.DotTraverserTokens:
|
|
setExprTrailingTrivia(tokens.Parentheses, &tokens.Index, t)
|
|
case *syntax.BracketTraverserTokens:
|
|
setExprTrailingTrivia(tokens.Parentheses, &tokens.CloseBracket, t)
|
|
default:
|
|
panic(fmt.Errorf("unexpected traverser of type %T", tokens))
|
|
}
|
|
}
|
|
|
|
func printTraverser(w io.Writer, p *printer, t hcl.Traverser, tokens syntax.TraverserTokens) {
|
|
var index string
|
|
switch t := t.(type) {
|
|
case hcl.TraverseAttr:
|
|
index = t.Name
|
|
case hcl.TraverseIndex:
|
|
index = literalText(t.Key, nil, true, true)
|
|
default:
|
|
panic(fmt.Errorf("unexpected traverser of type %T", t))
|
|
}
|
|
|
|
switch tokens := tokens.(type) {
|
|
case *syntax.DotTraverserTokens:
|
|
p.fprintf(w, "%(%v%v%)",
|
|
tokens.Parentheses,
|
|
tokens.Dot, identToken(tokens.Index, index),
|
|
tokens.Parentheses)
|
|
case *syntax.BracketTraverserTokens:
|
|
p.fprintf(w, "%(%v%v%v%)",
|
|
tokens.Parentheses,
|
|
tokens.OpenBracket, identToken(tokens.Index, index), tokens.CloseBracket,
|
|
tokens.Parentheses)
|
|
default:
|
|
panic(fmt.Errorf("unexpected traverser tokens of type %T", tokens))
|
|
}
|
|
}
|
|
|
|
func printRelativeTraversal(w io.Writer, p *printer, traversal hcl.Traversal, tokens []syntax.TraverserTokens) {
|
|
for i, traverser := range traversal {
|
|
// Fetch the traversal tokens.
|
|
var traverserTokens syntax.TraverserTokens
|
|
if i < len(tokens) {
|
|
traverserTokens = tokens[i]
|
|
}
|
|
printTraverser(w, p, traverser, traverserTokens)
|
|
}
|
|
|
|
// Print any remaining trivia.
|
|
if len(traversal) < len(tokens) {
|
|
for _, tokens := range tokens[len(traversal):] {
|
|
var trivia syntax.TriviaList
|
|
switch tokens := tokens.(type) {
|
|
case *syntax.DotTraverserTokens:
|
|
trivia = tokens.Dot.LeadingTrivia
|
|
trivia = append(trivia, tokens.Dot.TrailingTrivia...)
|
|
trivia = append(trivia, tokens.Index.LeadingTrivia...)
|
|
trivia = append(trivia, tokens.Index.TrailingTrivia...)
|
|
case *syntax.BracketTraverserTokens:
|
|
trivia = tokens.OpenBracket.LeadingTrivia
|
|
trivia = append(trivia, tokens.OpenBracket.TrailingTrivia...)
|
|
trivia = append(trivia, tokens.Index.LeadingTrivia...)
|
|
trivia = append(trivia, tokens.Index.TrailingTrivia...)
|
|
trivia = append(trivia, tokens.CloseBracket.LeadingTrivia...)
|
|
trivia = append(trivia, tokens.CloseBracket.TrailingTrivia...)
|
|
}
|
|
p.fprintf(w, "%v", trivia)
|
|
}
|
|
}
|
|
}
|
|
|
|
// RelativeTraversalExpression represents a semantically-analyzed relative traversal expression.
|
|
type RelativeTraversalExpression struct {
|
|
// The syntax node associated with the relative traversal expression.
|
|
Syntax *hclsyntax.RelativeTraversalExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.RelativeTraversalTokens
|
|
|
|
// The expression that computes the value being traversed.
|
|
Source Expression
|
|
// The traversal's parts.
|
|
Parts []Traversable
|
|
|
|
// The traversers.
|
|
Traversal hcl.Traversal
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the relative traversal expression.
|
|
func (x *RelativeTraversalExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the relative traversal expression.
|
|
func (x *RelativeTraversalExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the relative traversal expression.
|
|
func (x *RelativeTraversalExpression) Type() Type {
|
|
return GetTraversableType(x.Parts[len(x.Parts)-1])
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) typecheck(typecheckOperands, allowMissingVariables bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
sourceDiags := x.Source.Typecheck(true)
|
|
diagnostics = append(diagnostics, sourceDiags...)
|
|
}
|
|
|
|
parts, partDiags := bindTraversalParts(x.Source.Type(), x.Traversal, allowMissingVariables)
|
|
diagnostics = append(diagnostics, partDiags...)
|
|
|
|
x.Parts = parts
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
return x.typecheck(typecheckOperands, false)
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.RelativeTraversalExpr{
|
|
Source: &syntaxExpr{expr: x.Source},
|
|
Traversal: x.Traversal,
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Source)
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) HasTrailingTrivia() bool {
|
|
if parens := x.Tokens.GetParentheses(); parens.Any() {
|
|
return true
|
|
}
|
|
if x.Tokens != nil && len(x.Tokens.Traversal) > 0 {
|
|
return true
|
|
}
|
|
return x.Source.HasTrailingTrivia()
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Source)
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewRelativeTraversalTokens(x.Traversal)
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, x.Source, t)
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
if parens := x.Tokens.GetParentheses(); parens.Any() {
|
|
return parens.GetTrailingTrivia()
|
|
}
|
|
if traversal := x.Tokens.GetTraversal(x.Traversal); len(traversal) > 0 {
|
|
_, trailingTrivia := getTraverserTrivia(traversal[len(traversal)-1])
|
|
return trailingTrivia
|
|
}
|
|
return x.Source.GetTrailingTrivia()
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewRelativeTraversalTokens(x.Traversal)
|
|
}
|
|
if parens := x.Tokens.GetParentheses(); parens.Any() {
|
|
parens.SetTrailingTrivia(t)
|
|
return
|
|
}
|
|
if len(x.Tokens.Traversal) > 0 {
|
|
setTraverserTrailingTrivia(x.Tokens.Traversal[len(x.Tokens.Traversal)-1], t)
|
|
return
|
|
}
|
|
x.Source.SetTrailingTrivia(t)
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *RelativeTraversalExpression) print(w io.Writer, p *printer) {
|
|
// Print the source expression.
|
|
p.fprintf(w, "%(%v", x.Tokens.GetParentheses(), x.Source)
|
|
|
|
// Print the traversal.
|
|
printRelativeTraversal(w, p, x.Traversal, x.Tokens.GetTraversal(x.Traversal))
|
|
|
|
// Print the closing parentheses, if any.
|
|
p.fprintf(w, "%)", x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*RelativeTraversalExpression) isExpression() {}
|
|
|
|
// ScopeTraversalExpression represents a semantically-analyzed scope traversal expression.
|
|
type ScopeTraversalExpression struct {
|
|
// The syntax node associated with the scope traversal expression.
|
|
Syntax *hclsyntax.ScopeTraversalExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.ScopeTraversalTokens
|
|
|
|
// The traversal's parts.
|
|
Parts []Traversable
|
|
|
|
// The root name.
|
|
RootName string
|
|
// The traversers.
|
|
Traversal hcl.Traversal
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the scope traversal expression.
|
|
func (x *ScopeTraversalExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the scope traversal expression.
|
|
func (x *ScopeTraversalExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the scope traversal expression.
|
|
func (x *ScopeTraversalExpression) Type() Type {
|
|
return GetTraversableType(x.Parts[len(x.Parts)-1])
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) typecheck(typecheckOperands, allowMissingVariables bool) hcl.Diagnostics {
|
|
parts, diagnostics := bindTraversalParts(x.Parts[0], x.Traversal.SimpleSplit().Rel, allowMissingVariables)
|
|
x.Parts = parts
|
|
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
return x.typecheck(typecheckOperands, false)
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
root, hasValue := x.Parts[0].(ValueTraversable)
|
|
if !hasValue {
|
|
return cty.UnknownVal(cty.DynamicPseudoType), nil
|
|
}
|
|
|
|
rootValue, diags := root.Value(context)
|
|
if diags.HasErrors() {
|
|
return cty.NilVal, diags
|
|
}
|
|
diagnostics = append(diagnostics, diags...)
|
|
|
|
if len(x.Traversal) == 1 {
|
|
return rootValue, diagnostics
|
|
}
|
|
return x.Traversal[1:].TraverseRel(rootValue)
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) HasTrailingTrivia() bool {
|
|
if parens := x.Tokens.GetParentheses(); parens.Any() {
|
|
return true
|
|
}
|
|
if x.Tokens != nil && len(x.Tokens.Traversal) > 0 {
|
|
return true
|
|
}
|
|
return x.Tokens != nil
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetRoot(x.Traversal))
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewScopeTraversalTokens(x.Traversal)
|
|
}
|
|
x.Tokens.Root.LeadingTrivia = t
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
if parens := x.Tokens.GetParentheses(); parens.Any() {
|
|
return parens.GetTrailingTrivia()
|
|
}
|
|
if traversal := x.Tokens.GetTraversal(x.Traversal); len(traversal) > 0 {
|
|
_, trailingTrivia := getTraverserTrivia(traversal[len(traversal)-1])
|
|
return trailingTrivia
|
|
}
|
|
return x.Tokens.GetRoot(x.Traversal).TrailingTrivia
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewScopeTraversalTokens(x.Traversal)
|
|
}
|
|
if parens := x.Tokens.GetParentheses(); parens.Any() {
|
|
parens.SetTrailingTrivia(t)
|
|
return
|
|
}
|
|
if len(x.Tokens.Traversal) > 0 {
|
|
setTraverserTrailingTrivia(x.Tokens.Traversal[len(x.Tokens.Traversal)-1], t)
|
|
return
|
|
}
|
|
x.Tokens.Root.TrailingTrivia = t
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *ScopeTraversalExpression) print(w io.Writer, p *printer) {
|
|
// Print the root name.
|
|
p.fprintf(w, "%(%v", x.Tokens.GetParentheses(), x.Tokens.GetRoot(x.Traversal))
|
|
|
|
// Print the traversal.
|
|
printRelativeTraversal(w, p, x.Traversal[1:], x.Tokens.GetTraversal(x.Traversal))
|
|
|
|
// Print the closing parentheses, if any.
|
|
p.fprintf(w, "%)", x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*ScopeTraversalExpression) isExpression() {}
|
|
|
|
type SplatVariable struct {
|
|
Variable
|
|
|
|
symbol hclsyntax.AnonSymbolExpr
|
|
}
|
|
|
|
func (v *SplatVariable) Value(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
return (&v.symbol).Value(context)
|
|
}
|
|
|
|
// SplatExpression represents a semantically-analyzed splat expression.
|
|
type SplatExpression struct {
|
|
// The syntax node associated with the splat expression.
|
|
Syntax *hclsyntax.SplatExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.SplatTokens
|
|
|
|
// The expression being splatted.
|
|
Source Expression
|
|
// The expression applied to each element of the splat.
|
|
Each Expression
|
|
// The local variable definition associated with the current item being processed. This definition is not part of
|
|
// a scope, and can only be referenced by an AnonSymbolExpr.
|
|
Item *SplatVariable
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the splat expression.
|
|
func (x *SplatExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the splat expression.
|
|
func (x *SplatExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the splat expression.
|
|
func (x *SplatExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func splatItemType(source Expression, splatSyntax *hclsyntax.SplatExpr) (Expression, Type) {
|
|
sourceType := unwrapIterableSourceType(source.Type())
|
|
itemType := sourceType
|
|
switch sourceType := sourceType.(type) {
|
|
case *ListType:
|
|
itemType = sourceType.ElementType
|
|
case *SetType:
|
|
itemType = sourceType.ElementType
|
|
case *TupleType:
|
|
itemType, _ = UnifyTypes(sourceType.ElementTypes...)
|
|
default:
|
|
if sourceType != DynamicType {
|
|
var tupleSyntax *hclsyntax.TupleConsExpr
|
|
if splatSyntax != nil {
|
|
tupleSyntax = &hclsyntax.TupleConsExpr{
|
|
Exprs: []hclsyntax.Expression{splatSyntax.Source},
|
|
SrcRange: splatSyntax.Source.Range(),
|
|
OpenRange: splatSyntax.Source.StartRange(),
|
|
}
|
|
}
|
|
|
|
source = &TupleConsExpression{
|
|
Syntax: tupleSyntax,
|
|
Tokens: syntax.NewTupleConsTokens(1),
|
|
Expressions: []Expression{source},
|
|
exprType: NewListType(source.Type()),
|
|
}
|
|
}
|
|
}
|
|
return source, itemType
|
|
}
|
|
|
|
func (x *SplatExpression) typecheck(retypeItem, typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
sourceDiags := x.Source.Typecheck(true)
|
|
diagnostics = append(diagnostics, sourceDiags...)
|
|
}
|
|
|
|
if retypeItem {
|
|
x.Source, x.Item.VariableType = splatItemType(x.Source, x.Syntax)
|
|
}
|
|
|
|
if typecheckOperands {
|
|
eachDiags := x.Each.Typecheck(true)
|
|
diagnostics = append(diagnostics, eachDiags...)
|
|
}
|
|
|
|
x.exprType = wrapIterableResultType(x.Source.Type(), NewListType(x.Each.Type()))
|
|
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *SplatExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
return x.typecheck(true, typecheckOperands)
|
|
}
|
|
|
|
func (x *SplatExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.SplatExpr{
|
|
Source: &syntaxExpr{expr: x.Source},
|
|
Each: &syntaxExpr{expr: x.Each},
|
|
Item: &x.Item.symbol,
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *SplatExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Source)
|
|
}
|
|
|
|
func (x *SplatExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Each)
|
|
}
|
|
|
|
func (x *SplatExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Source)
|
|
}
|
|
|
|
func (x *SplatExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewSplatTokens(false)
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, &x.Tokens.Open, t)
|
|
}
|
|
|
|
func (x *SplatExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
if parens := x.Tokens.GetParentheses(); parens.Any() {
|
|
return parens.GetTrailingTrivia()
|
|
}
|
|
if closeTok := x.Tokens.GetClose(); closeTok != nil {
|
|
return closeTok.TrailingTrivia
|
|
}
|
|
return x.Tokens.GetStar().TrailingTrivia
|
|
}
|
|
|
|
func (x *SplatExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewSplatTokens(false)
|
|
}
|
|
if x.Tokens.Parentheses.Any() {
|
|
x.Tokens.Parentheses.SetTrailingTrivia(t)
|
|
return
|
|
}
|
|
if x.Tokens.Close == nil {
|
|
x.Tokens.Star.TrailingTrivia = t
|
|
return
|
|
}
|
|
x.Tokens.Close.TrailingTrivia = t
|
|
}
|
|
|
|
func (x *SplatExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *SplatExpression) print(w io.Writer, p *printer) {
|
|
isDot := x.Tokens.GetClose() == nil
|
|
|
|
p.fprintf(w, "%(%v%v%v", x.Tokens.GetParentheses(), x.Source, x.Tokens.GetOpen(), x.Tokens.GetStar())
|
|
if !isDot {
|
|
p.fprintf(w, "%v", x.Tokens.GetClose())
|
|
}
|
|
p.fprintf(w, "%v%)", x.Each, x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*SplatExpression) isExpression() {}
|
|
|
|
// TemplateExpression represents a semantically-analyzed template expression.
|
|
type TemplateExpression struct {
|
|
// The syntax node associated with the template expression.
|
|
Syntax *hclsyntax.TemplateExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.TemplateTokens
|
|
|
|
// The parts of the template expression.
|
|
Parts []Expression
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the template expression.
|
|
func (x *TemplateExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the template expression.
|
|
func (x *TemplateExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the template expression.
|
|
func (x *TemplateExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *TemplateExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
for _, part := range x.Parts {
|
|
partDiags := part.Typecheck(true)
|
|
diagnostics = append(diagnostics, partDiags...)
|
|
}
|
|
}
|
|
|
|
x.exprType = liftOperationType(StringType, x.Parts...)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *TemplateExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.TemplateExpr{
|
|
Parts: make([]hclsyntax.Expression, len(x.Parts)),
|
|
}
|
|
for i, p := range x.Parts {
|
|
syntax.Parts[i] = &syntaxExpr{expr: p}
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *TemplateExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *TemplateExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *TemplateExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetOpen())
|
|
}
|
|
|
|
func (x *TemplateExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewTemplateTokens()
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, &x.Tokens.Open, t)
|
|
}
|
|
|
|
func (x *TemplateExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetClose())
|
|
}
|
|
|
|
func (x *TemplateExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewTemplateTokens()
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.Parentheses, &x.Tokens.Close, t)
|
|
}
|
|
|
|
func (x *TemplateExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *TemplateExpression) print(w io.Writer, p *printer) {
|
|
// Print the opening quote.
|
|
p.fprintf(w, "%(%v", x.Tokens.GetParentheses(), x.Tokens.GetOpen())
|
|
|
|
isHeredoc := x.Tokens.GetOpen().Raw.Type == hclsyntax.TokenOHeredoc
|
|
|
|
// Print the expressions.
|
|
for _, part := range x.Parts {
|
|
if lit, ok := part.(*LiteralValueExpression); ok && StringType.AssignableFrom(lit.Type()) {
|
|
lit.printLit(w, p, !isHeredoc)
|
|
} else {
|
|
p.fprintf(w, "%v", part)
|
|
}
|
|
}
|
|
|
|
// Print the closing quote
|
|
p.fprintf(w, "%v%)", x.Tokens.GetClose(), x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*TemplateExpression) isExpression() {}
|
|
|
|
// TemplateJoinExpression represents a semantically-analyzed template join expression.
|
|
type TemplateJoinExpression struct {
|
|
// The syntax node associated with the template join expression.
|
|
Syntax *hclsyntax.TemplateJoinExpr
|
|
|
|
// The tuple being joined.
|
|
Tuple Expression
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the template join expression.
|
|
func (x *TemplateJoinExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the template join expression.
|
|
func (x *TemplateJoinExpression) NodeTokens() syntax.NodeTokens {
|
|
return nil
|
|
}
|
|
|
|
// Type returns the type of the template join expression.
|
|
func (x *TemplateJoinExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
tupleDiags := x.Tuple.Typecheck(true)
|
|
diagnostics = append(diagnostics, tupleDiags...)
|
|
}
|
|
|
|
x.exprType = liftOperationType(StringType, x.Tuple)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.TemplateJoinExpr{
|
|
Tuple: &syntaxExpr{expr: x.Tuple},
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) HasLeadingTrivia() bool {
|
|
return x.Tuple.HasLeadingTrivia()
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) HasTrailingTrivia() bool {
|
|
return x.Tuple.HasTrailingTrivia()
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return x.Tuple.GetLeadingTrivia()
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
x.Tuple.SetLeadingTrivia(t)
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return x.Tuple.GetTrailingTrivia()
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
x.Tuple.SetTrailingTrivia(t)
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *TemplateJoinExpression) print(w io.Writer, p *printer) {
|
|
p.fprintf(w, "%v", x.Tuple)
|
|
}
|
|
|
|
func (*TemplateJoinExpression) isExpression() {}
|
|
|
|
// TupleConsExpression represents a semantically-analyzed tuple construction expression.
|
|
type TupleConsExpression struct {
|
|
// The syntax node associated with the tuple construction expression.
|
|
Syntax *hclsyntax.TupleConsExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.TupleConsTokens
|
|
|
|
// The elements of the tuple.
|
|
Expressions []Expression
|
|
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the tuple construction expression.
|
|
func (x *TupleConsExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the tuple construction expression.
|
|
func (x *TupleConsExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// Type returns the type of the tuple construction expression.
|
|
func (x *TupleConsExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *TupleConsExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
elementTypes := make([]Type, len(x.Expressions))
|
|
for i, expr := range x.Expressions {
|
|
if typecheckOperands {
|
|
exprDiags := expr.Typecheck(true)
|
|
diagnostics = append(diagnostics, exprDiags...)
|
|
}
|
|
|
|
elementTypes[i] = expr.Type()
|
|
}
|
|
|
|
x.exprType = NewTupleType(elementTypes...)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *TupleConsExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.TupleConsExpr{
|
|
Exprs: make([]hclsyntax.Expression, len(x.Expressions)),
|
|
}
|
|
for i, x := range x.Expressions {
|
|
syntax.Exprs[i] = &syntaxExpr{expr: x}
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *TupleConsExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *TupleConsExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *TupleConsExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetOpenBracket())
|
|
}
|
|
|
|
func (x *TupleConsExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewTupleConsTokens(len(x.Expressions))
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, &x.Tokens.OpenBracket, t)
|
|
}
|
|
|
|
func (x *TupleConsExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetCloseBracket())
|
|
}
|
|
|
|
func (x *TupleConsExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewTupleConsTokens(len(x.Expressions))
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.Parentheses, &x.Tokens.CloseBracket, t)
|
|
}
|
|
|
|
func (x *TupleConsExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *TupleConsExpression) print(w io.Writer, p *printer) {
|
|
// Print the opening bracket.
|
|
p.fprintf(w, "%(%v", x.Tokens.GetParentheses(), x.Tokens.GetOpenBracket())
|
|
|
|
// Print each element and its comma.
|
|
commas := x.Tokens.GetCommas(len(x.Expressions))
|
|
p.indented(func() {
|
|
for i, expr := range x.Expressions {
|
|
if !expr.HasLeadingTrivia() {
|
|
p.fprintf(w, "\n%s", p.indent)
|
|
}
|
|
p.fprintf(w, "%v", expr)
|
|
|
|
if i != len(x.Expressions)-1 {
|
|
var comma syntax.Token
|
|
if i < len(commas) {
|
|
comma = commas[i]
|
|
}
|
|
p.fprintf(w, "%v", comma)
|
|
}
|
|
}
|
|
|
|
// If there were commas left over, print the trivia for each.
|
|
//
|
|
// TODO(pdg): filter to only comments?
|
|
if len(x.Expressions) > 0 && len(x.Expressions)-1 <= len(commas) {
|
|
for _, comma := range commas[len(x.Expressions)-1:] {
|
|
p.fprintf(w, "%v", comma.AllTrivia().CollapseWhitespace())
|
|
}
|
|
}
|
|
})
|
|
|
|
// Print the closing bracket.
|
|
if x.Tokens != nil {
|
|
p.fprintf(w, "%v%)", x.Tokens.CloseBracket, x.Tokens.GetParentheses())
|
|
} else {
|
|
p.fprintf(w, "\n%s]", p.indent)
|
|
}
|
|
}
|
|
|
|
func (*TupleConsExpression) isExpression() {}
|
|
|
|
// UnaryOpExpression represents a semantically-analyzed unary operation.
|
|
type UnaryOpExpression struct {
|
|
// The syntax node associated with the unary operation.
|
|
Syntax *hclsyntax.UnaryOpExpr
|
|
// The tokens associated with the expression, if any.
|
|
Tokens *syntax.UnaryOpTokens
|
|
|
|
// The operation.
|
|
Operation *hclsyntax.Operation
|
|
// The operand of the operation.
|
|
Operand Expression
|
|
|
|
operandType Type
|
|
exprType Type
|
|
}
|
|
|
|
// SyntaxNode returns the syntax node associated with the unary operation.
|
|
func (x *UnaryOpExpression) SyntaxNode() hclsyntax.Node {
|
|
if x.Syntax == nil {
|
|
return syntax.None
|
|
}
|
|
return x.Syntax
|
|
}
|
|
|
|
// NodeTokens returns the tokens associated with the unary operation.
|
|
func (x *UnaryOpExpression) NodeTokens() syntax.NodeTokens {
|
|
return x.Tokens
|
|
}
|
|
|
|
// OperandType returns the operand type of the unary operation.
|
|
func (x *UnaryOpExpression) OperandType() Type {
|
|
return x.operandType
|
|
}
|
|
|
|
// Type returns the type of the unary operation.
|
|
func (x *UnaryOpExpression) Type() Type {
|
|
return x.exprType
|
|
}
|
|
|
|
func (x *UnaryOpExpression) Typecheck(typecheckOperands bool) hcl.Diagnostics {
|
|
var diagnostics hcl.Diagnostics
|
|
|
|
if typecheckOperands {
|
|
operandDiags := x.Operand.Typecheck(true)
|
|
diagnostics = append(diagnostics, operandDiags...)
|
|
}
|
|
|
|
// Compute the signature for the operator and typecheck the arguments.
|
|
signature := getOperationSignature(x.Operation, false)
|
|
contract.Assertf(len(signature.Parameters) == 1,
|
|
"expected unary operator signature to have 1 parameter, got %d", len(signature.Parameters))
|
|
|
|
x.operandType = signature.Parameters[0].Type
|
|
|
|
var rng hcl.Range
|
|
if x.Syntax != nil {
|
|
rng = x.Syntax.Range()
|
|
}
|
|
typecheckDiags := typecheckArgs(rng, signature, x.Operand)
|
|
diagnostics = append(diagnostics, typecheckDiags...)
|
|
|
|
x.exprType = liftOperationType(signature.ReturnType, x.Operand)
|
|
return diagnostics
|
|
}
|
|
|
|
func (x *UnaryOpExpression) Evaluate(context *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
|
syntax := &hclsyntax.UnaryOpExpr{
|
|
Op: x.Operation,
|
|
Val: &syntaxExpr{expr: x.Operand},
|
|
}
|
|
return syntax.Value(context)
|
|
}
|
|
|
|
func (x *UnaryOpExpression) HasLeadingTrivia() bool {
|
|
return exprHasLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens != nil)
|
|
}
|
|
|
|
func (x *UnaryOpExpression) HasTrailingTrivia() bool {
|
|
return exprHasTrailingTrivia(x.Tokens.GetParentheses(), x.Operand)
|
|
}
|
|
|
|
func (x *UnaryOpExpression) GetLeadingTrivia() syntax.TriviaList {
|
|
return getExprLeadingTrivia(x.Tokens.GetParentheses(), x.Tokens.GetOperator(x.Operation))
|
|
}
|
|
|
|
func (x *UnaryOpExpression) SetLeadingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewUnaryOpTokens(x.Operation)
|
|
}
|
|
setExprLeadingTrivia(x.Tokens.Parentheses, &x.Tokens.Operator, t)
|
|
}
|
|
|
|
func (x *UnaryOpExpression) GetTrailingTrivia() syntax.TriviaList {
|
|
return getExprTrailingTrivia(x.Tokens.GetParentheses(), x.Operand)
|
|
}
|
|
|
|
func (x *UnaryOpExpression) SetTrailingTrivia(t syntax.TriviaList) {
|
|
if x.Tokens == nil {
|
|
x.Tokens = syntax.NewUnaryOpTokens(x.Operation)
|
|
}
|
|
setExprTrailingTrivia(x.Tokens.Parentheses, x.Operand, t)
|
|
}
|
|
|
|
func (x *UnaryOpExpression) Format(f fmt.State, c rune) {
|
|
x.print(f, &printer{})
|
|
}
|
|
|
|
func (x *UnaryOpExpression) print(w io.Writer, p *printer) {
|
|
precedence := operatorPrecedence(x.Operation)
|
|
p.fprintf(w, "%[2](%[3]v%.[1]*[4]v%[5])",
|
|
precedence,
|
|
x.Tokens.GetParentheses(),
|
|
x.Tokens.GetOperator(x.Operation), x.Operand,
|
|
x.Tokens.GetParentheses())
|
|
}
|
|
|
|
func (*UnaryOpExpression) isExpression() {}
|