-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmerger.go
146 lines (127 loc) · 4.19 KB
/
merger.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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
package dstruct
import (
"fmt"
"reflect"
"github.com/MartinSimango/dstruct/dreflect"
)
// MergeStructs merges two structs
func MergeStructs(strcts ...interface{}) (a any, err error) {
if len(strcts) < 2 {
return nil, fmt.Errorf("failed to merge: number of structs to merge must be 2 or more")
}
for i := 0; i < len(strcts); i++ {
if reflect.ValueOf(strcts[i]).Kind() != reflect.Struct {
return nil, fmt.Errorf(
"failed to merged structs: %d interface is not a struct ",
(i + 1),
)
}
}
left := ExtendStruct(strcts[0])
var right Builder
for i := 1; i < len(strcts); i++ {
right = ExtendStruct(strcts[i])
mergedStruct, err := mergeStructs(left, right, reflect.Struct)
if err != nil {
return nil, err
}
left = ExtendStruct(mergedStruct)
}
return left.Build().Instance(), nil
}
// TODO clean this function up
func mergeStructs(left, right Builder, parentKind reflect.Kind) (any, error) {
// struct to be returned
newStruct := ExtendStruct(left.Build().Instance())
for name, field := range right.(*treeBuilderImpl).root.children {
elementName := field.data.name
cV := left.(*treeBuilderImpl).root.GetNode(name)
if cV == nil {
newStruct.AddField(elementName, field.data.value.Interface(), string(field.data.tag))
continue
}
if err := validateTypes(field.data.value, cV.data.value, field.data.qualifiedName); err != nil {
return nil, err
}
if field.data.value.Kind() == reflect.Slice {
vSliceType := dreflect.GetSliceType(field.data.value.Interface())
cVSliceType := dreflect.GetSliceType(cV.data.value.Interface())
if err := validateSliceTypes(vSliceType, cVSliceType, field.data.value, cV.data.value, field.data.qualifiedName); err != nil {
return nil, err
}
newStruct.RemoveField(field.data.qualifiedName)
if cVSliceType.Kind() == reflect.Struct {
newSliceTypeStruct, err := mergeStructs(left.GetField(name),
right.GetField(name), reflect.Slice)
if err != nil {
return nil, err
}
newStruct.AddField(field.data.name, newSliceTypeStruct, "")
} else {
newStruct.AddField(field.data.name, field.data.value.Interface(), "")
}
} else if field.data.value.Kind() == reflect.Struct {
updatedSchema, err := mergeStructs(left.GetField(name), right.GetField(name), reflect.Struct)
if err != nil {
return nil, err
}
newStruct.RemoveField(field.data.GetFieldName())
newStruct.AddField(field.data.name, updatedSchema, string(field.data.tag))
}
}
if parentKind == reflect.Slice {
sliceOfElementType := reflect.SliceOf(
reflect.ValueOf(newStruct.Build().Instance()).Elem().Type(),
)
return reflect.MakeSlice(sliceOfElementType, 0, 1024).Interface(), nil
}
return newStruct.Build().Instance(), nil
}
func shouldTypeMatch(kind reflect.Kind) bool {
if kind == reflect.Array || kind == reflect.Struct || kind == reflect.Slice {
return false
}
return true
}
func validateTypes(v, cV reflect.Value, fullFieldName string) error {
currentElementType := reflect.TypeOf(cV.Interface())
newElementType := reflect.TypeOf(v.Interface())
if shouldTypeMatch(v.Kind()) || shouldTypeMatch(cV.Kind()) {
if currentElementType != newElementType {
return fmt.Errorf(
"mismatching types for field '%s': %s and %s",
fullFieldName,
currentElementType,
newElementType,
)
}
} else {
if v.Kind() != cV.Kind() {
return fmt.Errorf("mismatching types for field '%s': %s and %s", fullFieldName, currentElementType, newElementType)
}
}
return nil
}
func validateSliceTypes(
vSliceType, cVSliceType reflect.Type,
v, cV reflect.Value,
fullFieldName string,
) error {
currentElementType := reflect.TypeOf(reflect.New(cVSliceType).Interface())
newElementType := reflect.TypeOf(reflect.New(vSliceType).Interface())
if shouldTypeMatch(vSliceType.Kind()) || shouldTypeMatch(cVSliceType.Kind()) {
if currentElementType != newElementType {
return fmt.Errorf(
"mismatching types for field '%s': %s and %s",
fullFieldName,
reflect.TypeOf(v.Interface()),
reflect.TypeOf(cV.Interface()),
)
}
} else {
if v.Kind() != cV.Kind() {
return fmt.Errorf("mismatching types for field '%s': %s and %s", fullFieldName, reflect.TypeOf(v.Interface()), reflect.TypeOf(cV.Interface()))
}
}
return nil
}