mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-04 12:27:37 +03:00
The Merge function has been refactored for clearer code by splitting it into multiple functions: "Merge", "MergeFromSource", "MergeFromDestination" and "MergeFrom" in the badjson package. These new functions improve the handling of raw JSON during merge and with this refactoring, the responsibility of each function is more defined.
120 lines
3.3 KiB
Go
120 lines
3.3 KiB
Go
package badjson
|
|
|
|
import (
|
|
"reflect"
|
|
|
|
"github.com/sagernet/sing/common"
|
|
E "github.com/sagernet/sing/common/exceptions"
|
|
"github.com/sagernet/sing/common/json"
|
|
)
|
|
|
|
func Omitempty[T any](value T) (T, error) {
|
|
objectContent, err := json.Marshal(value)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "marshal object")
|
|
}
|
|
rawNewObject, err := Decode(objectContent)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), err
|
|
}
|
|
newObjectContent, err := json.Marshal(rawNewObject)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "marshal new object")
|
|
}
|
|
var newObject T
|
|
err = json.Unmarshal(newObjectContent, &newObject)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "unmarshal new object")
|
|
}
|
|
return newObject, nil
|
|
}
|
|
|
|
func Merge[T any](source T, destination T) (T, error) {
|
|
rawSource, err := json.Marshal(source)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "marshal source")
|
|
}
|
|
rawDestination, err := json.Marshal(destination)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "marshal destination")
|
|
}
|
|
return MergeFrom[T](rawSource, rawDestination)
|
|
}
|
|
|
|
func MergeFromSource[T any](rawSource json.RawMessage, destination T) (T, error) {
|
|
rawDestination, err := json.Marshal(destination)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "marshal destination")
|
|
}
|
|
return MergeFrom[T](rawSource, rawDestination)
|
|
}
|
|
|
|
func MergeFromDestination[T any](source T, rawDestination json.RawMessage) (T, error) {
|
|
rawSource, err := json.Marshal(source)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "marshal source")
|
|
}
|
|
return MergeFrom[T](rawSource, rawDestination)
|
|
}
|
|
|
|
func MergeFrom[T any](rawSource json.RawMessage, rawDestination json.RawMessage) (T, error) {
|
|
rawMerged, err := MergeJSON(rawSource, rawDestination)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "merge options")
|
|
}
|
|
var merged T
|
|
err = json.Unmarshal(rawMerged, &merged)
|
|
if err != nil {
|
|
return common.DefaultValue[T](), E.Cause(err, "unmarshal merged options")
|
|
}
|
|
return merged, nil
|
|
}
|
|
|
|
func MergeJSON(rawSource json.RawMessage, rawDestination json.RawMessage) (json.RawMessage, error) {
|
|
source, err := Decode(rawSource)
|
|
if err != nil {
|
|
return nil, E.Cause(err, "decode source")
|
|
}
|
|
destination, err := Decode(rawDestination)
|
|
if err != nil {
|
|
return nil, E.Cause(err, "decode destination")
|
|
}
|
|
merged, err := mergeJSON(source, destination)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return json.Marshal(merged)
|
|
}
|
|
|
|
func mergeJSON(anySource any, anyDestination any) (any, error) {
|
|
switch destination := anyDestination.(type) {
|
|
case JSONArray:
|
|
switch source := anySource.(type) {
|
|
case JSONArray:
|
|
destination = append(destination, source...)
|
|
default:
|
|
destination = append(destination, source)
|
|
}
|
|
return destination, nil
|
|
case *JSONObject:
|
|
switch source := anySource.(type) {
|
|
case *JSONObject:
|
|
for _, entry := range source.Entries() {
|
|
oldValue, loaded := destination.Get(entry.Key)
|
|
if loaded {
|
|
var err error
|
|
entry.Value, err = mergeJSON(entry.Value, oldValue)
|
|
if err != nil {
|
|
return nil, E.Cause(err, "merge object item ", entry.Key)
|
|
}
|
|
}
|
|
destination.Put(entry.Key, entry.Value)
|
|
}
|
|
default:
|
|
return nil, E.New("cannot merge json object into ", reflect.TypeOf(destination))
|
|
}
|
|
return destination, nil
|
|
default:
|
|
return destination, nil
|
|
}
|
|
}
|