Tool to help you manage your Grafana dashboards using Git.

patch_test.go 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339
  1. // Copyright (c) 2012-2016 The go-diff authors. All rights reserved.
  2. // https://github.com/sergi/go-diff
  3. // See the included LICENSE file for license details.
  4. //
  5. // go-diff is a Go implementation of Google's Diff, Match, and Patch library
  6. // Original library is Copyright (c) 2006 Google Inc.
  7. // http://code.google.com/p/google-diff-match-patch/
  8. package diffmatchpatch
  9. import (
  10. "fmt"
  11. "strings"
  12. "testing"
  13. "github.com/stretchr/testify/assert"
  14. )
  15. func TestPatchString(t *testing.T) {
  16. type TestCase struct {
  17. Patch Patch
  18. Expected string
  19. }
  20. for i, tc := range []TestCase{
  21. {
  22. Patch: Patch{
  23. Start1: 20,
  24. Start2: 21,
  25. Length1: 18,
  26. Length2: 17,
  27. diffs: []Diff{
  28. {DiffEqual, "jump"},
  29. {DiffDelete, "s"},
  30. {DiffInsert, "ed"},
  31. {DiffEqual, " over "},
  32. {DiffDelete, "the"},
  33. {DiffInsert, "a"},
  34. {DiffEqual, "\nlaz"},
  35. },
  36. },
  37. Expected: "@@ -21,18 +22,17 @@\n jump\n-s\n+ed\n over \n-the\n+a\n %0Alaz\n",
  38. },
  39. } {
  40. actual := tc.Patch.String()
  41. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %#v", i, tc))
  42. }
  43. }
  44. func TestPatchFromText(t *testing.T) {
  45. type TestCase struct {
  46. Patch string
  47. ErrorMessagePrefix string
  48. }
  49. dmp := New()
  50. for i, tc := range []TestCase{
  51. {"", ""},
  52. {"@@ -21,18 +22,17 @@\n jump\n-s\n+ed\n over \n-the\n+a\n %0Alaz\n", ""},
  53. {"@@ -1 +1 @@\n-a\n+b\n", ""},
  54. {"@@ -1,3 +0,0 @@\n-abc\n", ""},
  55. {"@@ -0,0 +1,3 @@\n+abc\n", ""},
  56. {"@@ _0,0 +0,0 @@\n+abc\n", "Invalid patch string: @@ _0,0 +0,0 @@"},
  57. {"Bad\nPatch\n", "Invalid patch string"},
  58. } {
  59. patches, err := dmp.PatchFromText(tc.Patch)
  60. if tc.ErrorMessagePrefix == "" {
  61. assert.Nil(t, err)
  62. if tc.Patch == "" {
  63. assert.Equal(t, []Patch{}, patches, fmt.Sprintf("Test case #%d, %#v", i, tc))
  64. } else {
  65. assert.Equal(t, tc.Patch, patches[0].String(), fmt.Sprintf("Test case #%d, %#v", i, tc))
  66. }
  67. } else {
  68. e := err.Error()
  69. if strings.HasPrefix(e, tc.ErrorMessagePrefix) {
  70. e = tc.ErrorMessagePrefix
  71. }
  72. assert.Equal(t, tc.ErrorMessagePrefix, e)
  73. }
  74. }
  75. diffs := []Diff{
  76. {DiffDelete, "`1234567890-=[]\\;',./"},
  77. {DiffInsert, "~!@#$%^&*()_+{}|:\"<>?"},
  78. }
  79. patches, err := dmp.PatchFromText("@@ -1,21 +1,21 @@\n-%601234567890-=%5B%5D%5C;',./\n+~!@#$%25%5E&*()_+%7B%7D%7C:%22%3C%3E?\n")
  80. assert.Len(t, patches, 1)
  81. assert.Equal(t, diffs,
  82. patches[0].diffs,
  83. )
  84. assert.Nil(t, err)
  85. }
  86. func TestPatchToText(t *testing.T) {
  87. type TestCase struct {
  88. Patch string
  89. }
  90. dmp := New()
  91. for i, tc := range []TestCase{
  92. {"@@ -21,18 +22,17 @@\n jump\n-s\n+ed\n over \n-the\n+a\n laz\n"},
  93. {"@@ -1,9 +1,9 @@\n-f\n+F\n oo+fooba\n@@ -7,9 +7,9 @@\n obar\n-,\n+.\n tes\n"},
  94. } {
  95. patches, err := dmp.PatchFromText(tc.Patch)
  96. assert.Nil(t, err)
  97. actual := dmp.PatchToText(patches)
  98. assert.Equal(t, tc.Patch, actual, fmt.Sprintf("Test case #%d, %#v", i, tc))
  99. }
  100. }
  101. func TestPatchAddContext(t *testing.T) {
  102. type TestCase struct {
  103. Name string
  104. Patch string
  105. Text string
  106. Expected string
  107. }
  108. dmp := New()
  109. dmp.PatchMargin = 4
  110. for i, tc := range []TestCase{
  111. {"Simple case", "@@ -21,4 +21,10 @@\n-jump\n+somersault\n", "The quick brown fox jumps over the lazy dog.", "@@ -17,12 +17,18 @@\n fox \n-jump\n+somersault\n s ov\n"},
  112. {"Not enough trailing context", "@@ -21,4 +21,10 @@\n-jump\n+somersault\n", "The quick brown fox jumps.", "@@ -17,10 +17,16 @@\n fox \n-jump\n+somersault\n s.\n"},
  113. {"Not enough leading context", "@@ -3 +3,2 @@\n-e\n+at\n", "The quick brown fox jumps.", "@@ -1,7 +1,8 @@\n Th\n-e\n+at\n qui\n"},
  114. {"Ambiguity", "@@ -3 +3,2 @@\n-e\n+at\n", "The quick brown fox jumps. The quick brown fox crashes.", "@@ -1,27 +1,28 @@\n Th\n-e\n+at\n quick brown fox jumps. \n"},
  115. } {
  116. patches, err := dmp.PatchFromText(tc.Patch)
  117. assert.Nil(t, err)
  118. actual := dmp.PatchAddContext(patches[0], tc.Text)
  119. assert.Equal(t, tc.Expected, actual.String(), fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  120. }
  121. }
  122. func TestPatchMakeAndPatchToText(t *testing.T) {
  123. type TestCase struct {
  124. Name string
  125. Input1 interface{}
  126. Input2 interface{}
  127. Input3 interface{}
  128. Expected string
  129. }
  130. dmp := New()
  131. text1 := "The quick brown fox jumps over the lazy dog."
  132. text2 := "That quick brown fox jumped over a lazy dog."
  133. for i, tc := range []TestCase{
  134. {"Null case", "", "", nil, ""},
  135. {"Text2+Text1 inputs", text2, text1, nil, "@@ -1,8 +1,7 @@\n Th\n-at\n+e\n qui\n@@ -21,17 +21,18 @@\n jump\n-ed\n+s\n over \n-a\n+the\n laz\n"},
  136. {"Text1+Text2 inputs", text1, text2, nil, "@@ -1,11 +1,12 @@\n Th\n-e\n+at\n quick b\n@@ -22,18 +22,17 @@\n jump\n-s\n+ed\n over \n-the\n+a\n laz\n"},
  137. {"Diff input", dmp.DiffMain(text1, text2, false), nil, nil, "@@ -1,11 +1,12 @@\n Th\n-e\n+at\n quick b\n@@ -22,18 +22,17 @@\n jump\n-s\n+ed\n over \n-the\n+a\n laz\n"},
  138. {"Text1+Diff inputs", text1, dmp.DiffMain(text1, text2, false), nil, "@@ -1,11 +1,12 @@\n Th\n-e\n+at\n quick b\n@@ -22,18 +22,17 @@\n jump\n-s\n+ed\n over \n-the\n+a\n laz\n"},
  139. {"Text1+Text2+Diff inputs (deprecated)", text1, text2, dmp.DiffMain(text1, text2, false), "@@ -1,11 +1,12 @@\n Th\n-e\n+at\n quick b\n@@ -22,18 +22,17 @@\n jump\n-s\n+ed\n over \n-the\n+a\n laz\n"},
  140. {"Character encoding", "`1234567890-=[]\\;',./", "~!@#$%^&*()_+{}|:\"<>?", nil, "@@ -1,21 +1,21 @@\n-%601234567890-=%5B%5D%5C;',./\n+~!@#$%25%5E&*()_+%7B%7D%7C:%22%3C%3E?\n"},
  141. {"Long string with repeats", strings.Repeat("abcdef", 100), strings.Repeat("abcdef", 100) + "123", nil, "@@ -573,28 +573,31 @@\n cdefabcdefabcdefabcdefabcdef\n+123\n"},
  142. {"Corner case of #31 fixed by #32", "2016-09-01T03:07:14.807830741Z", "2016-09-01T03:07:15.154800781Z", nil, "@@ -15,16 +15,16 @@\n 07:1\n+5.15\n 4\n-.\n 80\n+0\n 78\n-3074\n 1Z\n"},
  143. } {
  144. var patches []Patch
  145. if tc.Input3 != nil {
  146. patches = dmp.PatchMake(tc.Input1, tc.Input2, tc.Input3)
  147. } else if tc.Input2 != nil {
  148. patches = dmp.PatchMake(tc.Input1, tc.Input2)
  149. } else if ps, ok := tc.Input1.([]Patch); ok {
  150. patches = ps
  151. } else {
  152. patches = dmp.PatchMake(tc.Input1)
  153. }
  154. actual := dmp.PatchToText(patches)
  155. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  156. }
  157. // Corner case of #28 wrong patch with timeout of 0
  158. dmp.DiffTimeout = 0
  159. text1 = "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus ut risus et enim consectetur convallis a non ipsum. Sed nec nibh cursus, interdum libero vel."
  160. text2 = "Lorem a ipsum dolor sit amet, consectetur adipiscing elit. Vivamus ut risus et enim consectetur convallis a non ipsum. Sed nec nibh cursus, interdum liberovel."
  161. diffs := dmp.DiffMain(text1, text2, true)
  162. // Additional check that the diff texts are equal to the originals even if we are using DiffMain with checklines=true #29
  163. assert.Equal(t, text1, dmp.DiffText1(diffs))
  164. assert.Equal(t, text2, dmp.DiffText2(diffs))
  165. patches := dmp.PatchMake(text1, diffs)
  166. actual := dmp.PatchToText(patches)
  167. assert.Equal(t, "@@ -1,14 +1,16 @@\n Lorem \n+a \n ipsum do\n@@ -148,13 +148,12 @@\n m libero\n- \n vel.\n", actual)
  168. // Check that empty Patch array is returned for no parameter call
  169. patches = dmp.PatchMake()
  170. assert.Equal(t, []Patch{}, patches)
  171. }
  172. func TestPatchSplitMax(t *testing.T) {
  173. type TestCase struct {
  174. Text1 string
  175. Text2 string
  176. Expected string
  177. }
  178. dmp := New()
  179. for i, tc := range []TestCase{
  180. {"abcdefghijklmnopqrstuvwxyz01234567890", "XabXcdXefXghXijXklXmnXopXqrXstXuvXwxXyzX01X23X45X67X89X0", "@@ -1,32 +1,46 @@\n+X\n ab\n+X\n cd\n+X\n ef\n+X\n gh\n+X\n ij\n+X\n kl\n+X\n mn\n+X\n op\n+X\n qr\n+X\n st\n+X\n uv\n+X\n wx\n+X\n yz\n+X\n 012345\n@@ -25,13 +39,18 @@\n zX01\n+X\n 23\n+X\n 45\n+X\n 67\n+X\n 89\n+X\n 0\n"},
  181. {"abcdef1234567890123456789012345678901234567890123456789012345678901234567890uvwxyz", "abcdefuvwxyz", "@@ -3,78 +3,8 @@\n cdef\n-1234567890123456789012345678901234567890123456789012345678901234567890\n uvwx\n"},
  182. {"1234567890123456789012345678901234567890123456789012345678901234567890", "abc", "@@ -1,32 +1,4 @@\n-1234567890123456789012345678\n 9012\n@@ -29,32 +1,4 @@\n-9012345678901234567890123456\n 7890\n@@ -57,14 +1,3 @@\n-78901234567890\n+abc\n"},
  183. {"abcdefghij , h : 0 , t : 1 abcdefghij , h : 0 , t : 1 abcdefghij , h : 0 , t : 1", "abcdefghij , h : 1 , t : 1 abcdefghij , h : 1 , t : 1 abcdefghij , h : 0 , t : 1", "@@ -2,32 +2,32 @@\n bcdefghij , h : \n-0\n+1\n , t : 1 abcdef\n@@ -29,32 +29,32 @@\n bcdefghij , h : \n-0\n+1\n , t : 1 abcdef\n"},
  184. } {
  185. patches := dmp.PatchMake(tc.Text1, tc.Text2)
  186. patches = dmp.PatchSplitMax(patches)
  187. actual := dmp.PatchToText(patches)
  188. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %#v", i, tc))
  189. }
  190. }
  191. func TestPatchAddPadding(t *testing.T) {
  192. type TestCase struct {
  193. Name string
  194. Text1 string
  195. Text2 string
  196. Expected string
  197. ExpectedWithPadding string
  198. }
  199. dmp := New()
  200. for i, tc := range []TestCase{
  201. {"Both edges full", "", "test", "@@ -0,0 +1,4 @@\n+test\n", "@@ -1,8 +1,12 @@\n %01%02%03%04\n+test\n %01%02%03%04\n"},
  202. {"Both edges partial", "XY", "XtestY", "@@ -1,2 +1,6 @@\n X\n+test\n Y\n", "@@ -2,8 +2,12 @@\n %02%03%04X\n+test\n Y%01%02%03\n"},
  203. {"Both edges none", "XXXXYYYY", "XXXXtestYYYY", "@@ -1,8 +1,12 @@\n XXXX\n+test\n YYYY\n", "@@ -5,8 +5,12 @@\n XXXX\n+test\n YYYY\n"},
  204. } {
  205. patches := dmp.PatchMake(tc.Text1, tc.Text2)
  206. actual := dmp.PatchToText(patches)
  207. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  208. dmp.PatchAddPadding(patches)
  209. actualWithPadding := dmp.PatchToText(patches)
  210. assert.Equal(t, tc.ExpectedWithPadding, actualWithPadding, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  211. }
  212. }
  213. func TestPatchApply(t *testing.T) {
  214. type TestCase struct {
  215. Name string
  216. Text1 string
  217. Text2 string
  218. TextBase string
  219. Expected string
  220. ExpectedApplies []bool
  221. }
  222. dmp := New()
  223. dmp.MatchDistance = 1000
  224. dmp.MatchThreshold = 0.5
  225. dmp.PatchDeleteThreshold = 0.5
  226. for i, tc := range []TestCase{
  227. {"Null case", "", "", "Hello world.", "Hello world.", []bool{}},
  228. {"Exact match", "The quick brown fox jumps over the lazy dog.", "That quick brown fox jumped over a lazy dog.", "The quick brown fox jumps over the lazy dog.", "That quick brown fox jumped over a lazy dog.", []bool{true, true}},
  229. {"Partial match", "The quick brown fox jumps over the lazy dog.", "That quick brown fox jumped over a lazy dog.", "The quick red rabbit jumps over the tired tiger.", "That quick red rabbit jumped over a tired tiger.", []bool{true, true}},
  230. {"Failed match", "The quick brown fox jumps over the lazy dog.", "That quick brown fox jumped over a lazy dog.", "I am the very model of a modern major general.", "I am the very model of a modern major general.", []bool{false, false}},
  231. {"Big delete, small Diff", "x1234567890123456789012345678901234567890123456789012345678901234567890y", "xabcy", "x123456789012345678901234567890-----++++++++++-----123456789012345678901234567890y", "xabcy", []bool{true, true}},
  232. {"Big delete, big Diff 1", "x1234567890123456789012345678901234567890123456789012345678901234567890y", "xabcy", "x12345678901234567890---------------++++++++++---------------12345678901234567890y", "xabc12345678901234567890---------------++++++++++---------------12345678901234567890y", []bool{false, true}},
  233. } {
  234. patches := dmp.PatchMake(tc.Text1, tc.Text2)
  235. actual, actualApplies := dmp.PatchApply(patches, tc.TextBase)
  236. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  237. assert.Equal(t, tc.ExpectedApplies, actualApplies, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  238. }
  239. dmp.PatchDeleteThreshold = 0.6
  240. for i, tc := range []TestCase{
  241. {"Big delete, big Diff 2", "x1234567890123456789012345678901234567890123456789012345678901234567890y", "xabcy", "x12345678901234567890---------------++++++++++---------------12345678901234567890y", "xabcy", []bool{true, true}},
  242. } {
  243. patches := dmp.PatchMake(tc.Text1, tc.Text2)
  244. actual, actualApplies := dmp.PatchApply(patches, tc.TextBase)
  245. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  246. assert.Equal(t, tc.ExpectedApplies, actualApplies, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  247. }
  248. dmp.MatchDistance = 0
  249. dmp.MatchThreshold = 0.0
  250. dmp.PatchDeleteThreshold = 0.5
  251. for i, tc := range []TestCase{
  252. {"Compensate for failed patch", "abcdefghijklmnopqrstuvwxyz--------------------1234567890", "abcXXXXXXXXXXdefghijklmnopqrstuvwxyz--------------------1234567YYYYYYYYYY890", "ABCDEFGHIJKLMNOPQRSTUVWXYZ--------------------1234567890", "ABCDEFGHIJKLMNOPQRSTUVWXYZ--------------------1234567YYYYYYYYYY890", []bool{false, true}},
  253. } {
  254. patches := dmp.PatchMake(tc.Text1, tc.Text2)
  255. actual, actualApplies := dmp.PatchApply(patches, tc.TextBase)
  256. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  257. assert.Equal(t, tc.ExpectedApplies, actualApplies, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  258. }
  259. dmp.MatchThreshold = 0.5
  260. dmp.MatchDistance = 1000
  261. for i, tc := range []TestCase{
  262. {"No side effects", "", "test", "", "test", []bool{true}},
  263. {"No side effects with major delete", "The quick brown fox jumps over the lazy dog.", "Woof", "The quick brown fox jumps over the lazy dog.", "Woof", []bool{true, true}},
  264. {"Edge exact match", "", "test", "", "test", []bool{true}},
  265. {"Near edge exact match", "XY", "XtestY", "XY", "XtestY", []bool{true}},
  266. {"Edge partial match", "y", "y123", "x", "x123", []bool{true}},
  267. } {
  268. patches := dmp.PatchMake(tc.Text1, tc.Text2)
  269. actual, actualApplies := dmp.PatchApply(patches, tc.TextBase)
  270. assert.Equal(t, tc.Expected, actual, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  271. assert.Equal(t, tc.ExpectedApplies, actualApplies, fmt.Sprintf("Test case #%d, %s", i, tc.Name))
  272. }
  273. }