2020-11-03 23:48:43 +00:00
|
|
|
package yqlib
|
2020-09-17 11:58:01 +00:00
|
|
|
|
|
|
|
import (
|
2020-10-17 11:10:47 +00:00
|
|
|
"fmt"
|
2020-09-17 11:58:01 +00:00
|
|
|
"testing"
|
|
|
|
|
2020-10-27 05:45:16 +00:00
|
|
|
"github.com/mikefarah/yq/v4/test"
|
2020-09-17 11:58:01 +00:00
|
|
|
)
|
|
|
|
|
2021-11-30 02:09:15 +00:00
|
|
|
var variableWithNewLine = `"cat
|
|
|
|
"`
|
2021-06-11 22:26:27 +00:00
|
|
|
|
2020-10-17 11:10:47 +00:00
|
|
|
var pathTests = []struct {
|
|
|
|
path string
|
|
|
|
expectedTokens []interface{}
|
|
|
|
expectedPostFix []interface{}
|
2020-12-01 07:08:41 +00:00
|
|
|
}{
|
2021-11-30 02:04:32 +00:00
|
|
|
{
|
|
|
|
`[.a, .b]`,
|
|
|
|
append(make([]interface{}, 0), "[", "a", "UNION", "b", "]"),
|
|
|
|
append(make([]interface{}, 0), "a", "b", "UNION", "COLLECT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.[env(myenv)]`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "TRAVERSE_ARRAY", "[", "ENV", "]"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "ENV", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
2021-10-18 00:43:06 +00:00
|
|
|
{
|
|
|
|
`.["cat"].["dog"]`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "TRAVERSE_ARRAY", "[", "cat (string)", "]", "SHORT_PIPE", "SELF", "TRAVERSE_ARRAY", "[", "dog (string)", "]"),
|
2021-11-30 02:04:32 +00:00
|
|
|
append(make([]interface{}, 0), "SELF", "cat (string)", "COLLECT", "TRAVERSE_ARRAY", "SELF", "dog (string)", "COLLECT", "TRAVERSE_ARRAY", "SHORT_PIPE"),
|
2021-10-18 00:43:06 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
`.["cat"]`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "TRAVERSE_ARRAY", "[", "cat (string)", "]"),
|
2021-11-30 02:04:32 +00:00
|
|
|
append(make([]interface{}, 0), "SELF", "cat (string)", "COLLECT", "TRAVERSE_ARRAY"),
|
2021-10-18 00:43:06 +00:00
|
|
|
},
|
2021-09-15 12:24:03 +00:00
|
|
|
{
|
|
|
|
"with(.a;.=3)",
|
|
|
|
append(make([]interface{}, 0), "WITH", "(", "a", "BLOCK", "SELF", "ASSIGN", "3 (int64)", ")"),
|
|
|
|
append(make([]interface{}, 0), "a", "SELF", "3 (int64)", "ASSIGN", "BLOCK", "WITH"),
|
|
|
|
},
|
2021-11-30 02:09:15 +00:00
|
|
|
{
|
|
|
|
"0x12",
|
|
|
|
append(make([]interface{}, 0), "18 (int64)"),
|
|
|
|
append(make([]interface{}, 0), "18 (int64)"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"0X12",
|
|
|
|
append(make([]interface{}, 0), "18 (int64)"),
|
|
|
|
append(make([]interface{}, 0), "18 (int64)"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
".a\n",
|
|
|
|
append(make([]interface{}, 0), "a"),
|
|
|
|
append(make([]interface{}, 0), "a"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variableWithNewLine,
|
|
|
|
append(make([]interface{}, 0), "cat\n (string)"),
|
|
|
|
append(make([]interface{}, 0), "cat\n (string)"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.[0]`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "TRAVERSE_ARRAY", "[", "0 (int64)", "]"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "0 (int64)", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.[0][1]`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "TRAVERSE_ARRAY", "[", "0 (int64)", "]", "TRAVERSE_ARRAY", "[", "1 (int64)", "]"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "0 (int64)", "COLLECT", "TRAVERSE_ARRAY", "1 (int64)", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"\""`,
|
|
|
|
append(make([]interface{}, 0), "\" (string)"),
|
|
|
|
append(make([]interface{}, 0), "\" (string)"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`[]|join(".")`,
|
|
|
|
append(make([]interface{}, 0), "[", "EMPTY", "]", "PIPE", "JOIN", "(", ". (string)", ")"),
|
|
|
|
append(make([]interface{}, 0), "EMPTY", "COLLECT", ". (string)", "JOIN", "PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`{"cool": .b or .c}`,
|
|
|
|
append(make([]interface{}, 0), "{", "cool (string)", "CREATE_MAP", "b", "OR", "c", "}"),
|
|
|
|
append(make([]interface{}, 0), "cool (string)", "b", "c", "OR", "CREATE_MAP", "COLLECT_OBJECT", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`{"cool": []|join(".")}`,
|
|
|
|
append(make([]interface{}, 0), "{", "cool (string)", "CREATE_MAP", "[", "EMPTY", "]", "PIPE", "JOIN", "(", ". (string)", ")", "}"),
|
|
|
|
append(make([]interface{}, 0), "cool (string)", "EMPTY", "COLLECT", ". (string)", "JOIN", "PIPE", "CREATE_MAP", "COLLECT_OBJECT", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a as $item ireduce (0; . + $item)`, // note - add code to shuffle reduce to this position for postfix
|
|
|
|
append(make([]interface{}, 0), "a", "ASSIGN_VARIABLE", "GET_VARIABLE", "REDUCE", "(", "0 (int64)", "BLOCK", "SELF", "ADD", "GET_VARIABLE", ")"),
|
|
|
|
append(make([]interface{}, 0), "a", "GET_VARIABLE", "ASSIGN_VARIABLE", "0 (int64)", "SELF", "GET_VARIABLE", "ADD", "BLOCK", "REDUCE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a | .b | .c`,
|
|
|
|
append(make([]interface{}, 0), "a", "PIPE", "b", "PIPE", "c"),
|
|
|
|
append(make([]interface{}, 0), "a", "b", "c", "PIPE", "PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`[]`,
|
|
|
|
append(make([]interface{}, 0), "[", "EMPTY", "]"),
|
|
|
|
append(make([]interface{}, 0), "EMPTY", "COLLECT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`{}`,
|
|
|
|
append(make([]interface{}, 0), "{", "EMPTY", "}"),
|
|
|
|
append(make([]interface{}, 0), "EMPTY", "COLLECT_OBJECT", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`[{}]`,
|
|
|
|
append(make([]interface{}, 0), "[", "{", "EMPTY", "}", "]"),
|
|
|
|
append(make([]interface{}, 0), "EMPTY", "COLLECT_OBJECT", "SHORT_PIPE", "COLLECT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.realnames as $names | $names["anon"]`,
|
|
|
|
append(make([]interface{}, 0), "realnames", "ASSIGN_VARIABLE", "GET_VARIABLE", "PIPE", "GET_VARIABLE", "TRAVERSE_ARRAY", "[", "anon (string)", "]"),
|
|
|
|
append(make([]interface{}, 0), "realnames", "GET_VARIABLE", "ASSIGN_VARIABLE", "GET_VARIABLE", "anon (string)", "COLLECT", "TRAVERSE_ARRAY", "PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.b[.a]`,
|
|
|
|
append(make([]interface{}, 0), "b", "TRAVERSE_ARRAY", "[", "a", "]"),
|
|
|
|
append(make([]interface{}, 0), "b", "a", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.b[.a]?`,
|
|
|
|
append(make([]interface{}, 0), "b", "TRAVERSE_ARRAY", "[", "a", "]"),
|
|
|
|
append(make([]interface{}, 0), "b", "a", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.[]`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "TRAVERSE_ARRAY", "[", "EMPTY", "]"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "EMPTY", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a[]`,
|
|
|
|
append(make([]interface{}, 0), "a", "TRAVERSE_ARRAY", "[", "EMPTY", "]"),
|
|
|
|
append(make([]interface{}, 0), "a", "EMPTY", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a[]?`,
|
|
|
|
append(make([]interface{}, 0), "a", "TRAVERSE_ARRAY", "[", "EMPTY", "]"),
|
|
|
|
append(make([]interface{}, 0), "a", "EMPTY", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a.[]`,
|
|
|
|
append(make([]interface{}, 0), "a", "SHORT_PIPE", "SELF", "TRAVERSE_ARRAY", "[", "EMPTY", "]"),
|
|
|
|
append(make([]interface{}, 0), "a", "SELF", "EMPTY", "COLLECT", "TRAVERSE_ARRAY", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a[0]`,
|
|
|
|
append(make([]interface{}, 0), "a", "TRAVERSE_ARRAY", "[", "0 (int64)", "]"),
|
|
|
|
append(make([]interface{}, 0), "a", "0 (int64)", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a[0]?`,
|
|
|
|
append(make([]interface{}, 0), "a", "TRAVERSE_ARRAY", "[", "0 (int64)", "]"),
|
|
|
|
append(make([]interface{}, 0), "a", "0 (int64)", "COLLECT", "TRAVERSE_ARRAY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a.[0]`,
|
|
|
|
append(make([]interface{}, 0), "a", "SHORT_PIPE", "SELF", "TRAVERSE_ARRAY", "[", "0 (int64)", "]"),
|
|
|
|
append(make([]interface{}, 0), "a", "SELF", "0 (int64)", "COLLECT", "TRAVERSE_ARRAY", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a[].c`,
|
|
|
|
append(make([]interface{}, 0), "a", "TRAVERSE_ARRAY", "[", "EMPTY", "]", "SHORT_PIPE", "c"),
|
|
|
|
append(make([]interface{}, 0), "a", "EMPTY", "COLLECT", "TRAVERSE_ARRAY", "c", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`[3]`,
|
|
|
|
append(make([]interface{}, 0), "[", "3 (int64)", "]"),
|
|
|
|
append(make([]interface{}, 0), "3 (int64)", "COLLECT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.key.array + .key.array2`,
|
|
|
|
append(make([]interface{}, 0), "key", "SHORT_PIPE", "array", "ADD", "key", "SHORT_PIPE", "array2"),
|
|
|
|
append(make([]interface{}, 0), "key", "array", "SHORT_PIPE", "key", "array2", "SHORT_PIPE", "ADD"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.key.array * .key.array2`,
|
|
|
|
append(make([]interface{}, 0), "key", "SHORT_PIPE", "array", "MULTIPLY", "key", "SHORT_PIPE", "array2"),
|
|
|
|
append(make([]interface{}, 0), "key", "array", "SHORT_PIPE", "key", "array2", "SHORT_PIPE", "MULTIPLY"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.key.array // .key.array2`,
|
|
|
|
append(make([]interface{}, 0), "key", "SHORT_PIPE", "array", "ALTERNATIVE", "key", "SHORT_PIPE", "array2"),
|
|
|
|
append(make([]interface{}, 0), "key", "array", "SHORT_PIPE", "key", "array2", "SHORT_PIPE", "ALTERNATIVE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a | .[].b == "apple"`,
|
|
|
|
append(make([]interface{}, 0), "a", "PIPE", "SELF", "TRAVERSE_ARRAY", "[", "EMPTY", "]", "SHORT_PIPE", "b", "EQUALS", "apple (string)"),
|
|
|
|
append(make([]interface{}, 0), "a", "SELF", "EMPTY", "COLLECT", "TRAVERSE_ARRAY", "b", "SHORT_PIPE", "apple (string)", "EQUALS", "PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`(.a | .[].b) == "apple"`,
|
|
|
|
append(make([]interface{}, 0), "(", "a", "PIPE", "SELF", "TRAVERSE_ARRAY", "[", "EMPTY", "]", "SHORT_PIPE", "b", ")", "EQUALS", "apple (string)"),
|
|
|
|
append(make([]interface{}, 0), "a", "SELF", "EMPTY", "COLLECT", "TRAVERSE_ARRAY", "b", "SHORT_PIPE", "PIPE", "apple (string)", "EQUALS"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.[] | select(. == "*at")`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "TRAVERSE_ARRAY", "[", "EMPTY", "]", "PIPE", "SELECT", "(", "SELF", "EQUALS", "*at (string)", ")"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "EMPTY", "COLLECT", "TRAVERSE_ARRAY", "SELF", "*at (string)", "EQUALS", "SELECT", "PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`[true]`,
|
|
|
|
append(make([]interface{}, 0), "[", "true (bool)", "]"),
|
|
|
|
append(make([]interface{}, 0), "true (bool)", "COLLECT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`[true, false]`,
|
|
|
|
append(make([]interface{}, 0), "[", "true (bool)", "UNION", "false (bool)", "]"),
|
|
|
|
append(make([]interface{}, 0), "true (bool)", "false (bool)", "UNION", "COLLECT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"mike": .a`,
|
|
|
|
append(make([]interface{}, 0), "mike (string)", "CREATE_MAP", "a"),
|
|
|
|
append(make([]interface{}, 0), "mike (string)", "a", "CREATE_MAP"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a: "mike"`,
|
|
|
|
append(make([]interface{}, 0), "a", "CREATE_MAP", "mike (string)"),
|
|
|
|
append(make([]interface{}, 0), "a", "mike (string)", "CREATE_MAP"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`{"mike": .a}`,
|
|
|
|
append(make([]interface{}, 0), "{", "mike (string)", "CREATE_MAP", "a", "}"),
|
|
|
|
append(make([]interface{}, 0), "mike (string)", "a", "CREATE_MAP", "COLLECT_OBJECT", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`{.a: "mike"}`,
|
|
|
|
append(make([]interface{}, 0), "{", "a", "CREATE_MAP", "mike (string)", "}"),
|
|
|
|
append(make([]interface{}, 0), "a", "mike (string)", "CREATE_MAP", "COLLECT_OBJECT", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`{.a: .c, .b.[]: .f.g[]}`,
|
|
|
|
append(make([]interface{}, 0), "{", "a", "CREATE_MAP", "c", "UNION", "b", "SHORT_PIPE", "SELF", "TRAVERSE_ARRAY", "[", "EMPTY", "]", "CREATE_MAP", "f", "SHORT_PIPE", "g", "TRAVERSE_ARRAY", "[", "EMPTY", "]", "}"),
|
|
|
|
append(make([]interface{}, 0), "a", "c", "CREATE_MAP", "b", "SELF", "EMPTY", "COLLECT", "TRAVERSE_ARRAY", "SHORT_PIPE", "f", "g", "EMPTY", "COLLECT", "TRAVERSE_ARRAY", "SHORT_PIPE", "CREATE_MAP", "UNION", "COLLECT_OBJECT", "SHORT_PIPE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`explode(.a.b)`,
|
|
|
|
append(make([]interface{}, 0), "EXPLODE", "(", "a", "SHORT_PIPE", "b", ")"),
|
|
|
|
append(make([]interface{}, 0), "a", "b", "SHORT_PIPE", "EXPLODE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a.b style="folded"`,
|
|
|
|
append(make([]interface{}, 0), "a", "SHORT_PIPE", "b", "ASSIGN_STYLE", "folded (string)"),
|
|
|
|
append(make([]interface{}, 0), "a", "b", "SHORT_PIPE", "folded (string)", "ASSIGN_STYLE"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`tag == "str"`,
|
|
|
|
append(make([]interface{}, 0), "GET_TAG", "EQUALS", "str (string)"),
|
|
|
|
append(make([]interface{}, 0), "GET_TAG", "str (string)", "EQUALS"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`. tag= "str"`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "ASSIGN_TAG", "str (string)"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "str (string)", "ASSIGN_TAG"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`lineComment == "str"`,
|
|
|
|
append(make([]interface{}, 0), "GET_COMMENT", "EQUALS", "str (string)"),
|
|
|
|
append(make([]interface{}, 0), "GET_COMMENT", "str (string)", "EQUALS"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`. lineComment= "str"`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "ASSIGN_COMMENT", "str (string)"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "str (string)", "ASSIGN_COMMENT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`. lineComment |= "str"`,
|
|
|
|
append(make([]interface{}, 0), "SELF", "ASSIGN_COMMENT", "str (string)"),
|
|
|
|
append(make([]interface{}, 0), "SELF", "str (string)", "ASSIGN_COMMENT"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.a.b tag="!!str"`,
|
|
|
|
append(make([]interface{}, 0), "a", "SHORT_PIPE", "b", "ASSIGN_TAG", "!!str (string)"),
|
|
|
|
append(make([]interface{}, 0), "a", "b", "SHORT_PIPE", "!!str (string)", "ASSIGN_TAG"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`""`,
|
|
|
|
append(make([]interface{}, 0), " (string)"),
|
|
|
|
append(make([]interface{}, 0), " (string)"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`.foo* | (. style="flow")`,
|
|
|
|
append(make([]interface{}, 0), "foo*", "PIPE", "(", "SELF", "ASSIGN_STYLE", "flow (string)", ")"),
|
|
|
|
append(make([]interface{}, 0), "foo*", "SELF", "flow (string)", "ASSIGN_STYLE", "PIPE"),
|
|
|
|
},
|
2020-09-17 11:58:01 +00:00
|
|
|
}
|
|
|
|
|
2021-01-12 23:18:53 +00:00
|
|
|
var tokeniser = newExpressionTokeniser()
|
|
|
|
var postFixer = newExpressionPostFixer()
|
2020-09-17 11:58:01 +00:00
|
|
|
|
2020-10-17 11:10:47 +00:00
|
|
|
func TestPathParsing(t *testing.T) {
|
|
|
|
for _, tt := range pathTests {
|
2020-09-17 11:58:01 +00:00
|
|
|
tokens, err := tokeniser.Tokenise(tt.path)
|
|
|
|
if err != nil {
|
|
|
|
t.Error(tt.path, err)
|
|
|
|
}
|
|
|
|
var tokenValues []interface{}
|
|
|
|
for _, token := range tokens {
|
2021-02-03 06:11:47 +00:00
|
|
|
tokenValues = append(tokenValues, token.toString(false))
|
2020-09-17 11:58:01 +00:00
|
|
|
}
|
2020-10-17 11:10:47 +00:00
|
|
|
test.AssertResultComplexWithContext(t, tt.expectedTokens, tokenValues, fmt.Sprintf("tokenise: %v", tt.path))
|
|
|
|
|
|
|
|
results, errorP := postFixer.ConvertToPostfix(tokens)
|
|
|
|
|
|
|
|
var readableResults []interface{}
|
|
|
|
for _, token := range results {
|
|
|
|
readableResults = append(readableResults, token.toString())
|
|
|
|
}
|
|
|
|
|
|
|
|
if errorP != nil {
|
|
|
|
t.Error(tt.path, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
test.AssertResultComplexWithContext(t, tt.expectedPostFix, readableResults, fmt.Sprintf("postfix: %v", tt.path))
|
|
|
|
|
2020-09-17 11:58:01 +00:00
|
|
|
}
|
|
|
|
}
|