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 }