mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-03 03:47:38 +03:00
107 lines
2.7 KiB
Go
107 lines
2.7 KiB
Go
package badjson
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"strings"
|
|
|
|
"github.com/sagernet/sing/common"
|
|
E "github.com/sagernet/sing/common/exceptions"
|
|
"github.com/sagernet/sing/common/json"
|
|
"github.com/sagernet/sing/common/x/collections"
|
|
"github.com/sagernet/sing/common/x/linkedhashmap"
|
|
)
|
|
|
|
type JSONObject struct {
|
|
linkedhashmap.Map[string, any]
|
|
}
|
|
|
|
func (m *JSONObject) IsEmpty() bool {
|
|
if m.Size() == 0 {
|
|
return true
|
|
}
|
|
return common.All(m.Entries(), func(it collections.MapEntry[string, any]) bool {
|
|
if valueInterface, valueMaybeEmpty := it.Value.(isEmpty); valueMaybeEmpty && valueInterface.IsEmpty() {
|
|
return true
|
|
}
|
|
return false
|
|
})
|
|
}
|
|
|
|
func (m *JSONObject) MarshalJSON() ([]byte, error) {
|
|
return m.MarshalJSONContext(context.Background())
|
|
}
|
|
|
|
func (m *JSONObject) MarshalJSONContext(ctx context.Context) ([]byte, error) {
|
|
buffer := new(bytes.Buffer)
|
|
buffer.WriteString("{")
|
|
items := common.Filter(m.Entries(), func(it collections.MapEntry[string, any]) bool {
|
|
if valueInterface, valueMaybeEmpty := it.Value.(isEmpty); valueMaybeEmpty && valueInterface.IsEmpty() {
|
|
return false
|
|
}
|
|
return true
|
|
})
|
|
iLen := len(items)
|
|
for i, entry := range items {
|
|
keyContent, err := json.MarshalContext(ctx, entry.Key)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
buffer.WriteString(strings.TrimSpace(string(keyContent)))
|
|
buffer.WriteString(": ")
|
|
valueContent, err := json.MarshalContext(ctx, entry.Value)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
buffer.WriteString(strings.TrimSpace(string(valueContent)))
|
|
if i < iLen-1 {
|
|
buffer.WriteString(", ")
|
|
}
|
|
}
|
|
buffer.WriteString("}")
|
|
return buffer.Bytes(), nil
|
|
}
|
|
|
|
func (m *JSONObject) UnmarshalJSON(content []byte) error {
|
|
return m.UnmarshalJSONContext(context.Background(), content)
|
|
}
|
|
|
|
func (m *JSONObject) UnmarshalJSONContext(ctx context.Context, content []byte) error {
|
|
decoder := json.NewDecoderContext(ctx, bytes.NewReader(content))
|
|
m.Clear()
|
|
objectStart, err := decoder.Token()
|
|
if err != nil {
|
|
return err
|
|
} else if objectStart != json.Delim('{') {
|
|
return E.New("expected json object start, but starts with ", objectStart)
|
|
}
|
|
err = m.decodeJSON(decoder)
|
|
if err != nil {
|
|
return E.Cause(err, "decode json object content")
|
|
}
|
|
objectEnd, err := decoder.Token()
|
|
if err != nil {
|
|
return err
|
|
} else if objectEnd != json.Delim('}') {
|
|
return E.New("expected json object end, but ends with ", objectEnd)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *JSONObject) decodeJSON(decoder *json.Decoder) error {
|
|
for decoder.More() {
|
|
var entryKey string
|
|
keyToken, err := decoder.Token()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
entryKey = keyToken.(string)
|
|
var entryValue any
|
|
entryValue, err = decodeJSON(decoder)
|
|
if err != nil {
|
|
return E.Cause(err, "decode value for ", entryKey)
|
|
}
|
|
m.Put(entryKey, entryValue)
|
|
}
|
|
return nil
|
|
}
|