diff_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. package plumbing_test
  2. import (
  3. "testing"
  4. "unicode/utf8"
  5. "github.com/sergi/go-diff/diffmatchpatch"
  6. "github.com/stretchr/testify/assert"
  7. "gopkg.in/src-d/go-git.v4/plumbing"
  8. "gopkg.in/src-d/go-git.v4/plumbing/object"
  9. "gopkg.in/src-d/hercules.v10"
  10. "gopkg.in/src-d/hercules.v10/internal"
  11. "gopkg.in/src-d/hercules.v10/internal/core"
  12. items "gopkg.in/src-d/hercules.v10/internal/plumbing"
  13. "gopkg.in/src-d/hercules.v10/internal/test"
  14. "gopkg.in/src-d/hercules.v10/internal/test/fixtures"
  15. )
  16. func TestFileDiffMeta(t *testing.T) {
  17. fd := fixtures.FileDiff()
  18. assert.Equal(t, fd.Name(), "FileDiff")
  19. assert.Equal(t, len(fd.Provides()), 1)
  20. assert.Equal(t, fd.Provides()[0], items.DependencyFileDiff)
  21. assert.Equal(t, len(fd.Requires()), 2)
  22. assert.Equal(t, fd.Requires()[0], items.DependencyTreeChanges)
  23. assert.Equal(t, fd.Requires()[1], items.DependencyBlobCache)
  24. assert.Len(t, fd.ListConfigurationOptions(), 2)
  25. assert.Equal(t, fd.ListConfigurationOptions()[0].Name, items.ConfigFileDiffDisableCleanup)
  26. assert.Equal(t, fd.ListConfigurationOptions()[1].Name, items.ConfigFileWhitespaceIgnore)
  27. assert.NoError(t, fd.Configure(map[string]interface{}{
  28. core.ConfigLogger: core.NewLogger(),
  29. items.ConfigFileDiffDisableCleanup: true,
  30. items.ConfigFileWhitespaceIgnore: true,
  31. }))
  32. assert.True(t, fd.CleanupDisabled)
  33. assert.True(t, fd.WhitespaceIgnore)
  34. }
  35. func TestFileDiffRegistration(t *testing.T) {
  36. summoned := core.Registry.Summon((&items.FileDiff{}).Name())
  37. assert.Len(t, summoned, 1)
  38. assert.Equal(t, summoned[0].Name(), "FileDiff")
  39. summoned = core.Registry.Summon((&items.FileDiff{}).Provides()[0])
  40. assert.True(t, len(summoned) >= 1)
  41. matched := false
  42. for _, tp := range summoned {
  43. matched = matched || tp.Name() == "FileDiff"
  44. }
  45. assert.True(t, matched)
  46. }
  47. func TestFileDiffConsume(t *testing.T) {
  48. fd := fixtures.FileDiff()
  49. deps := map[string]interface{}{}
  50. cache := map[plumbing.Hash]*items.CachedBlob{}
  51. items.AddHash(t, cache, "291286b4ac41952cbd1389fda66420ec03c1a9fe")
  52. items.AddHash(t, cache, "334cde09da4afcb74f8d2b3e6fd6cce61228b485")
  53. items.AddHash(t, cache, "dc248ba2b22048cc730c571a748e8ffcf7085ab9")
  54. deps[items.DependencyBlobCache] = cache
  55. changes := make(object.Changes, 3)
  56. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  57. "a1eb2ea76eb7f9bfbde9b243861474421000eb96"))
  58. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  59. "994eac1cd07235bb9815e547a75c84265dea00f5"))
  60. changes[0] = &object.Change{From: object.ChangeEntry{
  61. Name: "analyser.go",
  62. Tree: treeFrom,
  63. TreeEntry: object.TreeEntry{
  64. Name: "analyser.go",
  65. Mode: 0100644,
  66. Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),
  67. },
  68. }, To: object.ChangeEntry{
  69. Name: "analyser.go",
  70. Tree: treeTo,
  71. TreeEntry: object.TreeEntry{
  72. Name: "analyser.go",
  73. Mode: 0100644,
  74. Hash: plumbing.NewHash("334cde09da4afcb74f8d2b3e6fd6cce61228b485"),
  75. },
  76. }}
  77. changes[1] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  78. Name: ".travis.yml",
  79. Tree: treeTo,
  80. TreeEntry: object.TreeEntry{
  81. Name: ".travis.yml",
  82. Mode: 0100644,
  83. Hash: plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"),
  84. },
  85. },
  86. }
  87. changes[2] = &object.Change{From: object.ChangeEntry{
  88. Name: "rbtree.go",
  89. Tree: treeFrom,
  90. TreeEntry: object.TreeEntry{
  91. Name: "rbtree.go",
  92. Mode: 0100644,
  93. Hash: plumbing.NewHash("14c3fa5a1cca103032f10379467a3a2f210e5f94"),
  94. },
  95. }, To: object.ChangeEntry{},
  96. }
  97. deps[items.DependencyTreeChanges] = changes
  98. res, err := fd.Consume(deps)
  99. assert.Nil(t, err)
  100. diffs := res[items.DependencyFileDiff].(map[string]items.FileDiffData)
  101. assert.Equal(t, len(diffs), 1)
  102. diff := diffs["analyser.go"]
  103. assert.Equal(t, diff.OldLinesOfCode, 307)
  104. assert.Equal(t, diff.NewLinesOfCode, 309)
  105. deletions := 0
  106. insertions := 0
  107. for _, edit := range diff.Diffs {
  108. switch edit.Type {
  109. case diffmatchpatch.DiffEqual:
  110. continue
  111. case diffmatchpatch.DiffInsert:
  112. insertions += utf8.RuneCountInString(edit.Text)
  113. case diffmatchpatch.DiffDelete:
  114. deletions += utf8.RuneCountInString(edit.Text)
  115. }
  116. }
  117. assert.Equal(t, deletions, 13)
  118. assert.Equal(t, insertions, 15)
  119. }
  120. func TestFileDiffConsumeInvalidBlob(t *testing.T) {
  121. fd := fixtures.FileDiff()
  122. deps := map[string]interface{}{}
  123. cache := map[plumbing.Hash]*items.CachedBlob{}
  124. items.AddHash(t, cache, "291286b4ac41952cbd1389fda66420ec03c1a9fe")
  125. items.AddHash(t, cache, "334cde09da4afcb74f8d2b3e6fd6cce61228b485")
  126. items.AddHash(t, cache, "dc248ba2b22048cc730c571a748e8ffcf7085ab9")
  127. cache[plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff")] = &items.CachedBlob{}
  128. deps[items.DependencyBlobCache] = cache
  129. changes := make(object.Changes, 1)
  130. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  131. "a1eb2ea76eb7f9bfbde9b243861474421000eb96"))
  132. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  133. "994eac1cd07235bb9815e547a75c84265dea00f5"))
  134. changes[0] = &object.Change{From: object.ChangeEntry{
  135. Name: "analyser.go",
  136. Tree: treeFrom,
  137. TreeEntry: object.TreeEntry{
  138. Name: "analyser.go",
  139. Mode: 0100644,
  140. Hash: plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff"),
  141. },
  142. }, To: object.ChangeEntry{
  143. Name: "analyser.go",
  144. Tree: treeTo,
  145. TreeEntry: object.TreeEntry{
  146. Name: "analyser.go",
  147. Mode: 0100644,
  148. Hash: plumbing.NewHash("334cde09da4afcb74f8d2b3e6fd6cce61228b485"),
  149. },
  150. }}
  151. deps[items.DependencyTreeChanges] = changes
  152. res, err := fd.Consume(deps)
  153. assert.Len(t, res[hercules.DependencyFileDiff].(map[string]items.FileDiffData), 1)
  154. assert.Nil(t, err)
  155. changes[0] = &object.Change{From: object.ChangeEntry{
  156. Name: "analyser.go",
  157. Tree: treeFrom,
  158. TreeEntry: object.TreeEntry{
  159. Name: "analyser.go",
  160. Mode: 0100644,
  161. Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),
  162. },
  163. }, To: object.ChangeEntry{
  164. Name: "analyser.go",
  165. Tree: treeTo,
  166. TreeEntry: object.TreeEntry{
  167. Name: "analyser.go",
  168. Mode: 0100644,
  169. Hash: plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff"),
  170. },
  171. }}
  172. res, err = fd.Consume(deps)
  173. assert.Len(t, res[hercules.DependencyFileDiff].(map[string]items.FileDiffData), 1)
  174. assert.Nil(t, err)
  175. }
  176. func TestCountLines(t *testing.T) {
  177. blob, err := test.Repository.BlobObject(
  178. plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"))
  179. assert.Nil(t, err)
  180. cb := &items.CachedBlob{Blob: *blob}
  181. cb.Cache()
  182. lines, err := cb.CountLines()
  183. assert.Equal(t, lines, 12)
  184. assert.Nil(t, err)
  185. blob, err = internal.CreateDummyBlob(
  186. plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"), true)
  187. assert.Nil(t, err)
  188. cb = &items.CachedBlob{Blob: *blob}
  189. err = cb.Cache()
  190. assert.Equal(t, err.Error(), "dummy failure")
  191. // test_data/blob
  192. blob, err = test.Repository.BlobObject(
  193. plumbing.NewHash("c86626638e0bc8cf47ca49bb1525b40e9737ee64"))
  194. assert.Nil(t, err)
  195. cb = &items.CachedBlob{Blob: *blob}
  196. cb.Cache()
  197. lines, err = cb.CountLines()
  198. assert.Equal(t, lines, 0)
  199. assert.NotNil(t, err)
  200. assert.Equal(t, err.Error(), items.ErrorBinary.Error())
  201. }
  202. func TestBlobToString(t *testing.T) {
  203. blob, _ := test.Repository.BlobObject(
  204. plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"))
  205. cb := &items.CachedBlob{Blob: *blob}
  206. err := cb.Cache()
  207. assert.Nil(t, err)
  208. str := string(cb.Data)
  209. assert.Equal(t, str, `language: go
  210. go:
  211. - 1.7
  212. go_import_path: gopkg.in/src-d/hercules.v1
  213. `+" "+`
  214. script:
  215. - go test -v -cpu=1,2 ./...
  216. notifications:
  217. email: false
  218. `)
  219. blob, _ = internal.CreateDummyBlob(
  220. plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"), true)
  221. cb = &items.CachedBlob{Blob: *blob}
  222. err = cb.Cache()
  223. assert.NotNil(t, err)
  224. }
  225. func TestFileDiffDarkMagic(t *testing.T) {
  226. fd := fixtures.FileDiff()
  227. deps := map[string]interface{}{}
  228. cache := map[plumbing.Hash]*items.CachedBlob{}
  229. items.AddHash(t, cache, "448eb3f312849b0ca766063d06b09481c987b309") // 1.java
  230. items.AddHash(t, cache, "3312c92f3e8bdfbbdb30bccb6acd1b85bc338dfc") // 2.java
  231. deps[items.DependencyBlobCache] = cache
  232. changes := make(object.Changes, 1)
  233. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  234. "f02289bfe843388a1bb3c7dea210374082dd86b9"))
  235. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  236. "eca91acf1fd828f20dcb653a061d8c97d965bc6c"))
  237. changes[0] = &object.Change{From: object.ChangeEntry{
  238. Name: "test.java",
  239. Tree: treeFrom,
  240. TreeEntry: object.TreeEntry{
  241. Name: "test.java",
  242. Mode: 0100644,
  243. Hash: plumbing.NewHash("448eb3f312849b0ca766063d06b09481c987b309"),
  244. },
  245. }, To: object.ChangeEntry{
  246. Name: "test.java",
  247. Tree: treeTo,
  248. TreeEntry: object.TreeEntry{
  249. Name: "test.java",
  250. Mode: 0100644,
  251. Hash: plumbing.NewHash("3312c92f3e8bdfbbdb30bccb6acd1b85bc338dfc"),
  252. },
  253. }}
  254. deps[items.DependencyTreeChanges] = changes
  255. res, err := fd.Consume(deps)
  256. assert.Nil(t, err)
  257. magicDiffs := res[items.DependencyFileDiff].(map[string]items.FileDiffData)["test.java"]
  258. fd.CleanupDisabled = true
  259. res, err = fd.Consume(deps)
  260. assert.Nil(t, err)
  261. plainDiffs := res[items.DependencyFileDiff].(map[string]items.FileDiffData)["test.java"]
  262. assert.NotEqual(t, magicDiffs.Diffs, plainDiffs.Diffs)
  263. assert.Equal(t, magicDiffs.OldLinesOfCode, plainDiffs.OldLinesOfCode)
  264. assert.Equal(t, magicDiffs.NewLinesOfCode, plainDiffs.NewLinesOfCode)
  265. }
  266. func TestFileDiffWhitespaceDarkMagic(t *testing.T) {
  267. fd := fixtures.FileDiff()
  268. deps := map[string]interface{}{}
  269. cache := map[plumbing.Hash]*items.CachedBlob{}
  270. items.AddHash(t, cache, "448eb3f312849b0ca766063d06b09481c987b309") // 1.java
  271. items.AddHash(t, cache, "3312c92f3e8bdfbbdb30bccb6acd1b85bc338dfc") // 2.java
  272. deps[items.DependencyBlobCache] = cache
  273. changes := make(object.Changes, 1)
  274. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  275. "f02289bfe843388a1bb3c7dea210374082dd86b9"))
  276. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  277. "eca91acf1fd828f20dcb653a061d8c97d965bc6c"))
  278. changes[0] = &object.Change{From: object.ChangeEntry{
  279. Name: "test.java",
  280. Tree: treeFrom,
  281. TreeEntry: object.TreeEntry{
  282. Name: "test.java",
  283. Mode: 0100644,
  284. Hash: plumbing.NewHash("448eb3f312849b0ca766063d06b09481c987b309"),
  285. },
  286. }, To: object.ChangeEntry{
  287. Name: "test.java",
  288. Tree: treeTo,
  289. TreeEntry: object.TreeEntry{
  290. Name: "test.java",
  291. Mode: 0100644,
  292. Hash: plumbing.NewHash("3312c92f3e8bdfbbdb30bccb6acd1b85bc338dfc"),
  293. },
  294. }}
  295. deps[items.DependencyTreeChanges] = changes
  296. res, err := fd.Consume(deps)
  297. assert.Nil(t, err)
  298. magicDiffs := res[items.DependencyFileDiff].(map[string]items.FileDiffData)["test.java"]
  299. fd.WhitespaceIgnore = true
  300. res, err = fd.Consume(deps)
  301. assert.Nil(t, err)
  302. plainDiffs := res[items.DependencyFileDiff].(map[string]items.FileDiffData)["test.java"]
  303. assert.NotEqual(t, magicDiffs.Diffs, plainDiffs.Diffs)
  304. assert.Equal(t, magicDiffs.OldLinesOfCode, plainDiffs.OldLinesOfCode)
  305. assert.Equal(t, magicDiffs.NewLinesOfCode, plainDiffs.NewLinesOfCode)
  306. }
  307. func TestFileDiffFork(t *testing.T) {
  308. fd1 := fixtures.FileDiff()
  309. clones := fd1.Fork(1)
  310. assert.Len(t, clones, 1)
  311. fd2 := clones[0].(*items.FileDiff)
  312. assert.True(t, fd1 == fd2)
  313. fd1.Merge([]core.PipelineItem{fd2})
  314. }