-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdecoder.go
101 lines (82 loc) · 2.23 KB
/
decoder.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
// SPDX-FileCopyrightText: 2022 Weston Schmidt <[email protected]>
// SPDX-License-Identifier: Apache-2.0
// json package is a goschtalt decoder package.
//
// The json package automatically registers the decoder as a default decoder
// with the goschtalt package so the usage is as simple as possible ... simply
// import the package and it should just work.
//
// Import the package like you do for pprof - like this:
//
// import (
// "fmt"
// "os"
// ...
//
// "github.com/goschtalt/goschtalt"
// _ "github.com/goschtalt/json-decoder"
// )
//
// See the example for how to use this extension package.
package json
import (
"bytes"
stljson "encoding/json"
"github.com/goschtalt/goschtalt"
"github.com/goschtalt/goschtalt/pkg/decoder"
"github.com/goschtalt/goschtalt/pkg/meta"
)
var _ decoder.Decoder = (*Decoder)(nil)
// Use init to automatically wire this encoder as one available for goschtalt
// simply by including this package.
func init() {
var d Decoder
goschtalt.DefaultOptions = append(goschtalt.DefaultOptions, goschtalt.WithDecoder(d))
}
// Decoder is a class for the json decoder.
type Decoder struct{}
// Extensions returns the supported extensions.
func (d Decoder) Extensions() []string {
return []string{"json"}
}
// Decode decodes a byte arreay into the meta.Object tree.
func (d Decoder) Decode(ctx decoder.Context, b []byte, m *meta.Object) error {
var raw map[string]any
if len(b) == 0 {
*m = meta.Object{}
return nil
}
dec := stljson.NewDecoder(bytes.NewBuffer(b))
dec.UseNumber()
if err := dec.Decode(&raw); err != nil {
return err
}
objs := meta.ObjectFromRaw(raw)
// TODO use the stream processor to find the line and column information.
*m = annotate(ctx.Filename, objs)
return nil
}
func annotate(filename string, obj meta.Object) meta.Object {
kind := obj.Kind()
origin := meta.Origin{
File: filename,
}
if kind == meta.Value {
obj.Origins = []meta.Origin{origin}
return obj
}
if kind == meta.Array {
for i := range obj.Array {
val := obj.Array[i]
obj.Array[i] = annotate(filename, val)
}
obj.Origins = []meta.Origin{origin}
return obj
}
// Map
obj.Origins = []meta.Origin{origin}
for k, v := range obj.Map {
obj.Map[k] = annotate(filename, v)
}
return obj
}