Skip to content

Commit dffcdac

Browse files
committed
Template handler to get json objects by path
1 parent 68fadf5 commit dffcdac

24 files changed

+4645
-7
lines changed

README.md

+8-5
Original file line numberDiff line numberDiff line change
@@ -279,10 +279,12 @@ OpenMock leverages [https://golang.org/pkg/text/template/](https://golang.org/pk
279279
```bash
280280
# Supported functions defined in ./template_helper.go
281281
282-
- jsonPath # doc: https://github.com/antchfx/xpath
283-
- xmlPath # doc: https://github.com/antchfx/xpath
284-
- uuidv5 # uuid v5 sha1 hash
285-
- redisDo # run redis commands. For example {{redisDo "RPUSH" "arr" "hi"}}
282+
-
283+
- jsonPath # doc: https://github.com/antchfx/xpath
284+
- gJsonPath # doc: https://github.com/tidwall/gjson
285+
- xmlPath # doc: https://github.com/antchfx/xpath
286+
- uuidv5 # uuid v5 sha1 hash
287+
- redisDo # run redis commands. For example {{redisDo "RPUSH" "arr" "hi"}}
286288
- ...
287289
288290
# Supported functions inherited from
@@ -296,6 +298,7 @@ OpenMock leverages [https://golang.org/pkg/text/template/](https://golang.org/pk
296298
# Examples
297299
{{.HTTPHeader.Get "X-Token" | eq "t1234"}}
298300
{{.HTTPBody | jsonPath "user/first_name" | replace "A" "a" | uuidv5 }}
301+
{{.HTTPBody | gJsonPath "users.0.first_name" }}
299302
{{.HTTPBody | xmlPath "node1/node2/node3"}}
300303
```
301304

@@ -602,7 +605,7 @@ with the `proto.MessageV2` method.
602605

603606
Please note that OpenMock expects the `payload` or `payload_from_file` for a reply_grpc action to be in the json
604607
form of your `Response` protobuf message. The request should be in the `Request` protobuf message format
605-
as it is parsed into json to support JSONPath operations.
608+
as it is parsed into json to support `jsonPath` and `gJsonPath` operations.
606609

607610
Example configuration by directly importing the `github.com/checkr/openmock` package into a wrapper project.
608611
```

demo_templates/http.yaml

+12
Original file line numberDiff line numberDiff line change
@@ -194,6 +194,18 @@
194194
status_code: 200
195195
body: '{{ .HTTPBody | jsonPath "foo" }}'
196196

197+
- key: json-payload-from-http-body
198+
kind: Behavior
199+
expect:
200+
condition: '{{ .HTTPBody | gJsonPath "context.type" | toString | eq "foo" }}'
201+
http:
202+
method: POST
203+
path: /json_from_body
204+
actions:
205+
- reply_http:
206+
status_code: 200
207+
body: '{{ .HTTPBody | gJsonPath "context.payload" }}'
208+
197209
- key: base64-basicauth-with-env
198210
kind: Behavior
199211
expect:

go.mod

+2
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,8 @@ require (
5656
github.com/streadway/amqp v0.0.0-20180806233856-70e15c650864
5757
github.com/stretchr/testify v1.5.1
5858
github.com/teamwork/reload v1.3.0
59+
github.com/tidwall/gjson v1.6.0
60+
github.com/tidwall/pretty v1.0.1 // indirect
5961
github.com/yuin/gopher-lua v0.0.0-20180827083657-b942cacc89fe // indirect
6062
golang.org/x/net v0.0.0-20200226121028-0de0cce0169b
6163
google.golang.org/genproto v0.0.0-20200224152610-e50cd9704f63 // indirect

go.sum

+6
Original file line numberDiff line numberDiff line change
@@ -270,8 +270,14 @@ github.com/stretchr/testify v1.5.1 h1:nOGnQDM7FYENwehXlg/kFVnos3rEvtKTjRvOWSzb6H
270270
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
271271
github.com/teamwork/reload v1.3.0 h1:ElBaBhtRwVRxsPRGK1wYugr2knciOXFMZT5ltPpEwAI=
272272
github.com/teamwork/reload v1.3.0/go.mod h1:kHdVPdfdmA+ygkBbigWUeerpy6EK4Kcukx1TNyePXHA=
273+
github.com/tidwall/gjson v1.6.0 h1:9VEQWz6LLMUsUl6PueE49ir4Ka6CzLymOAZDxpFsTDc=
274+
github.com/tidwall/gjson v1.6.0/go.mod h1:P256ACg0Mn+j1RXIDXoss50DeIABTYK1PULOJHhxOls=
275+
github.com/tidwall/match v1.0.1 h1:PnKP62LPNxHKTwvHHZZzdOAOCtsJTjo6dZLCwpKm5xc=
276+
github.com/tidwall/match v1.0.1/go.mod h1:LujAq0jyVjBy028G1WhWfIzbpQfMO8bBZ6Tyb0+pL9E=
273277
github.com/tidwall/pretty v1.0.0 h1:HsD+QiTn7sK6flMKIvNmpqz1qrpP3Ps6jOKIKMooyg4=
274278
github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
279+
github.com/tidwall/pretty v1.0.1 h1:WE4RBSZ1x6McVVC8S/Md+Qse8YUv6HRObAx6ke00NY8=
280+
github.com/tidwall/pretty v1.0.1/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
275281
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
276282
github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
277283
github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=

template_helper.go

+33
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import (
44
"crypto/hmac"
55
"crypto/sha256"
66
"encoding/hex"
7+
"errors"
78
"reflect"
89
"regexp"
910
"strings"
@@ -13,13 +14,15 @@ import (
1314
"github.com/antchfx/xmlquery"
1415
"github.com/google/uuid"
1516
"github.com/sirupsen/logrus"
17+
"github.com/tidwall/gjson"
1618
)
1719

1820
func genLocalFuncMap(om *OpenMock) template.FuncMap {
1921
return template.FuncMap{
2022
"htmlEscapeString": template.HTMLEscapeString,
2123
"isLastIndex": isLastIndex,
2224
"jsonPath": jsonPath,
25+
"gJsonPath": gJsonPath,
2326
"redisDo": redisDo(om),
2427
"regexFindAllSubmatch": regexFindAllSubmatch,
2528
"regexFindFirstSubmatch": regexFindFirstSubmatch,
@@ -56,6 +59,36 @@ func jsonPath(expr string, tmpl string) (ret string, err error) {
5659
return "", nil
5760
}
5861

62+
func gJsonPath(expr string, tmpl string) (ret string, err error) {
63+
defer func() {
64+
if r := recover(); r != nil {
65+
err = r.(error)
66+
}
67+
logrus.WithFields(logrus.Fields{
68+
"err": err,
69+
"tmpl": tmpl,
70+
"expr": expr,
71+
}).Debug("running gJsonPath")
72+
}()
73+
74+
if tmpl == "" {
75+
return "", nil
76+
}
77+
78+
if !gjson.Valid(tmpl) {
79+
return "", errors.New("Invalid json")
80+
}
81+
82+
node := gjson.Parse(tmpl).Get(expr)
83+
if node.Exists() {
84+
if node.Type.String() == "String" {
85+
return node.String(), nil
86+
}
87+
return node.Raw, nil
88+
}
89+
return "", nil
90+
}
91+
5992
func xmlPath(expr string, tmpl string) (ret string, err error) {
6093
defer func() {
6194
if r := recover(); r != nil {

template_helper_test.go

+56
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,62 @@ func TestJSONPath(t *testing.T) {
4747
assert.Equal(t, "112879785776", ret)
4848
}
4949

50+
func TestGJsonPath(t *testing.T) {
51+
var tmpl string
52+
var ret string
53+
var err error
54+
55+
tmpl = `{"payload": {"user": {"username": "johnny", "first_name": "John", "valid": true, "id": 7, "profile": null}}}`
56+
57+
ret, err = gJsonPath("payload.user", tmpl)
58+
assert.NoError(t, err)
59+
assert.Equal(t, "{\"username\": \"johnny\", \"first_name\": \"John\", \"valid\": true, \"id\": 7, \"profile\": null}", ret)
60+
61+
ret, err = gJsonPath("payload.user.first_name", tmpl)
62+
assert.NoError(t, err)
63+
assert.Equal(t, "John", ret)
64+
65+
ret, err = gJsonPath("payload.user.valid", tmpl)
66+
assert.NoError(t, err)
67+
assert.Equal(t, "true", ret)
68+
69+
ret, err = gJsonPath("payload.user.id", tmpl)
70+
assert.NoError(t, err)
71+
assert.Equal(t, "7", ret)
72+
73+
ret, err = gJsonPath("payload.user.profile", tmpl)
74+
assert.NoError(t, err)
75+
assert.Equal(t, "null", ret)
76+
77+
tmpl = `{"payload": {"rivers": ["klamath", "merced", "american", "mississippi"]}}`
78+
79+
ret, err = gJsonPath("payload.rivers.#", tmpl)
80+
assert.NoError(t, err)
81+
assert.Equal(t, "4", ret)
82+
83+
ret, err = gJsonPath("payload.rivers.1", tmpl)
84+
assert.NoError(t, err)
85+
assert.Equal(t, "merced", ret)
86+
87+
ret, err = gJsonPath("payload.riv*.2", tmpl)
88+
assert.NoError(t, err)
89+
assert.Equal(t, "american", ret)
90+
91+
ret, err = gJsonPath("payload.r?vers.0", tmpl)
92+
assert.NoError(t, err)
93+
assert.Equal(t, "klamath", ret)
94+
95+
tmpl = `{"payload": {"rivers": [{"name": "klamath", "length": 257}, {"name": "merced", "length": 145}]}}`
96+
ret, err = gJsonPath("payload.rivers.#.length", tmpl)
97+
assert.NoError(t, err)
98+
assert.Equal(t, "[257,145]", ret)
99+
100+
tmpl = `{"payload": {"rivers": [{"name": "klamath", "length": 257}, {"name": "merced", "length": 145}]}}`
101+
ret, err = gJsonPath("payload.rivers.1.name", tmpl)
102+
assert.NoError(t, err)
103+
assert.Equal(t, "merced", ret)
104+
}
105+
50106
func TestHelpers(t *testing.T) {
51107
t.Run("uuid4 helpers", func(t *testing.T) {
52108
raw := `{{ uuidv4 }}`

template_test.go

+4-2
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,8 @@ func TestTemplateRender(t *testing.T) {
3535
raw := `{
3636
"transaction_id": "{{.KafkaPayload | jsonPath "transaction_id"}}",
3737
"first_name": "{{.AMQPPayload | xmlPath "user/first_name"}}",
38-
"middle_name": "{{.HTTPBody | jsonPath "user/middle_name"}}"
38+
"middle_name": "{{.HTTPBody | jsonPath "user/middle_name"}}",
39+
"user": {{.HTTPBody | gJsonPath "user"}}
3940
}`
4041
c := Context{
4142
HTTPBody: `{"user": {"middle_name": "H"}}`,
@@ -49,7 +50,8 @@ func TestTemplateRender(t *testing.T) {
4950
{
5051
"transaction_id": "t1234",
5152
"first_name": "John",
52-
"middle_name": "H"
53+
"middle_name": "H",
54+
"user": {"middle_name": "H"}
5355
}
5456
`)
5557
})

vendor/github.com/tidwall/gjson/.travis.yml

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

vendor/github.com/tidwall/gjson/LICENSE

+20
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)