/
merge.go
68 lines (55 loc) · 1.77 KB
/
merge.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
/*
Copyright 2021 The KodeRover Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package yaml
import "sigs.k8s.io/yaml"
// MergeAndUnmarshal merges a couple of yaml files into one yaml file, and return a map[string]interface{},
// a field in latter file will override the same field in former file.
func MergeAndUnmarshal(yamls [][]byte) (map[string]interface{}, error) {
base := map[string]interface{}{}
for _, y := range yamls {
currentMap := map[string]interface{}{}
if err := yaml.Unmarshal(y, ¤tMap); err != nil {
return nil, err
}
// Merge with the previous map
base = mergeMaps(base, currentMap)
}
return base, nil
}
// Merge merges a couple of yaml files into one yaml file,
// a field in latter file will override the same field in former file.
func Merge(yamls [][]byte) ([]byte, error) {
m, err := MergeAndUnmarshal(yamls)
if err != nil {
return nil, err
}
return yaml.Marshal(m)
}
func mergeMaps(a, b map[string]interface{}) map[string]interface{} {
out := make(map[string]interface{}, len(a))
for k, v := range a {
out[k] = v
}
for k, v := range b {
if v, ok := v.(map[string]interface{}); ok {
if bv, ok := out[k]; ok {
if bv, ok := bv.(map[string]interface{}); ok {
out[k] = mergeMaps(bv, v)
continue
}
}
}
out[k] = v
}
return out
}