-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathparser.go
124 lines (99 loc) · 2.19 KB
/
parser.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
package jsonparse
import (
"bytes"
"encoding/json"
"io"
"io/ioutil"
"net/http"
"strconv"
"strings"
"github.com/caddyserver/caddy/v2"
)
type fetcher interface {
Fetch(interface{}, string) (interface{}, bool)
}
type fetcherFunc func(interface{}, string) (interface{}, bool)
func (f fetcherFunc) Fetch(v interface{}, key string) (interface{}, bool) {
return f(v, key)
}
type fetchers []fetcher
func (fs fetchers) Fetch(v interface{}, key string) (interface{}, bool) {
for _, f := range fs {
v, ok := f.Fetch(v, key)
if ok {
return v, true
}
}
return nil, false
}
func fromMap(v interface{}, key string) (interface{}, bool) {
// convert value to map
m, ok := v.(map[string]interface{})
if !ok {
return nil, false
}
// ensure key exists
if val, ok := m[key]; ok {
return val, true
}
return nil, true
}
func fromArray(v interface{}, key string) (interface{}, bool) {
// convert key to int
i, err := strconv.Atoi(key)
if err != nil {
return nil, false
}
// convert value to array
a, ok := v.([]interface{})
if !ok {
return nil, false
}
// ensure index
if len(a) > i {
return a[i], true
}
return nil, true
}
func fetchValue(v interface{}, key string) interface{} {
f := fetchers{
fetcherFunc(fromMap),
fetcherFunc(fromArray),
}
var current interface{} = v
for _, k := range strings.Split(key, ".") {
val, ok := f.Fetch(current, k)
if !ok {
return nil
}
current = val
}
return current
}
func newReplacerFunc(r *http.Request) (caddy.ReplacerFunc, error) {
var v interface{}
bodyCopy := bytes.Buffer{}
tee := io.TeeReader(r.Body, &bodyCopy) // preserve the body
err := json.NewDecoder(tee).Decode(&v)
if err != nil {
return nil, err
}
// replace the body for further handlers
r.Body = ioutil.NopCloser(&bodyCopy)
// prevent repetitive parsing. cache values
values := map[string]interface{}{}
return func(key string) (interface{}, bool) {
prefix := "json."
if !strings.HasPrefix(key, prefix) {
return nil, false
}
key = strings.TrimPrefix(key, prefix)
// use cache if previously fetched
if val, ok := values[key]; ok {
return val, true
}
val := fetchValue(v, key)
values[key] = val // cache
return val, true
}, nil
}