2024-10-11 14:39:53 +00:00
|
|
|
package jsonloader
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
2025-01-07 10:49:23 +00:00
|
|
|
"errors"
|
2024-10-11 14:39:53 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
|
|
|
|
"github.com/databricks/cli/libs/dyn"
|
|
|
|
)
|
|
|
|
|
|
|
|
func LoadJSON(data []byte, source string) (dyn.Value, error) {
|
|
|
|
offset := BuildLineOffsets(data)
|
|
|
|
offset.SetSource(source)
|
|
|
|
|
|
|
|
reader := bytes.NewReader(data)
|
|
|
|
decoder := json.NewDecoder(reader)
|
|
|
|
|
|
|
|
// Start decoding from the top-level value
|
|
|
|
value, err := decodeValue(decoder, &offset)
|
|
|
|
if err != nil {
|
|
|
|
if err == io.EOF {
|
2025-01-07 10:49:23 +00:00
|
|
|
err = errors.New("unexpected end of JSON input")
|
2024-10-11 14:39:53 +00:00
|
|
|
}
|
|
|
|
return dyn.InvalidValue, fmt.Errorf("error decoding JSON at %s: %v", value.Location(), err)
|
|
|
|
}
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func decodeValue(decoder *json.Decoder, o *Offset) (dyn.Value, error) {
|
|
|
|
// Read the next JSON token
|
|
|
|
token, err := decoder.Token()
|
|
|
|
if err != nil {
|
|
|
|
return dyn.InvalidValue, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the current byte offset and the location.
|
|
|
|
// We will later use this location to store the location of the value in the file
|
|
|
|
// For objects and arrays, we will store the location of the opening '{' or '['
|
|
|
|
// For primitive types, we will store the location of the value itself (end of the value)
|
|
|
|
// We can't reliably calculate the beginning of the value for primitive types because
|
|
|
|
// the decoder doesn't provide the offset of the beginning of the value and the value might or might not be quoted.
|
|
|
|
offset := decoder.InputOffset()
|
|
|
|
location := o.GetPosition(offset)
|
|
|
|
|
|
|
|
switch tok := token.(type) {
|
|
|
|
case json.Delim:
|
|
|
|
if tok == '{' {
|
|
|
|
location = o.GetPosition(offset - 1)
|
|
|
|
// Decode JSON object
|
|
|
|
obj := dyn.NewMapping()
|
|
|
|
for decoder.More() {
|
|
|
|
// Decode the key
|
|
|
|
keyToken, err := decoder.Token()
|
|
|
|
if err != nil {
|
|
|
|
return invalidValueWithLocation(decoder, o), err
|
|
|
|
}
|
|
|
|
key, ok := keyToken.(string)
|
|
|
|
if !ok {
|
2025-01-07 10:49:23 +00:00
|
|
|
return invalidValueWithLocation(decoder, o), errors.New("expected string for object key")
|
2024-10-11 14:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the offset of the key by subtracting the length of the key and the '"' character
|
|
|
|
keyOffset := decoder.InputOffset() - int64(len(key)+1)
|
|
|
|
keyVal := dyn.NewValue(key, []dyn.Location{o.GetPosition(keyOffset)})
|
|
|
|
|
|
|
|
// Decode the value recursively
|
|
|
|
val, err := decodeValue(decoder, o)
|
|
|
|
if err != nil {
|
|
|
|
return invalidValueWithLocation(decoder, o), err
|
|
|
|
}
|
|
|
|
|
|
|
|
obj.Set(keyVal, val) //nolint:errcheck
|
|
|
|
}
|
|
|
|
// Consume the closing '}'
|
|
|
|
if _, err := decoder.Token(); err != nil {
|
|
|
|
return invalidValueWithLocation(decoder, o), err
|
|
|
|
}
|
|
|
|
return dyn.NewValue(obj, []dyn.Location{location}), nil
|
|
|
|
} else if tok == '[' {
|
|
|
|
location = o.GetPosition(offset - 1)
|
|
|
|
// Decode JSON array
|
|
|
|
var arr []dyn.Value
|
|
|
|
for decoder.More() {
|
|
|
|
val, err := decodeValue(decoder, o)
|
|
|
|
if err != nil {
|
|
|
|
return invalidValueWithLocation(decoder, o), err
|
|
|
|
}
|
|
|
|
arr = append(arr, val)
|
|
|
|
}
|
|
|
|
// Consume the closing ']'
|
|
|
|
if _, err := decoder.Token(); err != nil {
|
|
|
|
return invalidValueWithLocation(decoder, o), err
|
|
|
|
}
|
|
|
|
return dyn.NewValue(arr, []dyn.Location{location}), nil
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return dyn.NewValue(tok, []dyn.Location{location}), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return invalidValueWithLocation(decoder, o), fmt.Errorf("unexpected token: %v", token)
|
|
|
|
}
|
|
|
|
|
|
|
|
func invalidValueWithLocation(decoder *json.Decoder, o *Offset) dyn.Value {
|
|
|
|
location := o.GetPosition(decoder.InputOffset())
|
|
|
|
return dyn.InvalidValue.WithLocations([]dyn.Location{location})
|
|
|
|
}
|