diff_test.go 8.6 KB

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