This repository has been archived by the owner on Sep 25, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmango.go
133 lines (125 loc) · 2.67 KB
/
mango.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
125
126
127
128
129
130
131
132
133
package mango
import (
"encoding/json"
"fmt"
"github.com/go-kivik/mango/collate"
)
// Selector represents a CouchDB Find query selector. See
// http://docs.couchdb.org/en/2.0.0/api/database/find.html#find-selectors
type Selector struct {
op operator
field string
value interface{}
sel []Selector
}
// New returns a new selector, parsed from data.
func New(data string) (*Selector, error) {
s := &Selector{}
err := json.Unmarshal([]byte(data), &s)
return s, err
}
// UnmarshalJSON unmarshals a JSON selector as described in the CouchDB
// documentation.
// http://docs.couchdb.org/en/2.0.0/api/database/find.html#selector-syntax
func (s *Selector) UnmarshalJSON(data []byte) error {
var x map[string]json.RawMessage
if err := json.Unmarshal(data, &x); err != nil {
return err
}
if len(x) == 0 {
return nil
}
var sels []Selector
for k, v := range x {
var op operator
var field string
var value interface{}
field = k
if v[0] == '{' {
var e error
op, value, e = opPattern(v)
if e != nil {
return e
}
}
if op == "" {
op = opEq
if e := json.Unmarshal(v, &value); e != nil {
return e
}
}
sels = append(sels, Selector{
op: op,
field: field,
value: value,
})
}
if len(sels) == 1 {
*s = sels[0]
} else {
*s = Selector{
op: opAnd,
sel: sels,
}
}
return nil
}
func opPattern(data []byte) (op operator, value interface{}, err error) {
var x map[operator]json.RawMessage
if e := json.Unmarshal(data, &x); e != nil {
return operator(""), nil, e
}
if len(x) != 1 {
panic("got more than one result")
}
for k, v := range x {
switch k {
case opEq, opNE, opLT, opLTE, opGT, opGTE:
var value interface{}
if e := json.Unmarshal(v, &value); e != nil {
return "", nil, e
}
return k, value, nil
default:
return "", nil, fmt.Errorf("unknown mango operator '%s'", k)
}
}
return operator(""), nil, nil
}
type couchDoc map[string]interface{}
// Matches returns true if the provided doc matches the selector.
func (s *Selector) Matches(doc couchDoc) (bool, error) {
c := &collate.Raw{}
switch s.op {
case opNone:
return true, nil
case opEq, opGT, opGTE, opLT, opLTE:
v, ok := doc[s.field]
if !ok {
return false, nil
}
switch s.op {
case opEq:
return c.Eq(v, s.value), nil
case opGT:
return c.GT(v, s.value), nil
case opGTE:
return c.GTE(v, s.value), nil
case opLT:
return c.LT(v, s.value), nil
case opLTE:
return c.LTE(v, s.value), nil
}
case opAnd:
for _, sel := range s.sel {
m, e := sel.Matches(doc)
if e != nil || !m {
return m, e
}
}
return true, nil
default:
return false, fmt.Errorf("unknown mango operator '%s'", s.op)
}
return true, nil
}