-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathobject.go
More file actions
115 lines (103 loc) · 2.37 KB
/
object.go
File metadata and controls
115 lines (103 loc) · 2.37 KB
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
package libjson
import (
"encoding/json"
"errors"
"fmt"
"strconv"
)
type JSON struct {
obj any
}
func Get[T any](obj *JSON, path string) (T, error) {
val, err := obj.get(path)
if err != nil {
var e T
return e, err
}
if castVal, ok := val.(T); !ok {
var e T
return e, fmt.Errorf("Expected value of type %T, got type %T", e, val)
} else {
return castVal, nil
}
}
func indexByKey(data any, key any) (any, error) {
switch v := data.(type) {
case nil:
return nil, errors.New("Can not index into null")
case string:
return nil, errors.New("Can not index into string")
case float64:
return nil, errors.New("Can not index into number")
case []any:
if len(v) == 0 {
return nil, nil
}
if k, ok := key.(int); !ok {
return nil, fmt.Errorf("Can not use %T::%v to index into %T::%v", key, key, data, data)
} else {
return v[k], nil
}
case map[string]any:
if len(v) == 0 {
return nil, nil
}
if k, ok := key.(string); !ok {
return nil, fmt.Errorf("Can not use %T::%v to index into %T::%v", key, key, data, data)
} else {
return v[k], nil
}
default:
return nil, fmt.Errorf("Unsupported %T, can not index", data)
}
}
func parsePath(path string) (func(any) (any, error), error) {
if len(path) == 0 {
return nil, errors.New("Unexpected index syntax, top level element is available via '.'")
}
// fast paths for '.' path / parent access
if len(path) == 1 && path[0] == '.' {
return func(a any) (any, error) {
return a, nil
}, nil
}
// skip first . because we handled that above
path = path[1:]
keys := make([]any, 0, len(path)/4)
lastIndex := 0
for i, b := range path {
if b == '.' {
keys = append(keys, path[lastIndex:i])
lastIndex = i + 1
} else if i+1 == len(path) {
keys = append(keys, path[lastIndex:i+1])
}
}
return func(a any) (any, error) {
val := a
for _, k := range keys {
key := k.(string)
if key[0] >= '0' && key[0] <= '9' {
if k1, err := strconv.ParseInt(key, 10, 32); err == nil {
k = int(k1)
}
}
if v, err := indexByKey(val, k); err != nil {
return nil, err
} else {
val = v
}
}
return val, nil
}, nil
}
func (j *JSON) get(path string) (any, error) {
f, err := parsePath(path)
if err != nil {
return nil, fmt.Errorf("%w: %q", errors.ErrUnsupported, path)
}
return f(j.obj)
}
func (j *JSON) MarshalJSON() ([]byte, error) {
return json.Marshal(j.obj)
}