Skip to content

Commit 1c79f57

Browse files
rudo-thomasjohnbartholomew
authored andcommitted
fix: Fix error messages when a comprehension iterates over a non-array.
Problems: - When iterating over an empty string in a list comprehension, the result is an empty string. This is a bug, it should be an error. - When iterating over a non-empty string in a list comprehension, the expected and unexpected types in the error message are swapped. - Error messages mention "std.flatMap" when object/list comprehensions would iterate over a value that is neither array nor string. ``` $ jsonnet --version Jsonnet commandline interpreter (Go implementation) v0.21.0-rc2 $ jsonnet -e '[a for a in ""]' "" $ jsonnet -e '[a for a in "b"]' RUNTIME ERROR: Unexpected type array, expected string <cmdline>:1:1-17 During evaluation $ jsonnet -e '{[a]: 1 for a in 2}' RUNTIME ERROR: std.flatMap second param must be array / string, got number <cmdline>:1:1-20 <cmdline>:1:1-20 During evaluation $ jsonnet -e '[a for a in 1]' RUNTIME ERROR: std.flatMap second param must be array / string, got number <cmdline>:1:1-15 During evaluation ``` FWIW, the C++ implementation does not have any of these problems. It gives: ``` RUNTIME ERROR: In comprehension, can only iterate over array. ``` In the Go implementation comprehensions are desugared to a call to std.flatMap which does accept a string in the "arr" parameter. The fix: Desugar comprehensions to a call to a new hidden builtin which only accepts arrays.
1 parent 1add1e1 commit 1c79f57

16 files changed

+71
-2
lines changed

builtins.go

+14
Original file line numberDiff line numberDiff line change
@@ -345,6 +345,19 @@ func builtinFlatMap(i *interpreter, funcv, arrv value) (value, error) {
345345
}
346346
}
347347

348+
// builtinFlatMapArray is like builtinFlatMap, but only accepts array as the
349+
// arrv value. Desugared comprehensions contain a call to this function, rather
350+
// than builtinFlatMap, so that a better error message is printed when the
351+
// comprehension would iterate over a non-array.
352+
func builtinFlatMapArray(i *interpreter, funcv, arrv value) (value, error) {
353+
switch arrv := arrv.(type) {
354+
case *valueArray:
355+
return builtinFlatMap(i, funcv, arrv)
356+
default:
357+
return nil, i.typeErrorSpecific(arrv, &valueArray{})
358+
}
359+
}
360+
348361
func joinArrays(i *interpreter, sep *valueArray, arr *valueArray) (value, error) {
349362
result := make([]*cachedThunk, 0, arr.length())
350363
first := true
@@ -2880,4 +2893,5 @@ var funcBuiltins = buildBuiltinMap([]builtin{
28802893

28812894
// internal
28822895
&unaryBuiltin{name: "$objectFlatMerge", function: builtinUglyObjectFlatMerge, params: ast.Identifiers{"x"}},
2896+
&binaryBuiltin{name: "$flatMapArray", function: builtinFlatMapArray, params: ast.Identifiers{"func", "arr"}},
28832897
})

internal/program/desugarer.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -184,7 +184,7 @@ func desugarForSpec(inside ast.Node, loc ast.LocationRange, forSpec *ast.ForSpec
184184
if err != nil {
185185
return nil, err
186186
}
187-
current := buildStdCall("flatMap", loc, function, forSpec.Expr)
187+
current := buildStdCall("$flatMapArray", loc, function, forSpec.Expr)
188188
if forSpec.Outer == nil {
189189
return current, nil
190190
}

linter/internal/types/stdlib.go

+1
Original file line numberDiff line numberDiff line change
@@ -199,6 +199,7 @@ func prepareStdlib(g *typeGraph) {
199199
"mod": g.newSimpleFuncType(stringOrNumber, "a", "b"),
200200
"native": g.newSimpleFuncType(anyFunctionType, "x"),
201201
"$objectFlatMerge": g.newSimpleFuncType(anyObjectType, "x"),
202+
"$flatMapArray": g.newSimpleFuncType(anyArrayType, "func", "arr"),
202203

203204
// Boolean
204205

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
RUNTIME ERROR: Unexpected type string, expected array
2+
-------------------------------------------------
3+
testdata/array_comp_try_iterate_over_empty_string:1:1-16
4+
5+
[a for a in '']
6+
7+
-------------------------------------------------
8+
During evaluation
9+
10+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
[a for a in '']

testdata/array_comp_try_iterate_over_empty_string.linter.golden

Whitespace-only changes.

testdata/array_comp_try_iterate_over_obj.golden

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
RUNTIME ERROR: std.flatMap second param must be array / string, got object
1+
RUNTIME ERROR: Unexpected type object, expected array
22
-------------------------------------------------
33
testdata/array_comp_try_iterate_over_obj:1:1-16
44

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
RUNTIME ERROR: Unexpected type string, expected array
2+
-------------------------------------------------
3+
testdata/array_comp_try_iterate_over_string:1:1-17
4+
5+
[a for a in 'b']
6+
7+
-------------------------------------------------
8+
During evaluation
9+
10+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
[a for a in 'b']

testdata/array_comp_try_iterate_over_string.linter.golden

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
RUNTIME ERROR: Unexpected type object, expected array
2+
-------------------------------------------------
3+
testdata/object_comp_try_iterate_over_obj:1:1-23
4+
5+
{ [a]: 0 for a in {} }
6+
7+
-------------------------------------------------
8+
testdata/object_comp_try_iterate_over_obj:1:1-23
9+
10+
{ [a]: 0 for a in {} }
11+
12+
-------------------------------------------------
13+
During evaluation
14+
15+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
{ [a]: 0 for a in {} }

testdata/object_comp_try_iterate_over_obj.linter.golden

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
RUNTIME ERROR: Unexpected type string, expected array
2+
-------------------------------------------------
3+
testdata/object_comp_try_iterate_over_string:1:1-24
4+
5+
{ [a]: 0 for a in 'b' }
6+
7+
-------------------------------------------------
8+
testdata/object_comp_try_iterate_over_string:1:1-24
9+
10+
{ [a]: 0 for a in 'b' }
11+
12+
-------------------------------------------------
13+
During evaluation
14+
15+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
{ [a]: 0 for a in 'b' }

testdata/object_comp_try_iterate_over_string.linter.golden

Whitespace-only changes.

0 commit comments

Comments
 (0)