renames_test.go 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. package plumbing
  2. import (
  3. "bytes"
  4. "testing"
  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. "gopkg.in/src-d/hercules.v6/internal/core"
  9. "gopkg.in/src-d/hercules.v6/internal/test"
  10. )
  11. func fixtureRenameAnalysis() *RenameAnalysis {
  12. ra := RenameAnalysis{SimilarityThreshold: 80}
  13. ra.Initialize(test.Repository)
  14. return &ra
  15. }
  16. func TestRenameAnalysisMeta(t *testing.T) {
  17. ra := fixtureRenameAnalysis()
  18. assert.Equal(t, ra.Name(), "RenameAnalysis")
  19. assert.Equal(t, len(ra.Provides()), 1)
  20. assert.Equal(t, ra.Provides()[0], DependencyTreeChanges)
  21. assert.Equal(t, len(ra.Requires()), 2)
  22. assert.Equal(t, ra.Requires()[0], DependencyBlobCache)
  23. assert.Equal(t, ra.Requires()[1], DependencyTreeChanges)
  24. opts := ra.ListConfigurationOptions()
  25. assert.Len(t, opts, 1)
  26. assert.Equal(t, opts[0].Name, ConfigRenameAnalysisSimilarityThreshold)
  27. ra.SimilarityThreshold = 0
  28. facts := map[string]interface{}{}
  29. facts[ConfigRenameAnalysisSimilarityThreshold] = 70
  30. ra.Configure(facts)
  31. assert.Equal(t, ra.SimilarityThreshold, 70)
  32. delete(facts, ConfigRenameAnalysisSimilarityThreshold)
  33. ra.Configure(facts)
  34. assert.Equal(t, ra.SimilarityThreshold, 70)
  35. }
  36. func TestRenameAnalysisRegistration(t *testing.T) {
  37. summoned := core.Registry.Summon((&RenameAnalysis{}).Name())
  38. assert.Len(t, summoned, 1)
  39. assert.Equal(t, summoned[0].Name(), "RenameAnalysis")
  40. summoned = core.Registry.Summon((&RenameAnalysis{}).Provides()[0])
  41. assert.True(t, len(summoned) >= 1)
  42. matched := false
  43. for _, tp := range summoned {
  44. matched = matched || tp.Name() == "RenameAnalysis"
  45. }
  46. assert.True(t, matched)
  47. }
  48. func TestRenameAnalysisInitializeInvalidThreshold(t *testing.T) {
  49. ra := RenameAnalysis{SimilarityThreshold: -10}
  50. ra.Initialize(test.Repository)
  51. assert.Equal(t, ra.SimilarityThreshold, RenameAnalysisDefaultThreshold)
  52. ra = RenameAnalysis{SimilarityThreshold: 110}
  53. ra.Initialize(test.Repository)
  54. assert.Equal(t, ra.SimilarityThreshold, RenameAnalysisDefaultThreshold)
  55. ra = RenameAnalysis{SimilarityThreshold: 0}
  56. ra.Initialize(test.Repository)
  57. ra = RenameAnalysis{SimilarityThreshold: 100}
  58. ra.Initialize(test.Repository)
  59. }
  60. func TestRenameAnalysisConsume(t *testing.T) {
  61. ra := fixtureRenameAnalysis()
  62. changes := make(object.Changes, 3)
  63. // 2b1ed978194a94edeabbca6de7ff3b5771d4d665
  64. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  65. "96c6ece9b2f3c7c51b83516400d278dea5605100"))
  66. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  67. "251f2094d7b523d5bcc60e663b6cf38151bf8844"))
  68. changes[0] = &object.Change{From: object.ChangeEntry{
  69. Name: "analyser.go",
  70. Tree: treeFrom,
  71. TreeEntry: object.TreeEntry{
  72. Name: "analyser.go",
  73. Mode: 0100644,
  74. Hash: plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1"),
  75. },
  76. }, To: object.ChangeEntry{},
  77. }
  78. changes[1] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  79. Name: "burndown.go",
  80. Tree: treeTo,
  81. TreeEntry: object.TreeEntry{
  82. Name: "burndown.go",
  83. Mode: 0100644,
  84. Hash: plumbing.NewHash("29c9fafd6a2fae8cd20298c3f60115bc31a4c0f2"),
  85. },
  86. },
  87. }
  88. changes[2] = &object.Change{From: object.ChangeEntry{
  89. Name: "cmd/hercules/main.go",
  90. Tree: treeFrom,
  91. TreeEntry: object.TreeEntry{
  92. Name: "cmd/hercules/main.go",
  93. Mode: 0100644,
  94. Hash: plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"),
  95. },
  96. }, To: object.ChangeEntry{
  97. Name: "cmd/hercules/main.go",
  98. Tree: treeTo,
  99. TreeEntry: object.TreeEntry{
  100. Name: "cmd/hercules/main.go",
  101. Mode: 0100644,
  102. Hash: plumbing.NewHash("f7d918ec500e2f925ecde79b51cc007bac27de72"),
  103. },
  104. },
  105. }
  106. cache := map[plumbing.Hash]*CachedBlob{}
  107. AddHash(t, cache, "baa64828831d174f40140e4b3cfa77d1e917a2c1")
  108. AddHash(t, cache, "29c9fafd6a2fae8cd20298c3f60115bc31a4c0f2")
  109. AddHash(t, cache, "c29112dbd697ad9b401333b80c18a63951bc18d9")
  110. AddHash(t, cache, "f7d918ec500e2f925ecde79b51cc007bac27de72")
  111. deps := map[string]interface{}{}
  112. deps[DependencyBlobCache] = cache
  113. deps[DependencyTreeChanges] = changes
  114. ra.SimilarityThreshold = 37
  115. res, err := ra.Consume(deps)
  116. assert.Nil(t, err)
  117. renamed := res[DependencyTreeChanges].(object.Changes)
  118. assert.Equal(t, len(renamed), 2)
  119. ra.SimilarityThreshold = 39
  120. res, err = ra.Consume(deps)
  121. assert.Nil(t, err)
  122. renamed = res[DependencyTreeChanges].(object.Changes)
  123. assert.Equal(t, len(renamed), 3)
  124. }
  125. func TestSortableChanges(t *testing.T) {
  126. changes := sortableChanges{
  127. sortableChange{
  128. nil, plumbing.NewHash("0000000000000000000000000000000000000000"),
  129. }, sortableChange{
  130. nil, plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff"),
  131. },
  132. }
  133. assert.True(t, changes.Less(0, 1))
  134. assert.False(t, changes.Less(1, 0))
  135. assert.False(t, changes.Less(0, 0))
  136. changes.Swap(0, 1)
  137. assert.Equal(t, changes[0].hash.String(), "ffffffffffffffffffffffffffffffffffffffff")
  138. assert.Equal(t, changes[1].hash.String(), "0000000000000000000000000000000000000000")
  139. }
  140. func TestSortableBlobs(t *testing.T) {
  141. blobs := sortableBlobs{
  142. sortableBlob{
  143. nil, int64(0),
  144. }, sortableBlob{
  145. nil, int64(1),
  146. },
  147. }
  148. assert.True(t, blobs.Less(0, 1))
  149. assert.False(t, blobs.Less(1, 0))
  150. assert.False(t, blobs.Less(0, 0))
  151. blobs.Swap(0, 1)
  152. assert.Equal(t, blobs[0].size, int64(1))
  153. assert.Equal(t, blobs[1].size, int64(0))
  154. }
  155. func TestRenameAnalysisFork(t *testing.T) {
  156. ra1 := fixtureRenameAnalysis()
  157. clones := ra1.Fork(1)
  158. assert.Len(t, clones, 1)
  159. ra2 := clones[0].(*RenameAnalysis)
  160. assert.True(t, ra1 == ra2)
  161. ra1.Merge([]core.PipelineItem{ra2})
  162. }
  163. func TestRenameAnalysisSizesAreClose(t *testing.T) {
  164. ra := fixtureRenameAnalysis()
  165. assert.True(t, ra.sizesAreClose(941, 963))
  166. assert.True(t, ra.sizesAreClose(941, 1150))
  167. assert.True(t, ra.sizesAreClose(941, 803))
  168. assert.False(t, ra.sizesAreClose(1320, 1668))
  169. }
  170. func TestRenameAnalysisSortRenameCandidates(t *testing.T) {
  171. candidates := []int{0, 1, 2, 3}
  172. sortRenameCandidates(candidates, "test_regression.py", func(i int) string {
  173. return []string{"gather_nd_op.h", "test.py", "test_file_system.cc", "regression.py"}[i]
  174. })
  175. assert.Equal(t, candidates[0], 3)
  176. assert.Equal(t, candidates[1], 1)
  177. }
  178. func TestBlobsAreCloseText(t *testing.T) {
  179. blob1 := &CachedBlob{Data: []byte("hello, world!")}
  180. blob2 := &CachedBlob{Data: []byte("hello, world?")}
  181. blob1.Size = int64(len(blob1.Data))
  182. blob2.Size = int64(len(blob2.Data))
  183. ra := fixtureRenameAnalysis()
  184. result, err := ra.blobsAreClose(blob1, blob2)
  185. assert.Nil(t, err)
  186. assert.True(t, result)
  187. blob1.Data = []byte("hello, mloncode")
  188. blob1.Size = int64(len(blob1.Data))
  189. result, err = ra.blobsAreClose(blob1, blob2)
  190. assert.Nil(t, err)
  191. assert.False(t, result)
  192. }
  193. func TestBlobsAreCloseBinary(t *testing.T) {
  194. blob1 := &CachedBlob{}
  195. blob2 := &CachedBlob{}
  196. ra := fixtureRenameAnalysis()
  197. result, err := ra.blobsAreClose(blob1, blob2)
  198. assert.Nil(t, err)
  199. assert.True(t, result)
  200. blob1.Data = make([]byte, 100)
  201. blob2.Data = blob1.Data
  202. blob1.Size = 100
  203. blob2.Size = 100
  204. result, err = ra.blobsAreClose(blob1, blob2)
  205. assert.Nil(t, err)
  206. assert.True(t, result)
  207. blob1.Data = bytes.Repeat([]byte{1}, 100)
  208. blob2.Data = blob1.Data
  209. result, err = ra.blobsAreClose(blob1, blob2)
  210. assert.Nil(t, err)
  211. assert.True(t, result)
  212. for i := 0; i < 100; i++ {
  213. blob1.Data[i] = byte(i)
  214. }
  215. result, err = ra.blobsAreClose(blob1, blob2)
  216. assert.Nil(t, err)
  217. assert.True(t, result)
  218. blob2.Data = make([]byte, 100)
  219. for i := 0; i < 80; i++ {
  220. blob2.Data[i] = byte(i)
  221. }
  222. result, err = ra.blobsAreClose(blob1, blob2)
  223. assert.Nil(t, err)
  224. assert.True(t, result)
  225. blob2.Data[79] = 0
  226. result, err = ra.blobsAreClose(blob1, blob2)
  227. assert.Nil(t, err)
  228. assert.False(t, result)
  229. blob1.Data = []byte("hello, world!")
  230. blob1.Size = int64(len(blob2.Data))
  231. result, err = ra.blobsAreClose(blob1, blob2)
  232. assert.Nil(t, err)
  233. assert.False(t, result)
  234. blob1.Data, blob2.Data = blob2.Data, blob1.Data
  235. blob1.Size, blob2.Size = blob2.Size, blob1.Size
  236. result, err = ra.blobsAreClose(blob1, blob2)
  237. assert.Nil(t, err)
  238. assert.False(t, result)
  239. }