From 76ae022b0764b521475c44a5e34e50ca8a38fff0 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Wed, 21 Dec 2022 08:12:23 +0100 Subject: [PATCH 01/15] feat: add coll.JQ using gojq library --- coll/jq.go | 36 +++++++++++++++++++++++ docs-src/content/functions/coll.yml | 31 ++++++++++++++++++++ docs/content/functions/coll.md | 45 +++++++++++++++++++++++++++++ funcs/coll.go | 6 ++++ go.mod | 4 ++- go.sum | 8 ++++- 6 files changed, 128 insertions(+), 2 deletions(-) create mode 100644 coll/jq.go diff --git a/coll/jq.go b/coll/jq.go new file mode 100644 index 000000000..5d07ac4b1 --- /dev/null +++ b/coll/jq.go @@ -0,0 +1,36 @@ +package coll + +import ( + "github.com/itchyny/gojq" + "github.com/pkg/errors" +) + +// JQ - +func JQ(jqExpr string, in interface{}) (interface{}, error) { + query, err := gojq.Parse(jqExpr) + if err != nil { + return nil, errors.Wrapf(err, "couldn't parse JQ %s", jqExpr) + } + iter := query.Run(in) + var out interface{} + a := []interface{}{} + for { + v, ok := iter.Next() + if !ok { + break + } + if err, ok := v.(error); ok { + return nil, errors.Wrap(err, "executing JQ failed") + } + if v != nil { + a = append(a, v) + } + } + if len(a) == 1 { + out = a[0] + } else { + out = a + } + + return out, nil +} diff --git a/docs-src/content/functions/coll.yml b/docs-src/content/functions/coll.yml index eedfd57aa..5ebc18f4f 100644 --- a/docs-src/content/functions/coll.yml +++ b/docs-src/content/functions/coll.yml @@ -136,6 +136,37 @@ funcs: - | $ gomplate -i '{{ .books | jsonpath `$..works[?( @.edition_count > 400 )].title` }}' -c books=https://openlibrary.org/subjects/fantasy.json [Alice's Adventures in Wonderland Gulliver's Travels] + - name: coll.JQ + alias: jsonpath + description: | + Filters an input object or list using the JQ language implemented by gojq (https://github.com/itchyny/gojq). + + Any object or list may be used as input. The output depends somewhat on the expression; if multiple items are matched, an array is returned. + + JQ filter expressions can be tested at https://jqplay.org/ + + [jq Manual]: https://stedolan.github.io/jq/manual/ + [gojq differences to jq]: https://github.com/itchyny/gojq#difference-to-jq + pipeline: true + arguments: + - name: expression + required: true + description: The JQ expression + - name: in + required: true + description: The object or list to query + examples: + - | + $ gomplate \ + -i '{{ .books | jq `[.works[]|{"title":.title,"authors":[.authors[].name],"published":.first_publish_year}][0]` }}' \ + -c books=https://openlibrary.org/subjects/fantasy.json + { + "authors": [ + "Lewis Carroll" + ], + "published": 1865, + "title": "Alice's Adventures in Wonderland" + } - name: coll.Keys alias: keys description: | diff --git a/docs/content/functions/coll.md b/docs/content/functions/coll.md index e441da9b1..cfd785459 100644 --- a/docs/content/functions/coll.md +++ b/docs/content/functions/coll.md @@ -200,6 +200,51 @@ $ gomplate -i '{{ .books | jsonpath `$..works[?( @.edition_count > 400 )].title` [Alice's Adventures in Wonderland Gulliver's Travels] ``` +## `coll.JQ` + +**Alias:** `jq` + +Filters an input object or list using the JQ language implemented by gojq. + +Any object or list may be used as input. The output depends somewhat on the expression; if multiple items are matched, an array is returned. + +JQ filter expressions can be tested at https://jqplay.org/ + +[jq Manual]: https://stedolan.github.io/jq/manual/ +[gojq library]: https://github.com/itchyny/gojq +[gojq differences to jq]: https://github.com/itchyny/gojq#difference-to-jq + +### Usage + +```go +coll.JQ expression in +``` +```go +in | coll.JQ expression +``` + +### Arguments + +| name | description | +|------|-------------| +| `expression` | _(required)_ The JQ expression | +| `in` | _(required)_ The object or list to query | + +### Examples + +```console +$ gomplate \ + -i '{{ .books | coll.JQ `[ + .works[] + | { + "title": .title, + "authors": [.authors[].name], + "published": .first_publish_year + }][0]` }}' \ + -c books=https://openlibrary.org/subjects/fantasy.json +map[authors:[Lewis Carroll] published:1865 title:Alice's Adventures in Wonderland] +``` + ## `coll.Keys` **Alias:** `keys` diff --git a/funcs/coll.go b/funcs/coll.go index 4a3ca5ab0..4bdf32b6a 100644 --- a/funcs/coll.go +++ b/funcs/coll.go @@ -47,6 +47,7 @@ func CreateCollFuncs(ctx context.Context) map[string]interface{} { f["merge"] = ns.Merge f["sort"] = ns.Sort f["jsonpath"] = ns.JSONPath + f["gojq"] = ns.JQ f["flatten"] = ns.Flatten return f } @@ -142,6 +143,11 @@ func (CollFuncs) JSONPath(p string, in interface{}) (interface{}, error) { return coll.JSONPath(p, in) } +// JQ - +func (CollFuncs) JQ(jqExpr string, in interface{}) (interface{}, error) { + return coll.JQ(jqExpr, in) +} + // Flatten - func (CollFuncs) Flatten(args ...interface{}) ([]interface{}, error) { if len(args) == 0 || len(args) > 2 { diff --git a/go.mod b/go.mod index d31490cbd..b023601de 100644 --- a/go.mod +++ b/go.mod @@ -18,6 +18,7 @@ require ( github.com/hashicorp/consul/api v1.18.0 github.com/hashicorp/go-sockaddr v1.0.2 github.com/hashicorp/vault/api v1.8.2 + github.com/itchyny/gojq v0.12.10 github.com/johannesboyne/gofakes3 v0.0.0-20220627085814-c3ac35da23b2 github.com/joho/godotenv v1.4.0 github.com/pkg/errors v0.9.1 @@ -105,11 +106,12 @@ require ( github.com/hashicorp/yamux v0.0.0-20211028200310-0bc27b27de87 // indirect github.com/imdario/mergo v0.3.13 // indirect github.com/inconshreveable/mousetrap v1.0.1 // indirect + github.com/itchyny/timefmt-go v0.1.5 // indirect github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 // indirect github.com/jmespath/go-jmespath v0.4.0 // indirect github.com/kevinburke/ssh_config v1.2.0 // indirect github.com/mattn/go-colorable v0.1.12 // indirect - github.com/mattn/go-isatty v0.0.14 // indirect + github.com/mattn/go-isatty v0.0.16 // indirect github.com/mitchellh/copystructure v1.2.0 // indirect github.com/mitchellh/go-homedir v1.1.0 // indirect github.com/mitchellh/go-testing-interface v1.14.1 // indirect diff --git a/go.sum b/go.sum index a8be6fbfc..318db2c48 100644 --- a/go.sum +++ b/go.sum @@ -976,6 +976,10 @@ github.com/inconshreveable/mousetrap v1.0.1/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLf github.com/influxdata/influxdb1-client v0.0.0-20191209144304-8bf82d3c094d/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo= github.com/intel/goresctrl v0.2.0/go.mod h1:+CZdzouYFn5EsxgqAQTEzMfwKwuc0fVdMrT9FCCAVRQ= github.com/ionos-cloud/sdk-go/v6 v6.1.0/go.mod h1:Ox3W0iiEz0GHnfY9e5LmAxwklsxguuNFEUSu0gVRTME= +github.com/itchyny/gojq v0.12.10 h1:6TcS0VYWS6wgntpF/4tnrzwdCMjiTxRAxIqZWfDsDQU= +github.com/itchyny/gojq v0.12.10/go.mod h1:o3FT8Gkbg/geT4pLI0tF3hvip5F3Y/uskjRz9OYa38g= +github.com/itchyny/timefmt-go v0.1.5 h1:G0INE2la8S6ru/ZI5JecgyzbbJNs5lG1RcBqa7Jm6GE= +github.com/itchyny/timefmt-go v0.1.5/go.mod h1:nEP7L+2YmAbT2kZ2HfSs1d8Xtw9LY8D2stDBckWakZ8= github.com/j-keck/arping v0.0.0-20160618110441-2cf9dc699c56/go.mod h1:ymszkNOg6tORTn+6F6j+Jc8TOr5osrynvN6ivFWZ2GA= github.com/j-keck/arping v1.0.2/go.mod h1:aJbELhR92bSk7tp79AWM/ftfc90EfEi2bQJrbBFOsPw= github.com/jackc/chunkreader v1.0.0/go.mod h1:RT6O25fNZIuasFJRyZ4R/Y2BbhasbmZXF9QQ7T3kePo= @@ -1124,8 +1128,9 @@ github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hd github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= -github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= +github.com/mattn/go-isatty v0.0.16 h1:bq3VjFmv/sOjHtdEhmkEV4x1AJtvUvOJ2PFAZ5+peKQ= +github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= github.com/mattn/go-runewidth v0.0.2/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU= github.com/mattn/go-shellwords v1.0.3/go.mod h1:3xCvwCdWdlDJUrvuMn7Wuy9eWs4pE8vqg+NOMyg4B2o= github.com/mattn/go-shellwords v1.0.6/go.mod h1:3xCvwCdWdlDJUrvuMn7Wuy9eWs4pE8vqg+NOMyg4B2o= @@ -1977,6 +1982,7 @@ golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220731174439-a90be440212d/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220825204002-c680a09ffe64/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= From d4931bacded38582ff599f4c48ca55e1ea437a54 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Wed, 21 Dec 2022 09:17:28 +0100 Subject: [PATCH 02/15] fix: jq function naming (gojq -> jq) --- funcs/coll.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/funcs/coll.go b/funcs/coll.go index 4bdf32b6a..d23b6ef2a 100644 --- a/funcs/coll.go +++ b/funcs/coll.go @@ -47,7 +47,7 @@ func CreateCollFuncs(ctx context.Context) map[string]interface{} { f["merge"] = ns.Merge f["sort"] = ns.Sort f["jsonpath"] = ns.JSONPath - f["gojq"] = ns.JQ + f["jq"] = ns.JQ f["flatten"] = ns.Flatten return f } From 4828db11e4a8b5e423477c0053613979f7001fc2 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 22 Dec 2022 09:50:58 +0100 Subject: [PATCH 03/15] test: add tests (take from jsonpath_test.go) --- coll/jq_test.go | 150 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 150 insertions(+) create mode 100644 coll/jq_test.go diff --git a/coll/jq_test.go b/coll/jq_test.go new file mode 100644 index 000000000..abd29a2b4 --- /dev/null +++ b/coll/jq_test.go @@ -0,0 +1,150 @@ +package coll + +import ( + "encoding/json" + "testing" + + "github.com/stretchr/testify/assert" +) + +func TestJQ(t *testing.T) { + in := map[string]interface{}{ + "store": map[string]interface{}{ + "book": []interface{}{ + map[string]interface{}{ + "category": "reference", + "author": "Nigel Rees", + "title": "Sayings of the Century", + "price": 8.95, + }, + map[string]interface{}{ + "category": "fiction", + "author": "Evelyn Waugh", + "title": "Sword of Honour", + "price": 12.99, + }, + map[string]interface{}{ + "category": "fiction", + "author": "Herman Melville", + "title": "Moby Dick", + "isbn": "0-553-21311-3", + "price": 8.99, + }, + map[string]interface{}{ + "category": "fiction", + "author": "J. R. R. Tolkien", + "title": "The Lord of the Rings", + "isbn": "0-395-19395-8", + "price": 22.99, + }, + }, + "bicycle": map[string]interface{}{ + "color": "red", + "price": 19.95, + }, + }, + } + out, err := JQ(".store.bicycle.color", in) + assert.NoError(t, err) + assert.Equal(t, "red", out) + + out, err = JQ(".store.bicycle.price", in) + assert.NoError(t, err) + assert.Equal(t, 19.95, out) + + out, err = JQ(".store.bogus", in) + assert.NoError(t, err) + assert.Nil(t, out) + + _, err = JQ("{.store.unclosed", in) + assert.Error(t, err) + + out, err = JQ(".store", in) + assert.NoError(t, err) + assert.EqualValues(t, in["store"], out) + + out, err = JQ(".store.book[].author", in) + assert.NoError(t, err) + assert.Len(t, out, 4) + assert.Contains(t, out, "Nigel Rees") + assert.Contains(t, out, "Evelyn Waugh") + assert.Contains(t, out, "Herman Melville") + assert.Contains(t, out, "J. R. R. Tolkien") + + out, err = JQ(".store.book[]|select(.price < 10.0 )", in) + assert.NoError(t, err) + expected := []interface{}{ + map[string]interface{}{ + "category": "reference", + "author": "Nigel Rees", + "title": "Sayings of the Century", + "price": 8.95, + }, + map[string]interface{}{ + "category": "fiction", + "author": "Herman Melville", + "title": "Moby Dick", + "isbn": "0-553-21311-3", + "price": 8.99, + }, + } + assert.EqualValues(t, expected, out) + + in = map[string]interface{}{ + "a": map[string]interface{}{ + "aa": map[string]interface{}{ + "foo": map[string]interface{}{ + "aaa": map[string]interface{}{ + "aaaa": map[string]interface{}{ + "bar": 1234, + }, + }, + }, + }, + "ab": map[string]interface{}{ + "aba": map[string]interface{}{ + "foo": map[string]interface{}{ + "abaa": true, + "abab": "baz", + }, + }, + }, + }, + } + out, err = JQ(`tostream|select((.[0]|index("foo")) and (.[0][-1]!="foo") and (.[1])) as $s|($s[0]|index("foo")+1) as $ind|($ind|truncate_stream($s)) as $newstream|$newstream|reduce . as [$p,$v] ({};setpath($p;$v))|add`, in) + assert.NoError(t, err) + assert.Len(t, out, 3) + assert.Contains(t, out, map[string]interface{}{"aaaa": map[string]interface{}{"bar": 1234}}) + assert.Contains(t, out, true) + assert.Contains(t, out, "baz") + + type bicycleType struct { + Color string + } + type storeType struct { + Bicycle *bicycleType + safe interface{} + } + + structIn := &storeType{ + Bicycle: &bicycleType{ + Color: "red", + }, + safe: "hidden", + } + + // TODO: Check if this is a valid test case (taken from jsonpath_test.go) since the struct + // had to be converted to JSON and parsed from it again to be able to process using gojq. + v := map[string]interface{}{} + b, err := json.Marshal(structIn) + err = json.Unmarshal(b, &v) + out, err = JQ(".Bicycle.Color", v) + assert.NoError(t, err) + assert.Equal(t, "red", out) + + _, err = JQ(".safe", structIn) + assert.Error(t, err) + + _, err = JQ(".*", structIn) + assert.Error(t, err) +} From d627ef56e3ef2377e11429bc2534a9730a085ca5 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 22 Dec 2022 09:52:12 +0100 Subject: [PATCH 04/15] chore: add TODO for nil values (are they allowed?) --- coll/jq.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/coll/jq.go b/coll/jq.go index 5d07ac4b1..c1f812d9d 100644 --- a/coll/jq.go +++ b/coll/jq.go @@ -22,7 +22,7 @@ func JQ(jqExpr string, in interface{}) (interface{}, error) { if err, ok := v.(error); ok { return nil, errors.Wrap(err, "executing JQ failed") } - if v != nil { + if v != nil { // TODO: Check, if nil may be a valid result a = append(a, v) } } From c3ff3d4c526b0ecfadf88351e99c8059fe6c2f1d Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 16:31:50 +0100 Subject: [PATCH 05/15] refactor: use fmt.Errorf instead of errors.Wrapf Co-authored-by: Dave Henderson --- coll/jq.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/coll/jq.go b/coll/jq.go index c1f812d9d..16097d336 100644 --- a/coll/jq.go +++ b/coll/jq.go @@ -9,7 +9,7 @@ import ( func JQ(jqExpr string, in interface{}) (interface{}, error) { query, err := gojq.Parse(jqExpr) if err != nil { - return nil, errors.Wrapf(err, "couldn't parse JQ %s", jqExpr) + return nil, fmt.Errorf("couldn't parse JQ %s: %w", jqExpr, err) } iter := query.Run(in) var out interface{} From e5f3b16233e0c183e7fadd8d4914812fa3cb77da Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 16:32:18 +0100 Subject: [PATCH 06/15] fix: wrong alias for coll.JQ Co-authored-by: Dave Henderson --- docs-src/content/functions/coll.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs-src/content/functions/coll.yml b/docs-src/content/functions/coll.yml index 5ebc18f4f..53d80cdd7 100644 --- a/docs-src/content/functions/coll.yml +++ b/docs-src/content/functions/coll.yml @@ -137,7 +137,7 @@ funcs: $ gomplate -i '{{ .books | jsonpath `$..works[?( @.edition_count > 400 )].title` }}' -c books=https://openlibrary.org/subjects/fantasy.json [Alice's Adventures in Wonderland Gulliver's Travels] - name: coll.JQ - alias: jsonpath + alias: jq description: | Filters an input object or list using the JQ language implemented by gojq (https://github.com/itchyny/gojq). From 6e7a093525caf399a0b554a3579bd2f4802dd7c6 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 16:32:56 +0100 Subject: [PATCH 07/15] docs: add links to JQ Co-authored-by: Dave Henderson --- docs-src/content/functions/coll.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs-src/content/functions/coll.yml b/docs-src/content/functions/coll.yml index 53d80cdd7..7b5454015 100644 --- a/docs-src/content/functions/coll.yml +++ b/docs-src/content/functions/coll.yml @@ -139,7 +139,7 @@ funcs: - name: coll.JQ alias: jq description: | - Filters an input object or list using the JQ language implemented by gojq (https://github.com/itchyny/gojq). + Filters an input object or list using the [jq](https://stedolan.github.io/jq/) language, as implemented by [gojq](https://github.com/itchyny/gojq). Any object or list may be used as input. The output depends somewhat on the expression; if multiple items are matched, an array is returned. From 6c0701b2af7b97b617b81f9fd136e312c6204003 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 16:41:46 +0100 Subject: [PATCH 08/15] test: add assertions after json marshal/unmarshal Co-authored-by: Dave Henderson --- coll/jq_test.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/coll/jq_test.go b/coll/jq_test.go index abd29a2b4..27e90a37d 100644 --- a/coll/jq_test.go +++ b/coll/jq_test.go @@ -137,7 +137,9 @@ func TestJQ(t *testing.T) { // had to be converted to JSON and parsed from it again to be able to process using gojq. v := map[string]interface{}{} b, err := json.Marshal(structIn) + assert.NoError(err) err = json.Unmarshal(b, &v) + assert.NoError(err) out, err = JQ(".Bicycle.Color", v) assert.NoError(t, err) assert.Equal(t, "red", out) From db8f0c20ca7c77d70565b6236649f33074ec023c Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 16:43:59 +0100 Subject: [PATCH 09/15] refactor: use fmt.Errorf instead of errors.Wrapf Co-authored-by: Dave Henderson --- coll/jq.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/coll/jq.go b/coll/jq.go index 16097d336..d036309a6 100644 --- a/coll/jq.go +++ b/coll/jq.go @@ -20,7 +20,7 @@ func JQ(jqExpr string, in interface{}) (interface{}, error) { break } if err, ok := v.(error); ok { - return nil, errors.Wrap(err, "executing JQ failed") + return nil, fmt.Errorf("executing JQ failed: %w", err) } if v != nil { // TODO: Check, if nil may be a valid result a = append(a, v) From 5b50110085c08c2058cc34810e7f5dcf8d3bff95 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 17:17:59 +0100 Subject: [PATCH 10/15] fix: test syntax and null handling --- coll/jq.go | 7 +++---- coll/jq_test.go | 4 ++-- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/coll/jq.go b/coll/jq.go index d036309a6..db741a719 100644 --- a/coll/jq.go +++ b/coll/jq.go @@ -1,8 +1,9 @@ package coll import ( + "fmt" + "github.com/itchyny/gojq" - "github.com/pkg/errors" ) // JQ - @@ -22,9 +23,7 @@ func JQ(jqExpr string, in interface{}) (interface{}, error) { if err, ok := v.(error); ok { return nil, fmt.Errorf("executing JQ failed: %w", err) } - if v != nil { // TODO: Check, if nil may be a valid result - a = append(a, v) - } + a = append(a, v) } if len(a) == 1 { out = a[0] diff --git a/coll/jq_test.go b/coll/jq_test.go index 27e90a37d..2a5f3268e 100644 --- a/coll/jq_test.go +++ b/coll/jq_test.go @@ -137,9 +137,9 @@ func TestJQ(t *testing.T) { // had to be converted to JSON and parsed from it again to be able to process using gojq. v := map[string]interface{}{} b, err := json.Marshal(structIn) - assert.NoError(err) + assert.NoError(t, err) err = json.Unmarshal(b, &v) - assert.NoError(err) + assert.NoError(t, err) out, err = JQ(".Bicycle.Color", v) assert.NoError(t, err) assert.Equal(t, "red", out) From a3a57e75ef703068ed304cee2fb16499396d3a5a Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 17:50:54 +0100 Subject: [PATCH 11/15] docs: improve documentation --- docs-src/content/functions/coll.yml | 17 +++++++---------- docs/content/functions/coll.md | 22 +++++++++------------- 2 files changed, 16 insertions(+), 23 deletions(-) diff --git a/docs-src/content/functions/coll.yml b/docs-src/content/functions/coll.yml index 7b5454015..917423e65 100644 --- a/docs-src/content/functions/coll.yml +++ b/docs-src/content/functions/coll.yml @@ -141,12 +141,15 @@ funcs: description: | Filters an input object or list using the [jq](https://stedolan.github.io/jq/) language, as implemented by [gojq](https://github.com/itchyny/gojq). - Any object or list may be used as input. The output depends somewhat on the expression; if multiple items are matched, an array is returned. + Any JSON datatype may be used as input (NOTE: strings are not JSON-parsed but passed in as is). + If the expression results in multiple items (no matter if streamed or as an array) they are wrapped in an array. + Otherwise a single item is returned (even if resulting in an array with a single contained element). JQ filter expressions can be tested at https://jqplay.org/ - [jq Manual]: https://stedolan.github.io/jq/manual/ - [gojq differences to jq]: https://github.com/itchyny/gojq#difference-to-jq + See also: + - [jq manual](https://stedolan.github.io/jq/manual/) + - [gojq differences to jq](https://github.com/itchyny/gojq#difference-to-jq) pipeline: true arguments: - name: expression @@ -160,13 +163,7 @@ funcs: $ gomplate \ -i '{{ .books | jq `[.works[]|{"title":.title,"authors":[.authors[].name],"published":.first_publish_year}][0]` }}' \ -c books=https://openlibrary.org/subjects/fantasy.json - { - "authors": [ - "Lewis Carroll" - ], - "published": 1865, - "title": "Alice's Adventures in Wonderland" - } + map[authors:[Lewis Carroll] published:1865 title:Alice's Adventures in Wonderland] - name: coll.Keys alias: keys description: | diff --git a/docs/content/functions/coll.md b/docs/content/functions/coll.md index cfd785459..6e6042bdb 100644 --- a/docs/content/functions/coll.md +++ b/docs/content/functions/coll.md @@ -204,15 +204,17 @@ $ gomplate -i '{{ .books | jsonpath `$..works[?( @.edition_count > 400 )].title` **Alias:** `jq` -Filters an input object or list using the JQ language implemented by gojq. +Filters an input object or list using the [jq](https://stedolan.github.io/jq/) language, as implemented by [gojq](https://github.com/itchyny/gojq). -Any object or list may be used as input. The output depends somewhat on the expression; if multiple items are matched, an array is returned. +Any JSON datatype may be used as input (NOTE: strings are not JSON-parsed but passed in as is). +If the expression results in multiple items (no matter if streamed or as an array) they are wrapped in an array. +Otherwise a single item is returned (even if resulting in an array with a single contained element). JQ filter expressions can be tested at https://jqplay.org/ -[jq Manual]: https://stedolan.github.io/jq/manual/ -[gojq library]: https://github.com/itchyny/gojq -[gojq differences to jq]: https://github.com/itchyny/gojq#difference-to-jq +See also: +- [jq manual](https://stedolan.github.io/jq/manual/) +- [gojq differences to jq](https://github.com/itchyny/gojq#difference-to-jq) ### Usage @@ -234,14 +236,8 @@ in | coll.JQ expression ```console $ gomplate \ - -i '{{ .books | coll.JQ `[ - .works[] - | { - "title": .title, - "authors": [.authors[].name], - "published": .first_publish_year - }][0]` }}' \ - -c books=https://openlibrary.org/subjects/fantasy.json + -i '{{ .books | jq `[.works[]|{"title":.title,"authors":[.authors[].name],"published":.first_publish_year}][0]` }}' \ + -c books=https://openlibrary.org/subjects/fantasy.json map[authors:[Lewis Carroll] published:1865 title:Alice's Adventures in Wonderland] ``` From 97946ab2d0e52f0d425d692af8915e576a7668d5 Mon Sep 17 00:00:00 2001 From: Andreas Hochsteger Date: Thu, 29 Dec 2022 17:55:09 +0100 Subject: [PATCH 12/15] docs: add blank line --- docs-src/content/functions/coll.yml | 1 + docs/content/functions/coll.md | 1 + 2 files changed, 2 insertions(+) diff --git a/docs-src/content/functions/coll.yml b/docs-src/content/functions/coll.yml index 917423e65..046f7d74b 100644 --- a/docs-src/content/functions/coll.yml +++ b/docs-src/content/functions/coll.yml @@ -148,6 +148,7 @@ funcs: JQ filter expressions can be tested at https://jqplay.org/ See also: + - [jq manual](https://stedolan.github.io/jq/manual/) - [gojq differences to jq](https://github.com/itchyny/gojq#difference-to-jq) pipeline: true diff --git a/docs/content/functions/coll.md b/docs/content/functions/coll.md index 6e6042bdb..55f2b9cff 100644 --- a/docs/content/functions/coll.md +++ b/docs/content/functions/coll.md @@ -213,6 +213,7 @@ Otherwise a single item is returned (even if resulting in an array with a single JQ filter expressions can be tested at https://jqplay.org/ See also: + - [jq manual](https://stedolan.github.io/jq/manual/) - [gojq differences to jq](https://github.com/itchyny/gojq#difference-to-jq) From 1f4818cc8b75f590e800ada36257d1d50adbaab9 Mon Sep 17 00:00:00 2001 From: Dave Henderson Date: Thu, 29 Dec 2022 13:23:31 -0500 Subject: [PATCH 13/15] Support cancellation Signed-off-by: Dave Henderson --- coll/jq.go | 6 ++++-- coll/jq_test.go | 24 +++++++++++++----------- funcs/coll.go | 4 ++-- 3 files changed, 19 insertions(+), 15 deletions(-) diff --git a/coll/jq.go b/coll/jq.go index db741a719..0d3e80002 100644 --- a/coll/jq.go +++ b/coll/jq.go @@ -1,18 +1,20 @@ package coll import ( + "context" "fmt" "github.com/itchyny/gojq" ) // JQ - -func JQ(jqExpr string, in interface{}) (interface{}, error) { +func JQ(ctx context.Context, jqExpr string, in interface{}) (interface{}, error) { query, err := gojq.Parse(jqExpr) if err != nil { return nil, fmt.Errorf("couldn't parse JQ %s: %w", jqExpr, err) } - iter := query.Run(in) + + iter := query.RunWithContext(ctx, in) var out interface{} a := []interface{}{} for { diff --git a/coll/jq_test.go b/coll/jq_test.go index 2a5f3268e..34d8128e7 100644 --- a/coll/jq_test.go +++ b/coll/jq_test.go @@ -1,6 +1,7 @@ package coll import ( + "context" "encoding/json" "testing" @@ -8,6 +9,7 @@ import ( ) func TestJQ(t *testing.T) { + ctx := context.Background() in := map[string]interface{}{ "store": map[string]interface{}{ "book": []interface{}{ @@ -44,26 +46,26 @@ func TestJQ(t *testing.T) { }, }, } - out, err := JQ(".store.bicycle.color", in) + out, err := JQ(ctx, ".store.bicycle.color", in) assert.NoError(t, err) assert.Equal(t, "red", out) - out, err = JQ(".store.bicycle.price", in) + out, err = JQ(ctx, ".store.bicycle.price", in) assert.NoError(t, err) assert.Equal(t, 19.95, out) - out, err = JQ(".store.bogus", in) + out, err = JQ(ctx, ".store.bogus", in) assert.NoError(t, err) assert.Nil(t, out) - _, err = JQ("{.store.unclosed", in) + _, err = JQ(ctx, "{.store.unclosed", in) assert.Error(t, err) - out, err = JQ(".store", in) + out, err = JQ(ctx, ".store", in) assert.NoError(t, err) assert.EqualValues(t, in["store"], out) - out, err = JQ(".store.book[].author", in) + out, err = JQ(ctx, ".store.book[].author", in) assert.NoError(t, err) assert.Len(t, out, 4) assert.Contains(t, out, "Nigel Rees") @@ -71,7 +73,7 @@ func TestJQ(t *testing.T) { assert.Contains(t, out, "Herman Melville") assert.Contains(t, out, "J. R. R. Tolkien") - out, err = JQ(".store.book[]|select(.price < 10.0 )", in) + out, err = JQ(ctx, ".store.book[]|select(.price < 10.0 )", in) assert.NoError(t, err) expected := []interface{}{ map[string]interface{}{ @@ -111,7 +113,7 @@ func TestJQ(t *testing.T) { }, }, } - out, err = JQ(`tostream|select((.[0]|index("foo")) and (.[0][-1]!="foo") and (.[1])) as $s|($s[0]|index("foo")+1) as $ind|($ind|truncate_stream($s)) as $newstream|$newstream|reduce . as [$p,$v] ({};setpath($p;$v))|add`, in) + out, err = JQ(ctx, `tostream|select((.[0]|index("foo")) and (.[0][-1]!="foo") and (.[1])) as $s|($s[0]|index("foo")+1) as $ind|($ind|truncate_stream($s)) as $newstream|$newstream|reduce . as [$p,$v] ({};setpath($p;$v))|add`, in) assert.NoError(t, err) assert.Len(t, out, 3) assert.Contains(t, out, map[string]interface{}{"aaaa": map[string]interface{}{"bar": 1234}}) @@ -140,13 +142,13 @@ func TestJQ(t *testing.T) { assert.NoError(t, err) err = json.Unmarshal(b, &v) assert.NoError(t, err) - out, err = JQ(".Bicycle.Color", v) + out, err = JQ(ctx, ".Bicycle.Color", v) assert.NoError(t, err) assert.Equal(t, "red", out) - _, err = JQ(".safe", structIn) + _, err = JQ(ctx, ".safe", structIn) assert.Error(t, err) - _, err = JQ(".*", structIn) + _, err = JQ(ctx, ".*", structIn) assert.Error(t, err) } diff --git a/funcs/coll.go b/funcs/coll.go index d23b6ef2a..97bf911af 100644 --- a/funcs/coll.go +++ b/funcs/coll.go @@ -144,8 +144,8 @@ func (CollFuncs) JSONPath(p string, in interface{}) (interface{}, error) { } // JQ - -func (CollFuncs) JQ(jqExpr string, in interface{}) (interface{}, error) { - return coll.JQ(jqExpr, in) +func (f *CollFuncs) JQ(jqExpr string, in interface{}) (interface{}, error) { + return coll.JQ(f.ctx, jqExpr, in) } // Flatten - From e7f569b22a48391854a6080bce1a5bb7bfa5ddc0 Mon Sep 17 00:00:00 2001 From: Dave Henderson Date: Thu, 29 Dec 2022 14:56:08 -0500 Subject: [PATCH 14/15] Support (almost) all types, not just map[string]interface{} and []interface{} Signed-off-by: Dave Henderson --- coll/jq.go | 63 +++++++++++++++++++++++++++++- coll/jq_test.go | 102 +++++++++++++++++++++++++++++++++++++++++++----- 2 files changed, 153 insertions(+), 12 deletions(-) diff --git a/coll/jq.go b/coll/jq.go index 0d3e80002..16dc8f0ed 100644 --- a/coll/jq.go +++ b/coll/jq.go @@ -2,7 +2,9 @@ package coll import ( "context" + "encoding/json" "fmt" + "reflect" "github.com/itchyny/gojq" ) @@ -11,7 +13,13 @@ import ( func JQ(ctx context.Context, jqExpr string, in interface{}) (interface{}, error) { query, err := gojq.Parse(jqExpr) if err != nil { - return nil, fmt.Errorf("couldn't parse JQ %s: %w", jqExpr, err) + return nil, fmt.Errorf("jq parsing expression %q: %w", jqExpr, err) + } + + // convert input to a supported type, if necessary + in, err = jqConvertType(in) + if err != nil { + return nil, fmt.Errorf("jq type conversion: %w", err) } iter := query.RunWithContext(ctx, in) @@ -23,7 +31,7 @@ func JQ(ctx context.Context, jqExpr string, in interface{}) (interface{}, error) break } if err, ok := v.(error); ok { - return nil, fmt.Errorf("executing JQ failed: %w", err) + return nil, fmt.Errorf("jq execution: %w", err) } a = append(a, v) } @@ -35,3 +43,54 @@ func JQ(ctx context.Context, jqExpr string, in interface{}) (interface{}, error) return out, nil } + +// jqConvertType converts the input to a map[string]interface{}, []interface{}, +// or other supported primitive JSON types. +func jqConvertType(in interface{}) (interface{}, error) { + // if it's already a supported type, pass it through + switch in.(type) { + case map[string]interface{}, []interface{}, + string, []byte, + nil, bool, + int, int8, int16, int32, int64, + uint, uint8, uint16, uint32, uint64, + float32, float64: + return in, nil + } + + inType := reflect.TypeOf(in) + value := reflect.ValueOf(in) + + // pointers need to be dereferenced first + if inType.Kind() == reflect.Ptr { + inType = inType.Elem() + value = value.Elem() + } + + mapType := reflect.TypeOf(map[string]interface{}{}) + sliceType := reflect.TypeOf([]interface{}{}) + // if it can be converted to a map or slice, do that + if inType.ConvertibleTo(mapType) { + return value.Convert(mapType).Interface(), nil + } else if inType.ConvertibleTo(sliceType) { + return value.Convert(sliceType).Interface(), nil + } + + // if it's a struct, the simplest (though not necessarily most efficient) + // is to JSON marshal/unmarshal it + if inType.Kind() == reflect.Struct { + b, err := json.Marshal(in) + if err != nil { + return nil, fmt.Errorf("json marshal struct: %w", err) + } + var m map[string]interface{} + err = json.Unmarshal(b, &m) + if err != nil { + return nil, fmt.Errorf("json unmarshal struct: %w", err) + } + return m, nil + } + + // we maybe don't need to convert the value, so return it as-is + return in, nil +} diff --git a/coll/jq_test.go b/coll/jq_test.go index 34d8128e7..7f71891c9 100644 --- a/coll/jq_test.go +++ b/coll/jq_test.go @@ -119,6 +119,10 @@ func TestJQ(t *testing.T) { assert.Contains(t, out, map[string]interface{}{"aaaa": map[string]interface{}{"bar": 1234}}) assert.Contains(t, out, true) assert.Contains(t, out, "baz") +} + +func TestJQ_typeConversions(t *testing.T) { + ctx := context.Background() type bicycleType struct { Color string @@ -135,20 +139,98 @@ func TestJQ(t *testing.T) { safe: "hidden", } - // TODO: Check if this is a valid test case (taken from jsonpath_test.go) since the struct - // had to be converted to JSON and parsed from it again to be able to process using gojq. - v := map[string]interface{}{} - b, err := json.Marshal(structIn) - assert.NoError(t, err) - err = json.Unmarshal(b, &v) - assert.NoError(t, err) - out, err = JQ(ctx, ".Bicycle.Color", v) + out, err := JQ(ctx, ".Bicycle.Color", structIn) assert.NoError(t, err) assert.Equal(t, "red", out) - _, err = JQ(ctx, ".safe", structIn) - assert.Error(t, err) + out, err = JQ(ctx, ".safe", structIn) + assert.NoError(t, err) + assert.Nil(t, out) _, err = JQ(ctx, ".*", structIn) assert.Error(t, err) + + // a type with an underlying type of map[string]interface{}, just like + // gomplate.tmplctx + type mapType map[string]interface{} + + out, err = JQ(ctx, ".foo", mapType{"foo": "bar"}) + assert.NoError(t, err) + assert.Equal(t, "bar", out) + + // sometimes it'll be a pointer... + out, err = JQ(ctx, ".foo", &mapType{"foo": "bar"}) + assert.NoError(t, err) + assert.Equal(t, "bar", out) + + // underlying slice type + type sliceType []interface{} + + out, err = JQ(ctx, ".[1]", sliceType{"foo", "bar"}) + assert.NoError(t, err) + assert.Equal(t, "bar", out) + + out, err = JQ(ctx, ".[2]", &sliceType{"foo", "bar", "baz"}) + assert.NoError(t, err) + assert.Equal(t, "baz", out) + + // other basic types + out, err = JQ(ctx, ".", []byte("hello")) + assert.NoError(t, err) + assert.EqualValues(t, "hello", out) + + out, err = JQ(ctx, ".", "hello") + assert.NoError(t, err) + assert.EqualValues(t, "hello", out) + + out, err = JQ(ctx, ".", 1234) + assert.NoError(t, err) + assert.EqualValues(t, 1234, out) + + out, err = JQ(ctx, ".", true) + assert.NoError(t, err) + assert.EqualValues(t, true, out) + + out, err = JQ(ctx, ".", nil) + assert.NoError(t, err) + assert.Nil(t, out) + + // underlying basic types + type intType int + out, err = JQ(ctx, ".", intType(1234)) + assert.NoError(t, err) + assert.EqualValues(t, 1234, out) + + type byteArrayType []byte + out, err = JQ(ctx, ".", byteArrayType("hello")) + assert.NoError(t, err) + assert.EqualValues(t, "hello", out) +} + +func TestJQConvertType_passthroughTypes(t *testing.T) { + // non-marshalable values, like recursive structs, can't be used + type recursive struct{ Self *recursive } + v := &recursive{} + v.Self = v + _, err := jqConvertType(v) + assert.Error(t, err) + + testdata := []interface{}{ + map[string]interface{}{"foo": 1234}, + []interface{}{"foo", "bar", "baz", 1, 2, 3}, + "foo", + []byte("foo"), + json.RawMessage(`{"foo": "bar"}`), + true, + nil, + int(1234), int8(123), int16(123), int32(123), int64(123), + uint(123), uint8(123), uint16(123), uint32(123), uint64(123), + float32(123.45), float64(123.45), + } + + for _, d := range testdata { + out, err := jqConvertType(d) + assert.NoError(t, err) + assert.Equal(t, d, out) + } } From 600790bd10df9f9cfaf23de3869b7563f9c7dc49 Mon Sep 17 00:00:00 2001 From: Dave Henderson Date: Thu, 29 Dec 2022 16:50:58 -0500 Subject: [PATCH 15/15] add an integration test for coll.JQ Signed-off-by: Dave Henderson --- internal/tests/integration/collection_test.go | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/internal/tests/integration/collection_test.go b/internal/tests/integration/collection_test.go index cc83e14d6..5175ef64f 100644 --- a/internal/tests/integration/collection_test.go +++ b/internal/tests/integration/collection_test.go @@ -98,3 +98,8 @@ func TestColl_Pick(t *testing.T) { func TestColl_Omit(t *testing.T) { inOutTest(t, `{{ $data := dict "foo" 1 "bar" 2 "baz" 3 }}{{ coll.Omit "foo" "baz" $data }}`, "map[bar:2]") } + +func TestColl_JQ(t *testing.T) { + inOutTest(t, `{{ coll.JQ ".foo" (dict "foo" 1 "bar" 2 "baz" 3) }}`, "1") + inOutTest(t, `{{ coll.Slice "one" 2 "three" 4.0 | jq ".[2]" }}`, `three`) +}