merge.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. package jsonmerge
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "reflect"
  7. "strconv"
  8. "strings"
  9. )
  10. // Merger describes result of merge operation and provides
  11. // configuration.
  12. type Merger struct {
  13. // Errors is slice of non-critical errors of merge operations
  14. Errors []error
  15. // Replaced is describe replacements
  16. // Key is path in document like
  17. // "prop1.prop2.prop3" for object properties or
  18. // "arr1.1.prop" for arrays
  19. // Value is value of replacemet
  20. Replaced map[string]interface{}
  21. // CopyNonexistent enables setting fields into the result
  22. // which only exist in the patch.
  23. CopyNonexistent bool
  24. }
  25. func (m *Merger) mergeValue(path []string, patch map[string]interface{}, key string, value interface{}) interface{} {
  26. patchValue, patchHasValue := patch[key]
  27. if !patchHasValue {
  28. return value
  29. }
  30. _, patchValueIsObject := patchValue.(map[string]interface{})
  31. path = append(path, key)
  32. pathStr := strings.Join(path, ".")
  33. if _, ok := value.(map[string]interface{}); ok {
  34. if !patchValueIsObject {
  35. err := fmt.Errorf("patch value must be object for key \"%v\"", pathStr)
  36. m.Errors = append(m.Errors, err)
  37. return value
  38. }
  39. return m.mergeObjects(value, patchValue, path)
  40. }
  41. if _, ok := value.([]interface{}); ok && patchValueIsObject {
  42. return m.mergeObjects(value, patchValue, path)
  43. }
  44. if !reflect.DeepEqual(value, patchValue) {
  45. m.Replaced[pathStr] = patchValue
  46. }
  47. return patchValue
  48. }
  49. func (m *Merger) mergeObjects(data, patch interface{}, path []string) interface{} {
  50. if patchObject, ok := patch.(map[string]interface{}); ok {
  51. if dataArray, ok := data.([]interface{}); ok {
  52. ret := make([]interface{}, len(dataArray))
  53. for i, val := range dataArray {
  54. ret[i] = m.mergeValue(path, patchObject, strconv.Itoa(i), val)
  55. }
  56. return ret
  57. } else if dataObject, ok := data.(map[string]interface{}); ok {
  58. ret := make(map[string]interface{})
  59. for k, v := range dataObject {
  60. ret[k] = m.mergeValue(path, patchObject, k, v)
  61. }
  62. if m.CopyNonexistent {
  63. for k, v := range patchObject {
  64. if _, ok := dataObject[k]; !ok {
  65. ret[k] = v
  66. }
  67. }
  68. }
  69. return ret
  70. }
  71. }
  72. return data
  73. }
  74. // Merge merges patch document to data document
  75. //
  76. // Returning merged document. Result of merge operation can be
  77. // obtained from the Merger. Result information is discarded before
  78. // merging.
  79. func (m *Merger) Merge(data, patch interface{}) interface{} {
  80. m.Replaced = make(map[string]interface{})
  81. m.Errors = make([]error, 0)
  82. return m.mergeObjects(data, patch, nil)
  83. }
  84. // MergeBytesIndent merges patch document buffer to data document buffer
  85. //
  86. // Use prefix and indent for set indentation like in json.MarshalIndent
  87. //
  88. // Returning merged document buffer and error if any.
  89. func (m *Merger) MergeBytesIndent(dataBuff, patchBuff []byte, prefix, indent string) (mergedBuff []byte, err error) {
  90. var data, patch, merged interface{}
  91. err = unmarshalJSON(dataBuff, &data)
  92. if err != nil {
  93. err = fmt.Errorf("error in data JSON: %v", err)
  94. return
  95. }
  96. err = unmarshalJSON(patchBuff, &patch)
  97. if err != nil {
  98. err = fmt.Errorf("error in patch JSON: %v", err)
  99. return
  100. }
  101. merged = m.Merge(data, patch)
  102. mergedBuff, err = json.MarshalIndent(merged, prefix, indent)
  103. if err != nil {
  104. err = fmt.Errorf("error writing merged JSON: %v", err)
  105. }
  106. return
  107. }
  108. // MergeBytes merges patch document buffer to data document buffer
  109. //
  110. // Returning merged document buffer, merge info and
  111. // error if any
  112. func (m *Merger) MergeBytes(dataBuff, patchBuff []byte) (mergedBuff []byte, err error) {
  113. var data, patch, merged interface{}
  114. err = unmarshalJSON(dataBuff, &data)
  115. if err != nil {
  116. err = fmt.Errorf("error in data JSON: %v", err)
  117. return
  118. }
  119. err = unmarshalJSON(patchBuff, &patch)
  120. if err != nil {
  121. err = fmt.Errorf("error in patch JSON: %v", err)
  122. return
  123. }
  124. merged = m.Merge(data, patch)
  125. mergedBuff, err = json.Marshal(merged)
  126. if err != nil {
  127. err = fmt.Errorf("error writing merged JSON: %v", err)
  128. }
  129. return
  130. }
  131. func unmarshalJSON(buff []byte, data interface{}) error {
  132. decoder := json.NewDecoder(bytes.NewReader(buff))
  133. decoder.UseNumber()
  134. return decoder.Decode(data)
  135. }