Skip to content

Commit fadd95c

Browse files
committed
unknown upgraded splat values may have no elems
When upgrading an unknown splat value, the resulting collection may have 0 elements if the value ends up being `null`. This means the type must be dynamic.
1 parent a497928 commit fadd95c

File tree

2 files changed

+24
-8
lines changed

2 files changed

+24
-8
lines changed

hclsyntax/expression.go

+9-5
Original file line numberDiff line numberDiff line change
@@ -1439,10 +1439,10 @@ func (e *SplatExpr) Value(ctx *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
14391439
// upgrade to a different number of elements depending on whether
14401440
// sourceVal becomes null or not.
14411441
// We record this condition here so we can process any remaining
1442-
// expression after the * to derive the correct type. For example, it
1443-
// is valid to use a splat on a single object to retrieve a list of a
1444-
// single attribute, which means the final expression type still needs
1445-
// to be determined.
1442+
// expression after the * to verify the result of the traversal. For
1443+
// example, it is valid to use a splat on a single object to retrieve a
1444+
// list of a single attribute, but we still need to check if that
1445+
// attribute actually exists.
14461446
upgradedUnknown = !sourceVal.IsKnown()
14471447

14481448
sourceVal = cty.TupleVal([]cty.Value{sourceVal})
@@ -1512,7 +1512,11 @@ func (e *SplatExpr) Value(ctx *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
15121512
}
15131513
e.Item.clearValue(ctx) // clean up our temporary value
15141514

1515-
if !isKnown || upgradedUnknown {
1515+
if upgradedUnknown {
1516+
return cty.DynamicVal, diags
1517+
}
1518+
1519+
if !isKnown {
15161520
// We'll ingore the resultTy diagnostics in this case since they
15171521
// will just be the same errors we saw while iterating above.
15181522
ty, _ := resultTy()

hclsyntax/expression_test.go

+15-3
Original file line numberDiff line numberDiff line change
@@ -1142,7 +1142,7 @@ upper(
11421142
"unkstr": cty.UnknownVal(cty.String),
11431143
},
11441144
},
1145-
cty.UnknownVal(cty.Tuple([]cty.Type{cty.String})),
1145+
cty.DynamicVal,
11461146
0,
11471147
},
11481148
{
@@ -1152,7 +1152,7 @@ upper(
11521152
"unkstr": cty.UnknownVal(cty.String),
11531153
},
11541154
},
1155-
cty.UnknownVal(cty.Tuple([]cty.Type{cty.DynamicPseudoType})),
1155+
cty.DynamicVal,
11561156
1, // a string has no attribute "name"
11571157
},
11581158
{
@@ -1174,7 +1174,19 @@ upper(
11741174
})),
11751175
},
11761176
},
1177-
cty.UnknownVal(cty.Tuple([]cty.Type{cty.String})),
1177+
cty.DynamicVal,
1178+
0,
1179+
},
1180+
{
1181+
`unkobj.*.names`,
1182+
&hcl.EvalContext{
1183+
Variables: map[string]cty.Value{
1184+
"unkobj": cty.UnknownVal(cty.Object(map[string]cty.Type{
1185+
"names": cty.List(cty.String),
1186+
})),
1187+
},
1188+
},
1189+
cty.DynamicVal,
11781190
0,
11791191
},
11801192
{

0 commit comments

Comments
 (0)