couples_test.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. package leaves
  2. import (
  3. "bytes"
  4. "io/ioutil"
  5. "path"
  6. "strings"
  7. "testing"
  8. "github.com/gogo/protobuf/proto"
  9. "github.com/stretchr/testify/assert"
  10. "gopkg.in/src-d/go-git.v4/plumbing/object"
  11. gitplumbing "gopkg.in/src-d/go-git.v4/plumbing"
  12. "gopkg.in/src-d/hercules.v4/internal/core"
  13. "gopkg.in/src-d/hercules.v4/internal/pb"
  14. "gopkg.in/src-d/hercules.v4/internal/plumbing"
  15. "gopkg.in/src-d/hercules.v4/internal/plumbing/identity"
  16. "gopkg.in/src-d/hercules.v4/internal/test"
  17. )
  18. func fixtureCouples() *CouplesAnalysis {
  19. c := CouplesAnalysis{PeopleNumber: 3}
  20. c.Initialize(test.Repository)
  21. return &c
  22. }
  23. func TestCouplesMeta(t *testing.T) {
  24. c := fixtureCouples()
  25. assert.Equal(t, c.Name(), "Couples")
  26. assert.Equal(t, len(c.Provides()), 0)
  27. assert.Equal(t, len(c.Requires()), 2)
  28. assert.Equal(t, c.Requires()[0], identity.DependencyAuthor)
  29. assert.Equal(t, c.Requires()[1], plumbing.DependencyTreeChanges)
  30. assert.Equal(t, c.Flag(), "couples")
  31. assert.Len(t, c.ListConfigurationOptions(), 0)
  32. }
  33. func TestCouplesRegistration(t *testing.T) {
  34. summoned := core.Registry.Summon((&CouplesAnalysis{}).Name())
  35. assert.Len(t, summoned, 1)
  36. assert.Equal(t, summoned[0].Name(), "Couples")
  37. leaves := core.Registry.GetLeaves()
  38. matched := false
  39. for _, tp := range leaves {
  40. if tp.Flag() == (&CouplesAnalysis{}).Flag() {
  41. matched = true
  42. break
  43. }
  44. }
  45. assert.True(t, matched)
  46. }
  47. func generateChanges(names ...string) object.Changes {
  48. changes := make(object.Changes, 0, len(names))
  49. for _, name := range names {
  50. action := name[:1]
  51. name = name[1:]
  52. var change object.Change
  53. if action == "+" {
  54. change = object.Change{
  55. From: object.ChangeEntry{},
  56. To: object.ChangeEntry{Name: name},
  57. }
  58. } else if action == "-" {
  59. change = object.Change{
  60. From: object.ChangeEntry{Name: name},
  61. To: object.ChangeEntry{},
  62. }
  63. } else if action == "=" {
  64. change = object.Change{
  65. From: object.ChangeEntry{Name: name},
  66. To: object.ChangeEntry{Name: name},
  67. }
  68. } else {
  69. if action != ">" {
  70. panic("Invalid action.")
  71. }
  72. parts := strings.Split(name, ">")
  73. change = object.Change{
  74. From: object.ChangeEntry{Name: parts[0]},
  75. To: object.ChangeEntry{Name: parts[1]},
  76. }
  77. }
  78. changes = append(changes, &change)
  79. }
  80. return changes
  81. }
  82. func TestCouplesConsumeFinalize(t *testing.T) {
  83. c := fixtureCouples()
  84. deps := map[string]interface{}{}
  85. deps[identity.DependencyAuthor] = 0
  86. deps[core.DependencyCommit], _ = test.Repository.CommitObject(gitplumbing.NewHash(
  87. "a3ee37f91f0d705ec9c41ae88426f0ae44b2fbc3"))
  88. deps[plumbing.DependencyTreeChanges] = generateChanges("+LICENSE2", "+file2.go", "+rbtree2.go")
  89. c.Consume(deps)
  90. deps[plumbing.DependencyTreeChanges] = generateChanges("+README.md", "-LICENSE2", "=analyser.go", ">file2.go>file_test.go")
  91. c.Consume(deps)
  92. deps[identity.DependencyAuthor] = 1
  93. deps[plumbing.DependencyTreeChanges] = generateChanges("=README.md", "=analyser.go", "-rbtree2.go")
  94. c.Consume(deps)
  95. deps[identity.DependencyAuthor] = 2
  96. deps[plumbing.DependencyTreeChanges] = generateChanges("=file_test.go")
  97. c.Consume(deps)
  98. assert.Equal(t, len(c.people[0]), 6)
  99. assert.Equal(t, c.people[0]["README.md"], 1)
  100. assert.Equal(t, c.people[0]["LICENSE2"], 2)
  101. assert.Equal(t, c.people[0]["analyser.go"], 1)
  102. assert.Equal(t, c.people[0]["file2.go"], 1)
  103. assert.Equal(t, c.people[0]["file_test.go"], 1)
  104. assert.Equal(t, c.people[0]["rbtree2.go"], 1)
  105. assert.Equal(t, len(c.people[1]), 3)
  106. assert.Equal(t, c.people[1]["README.md"], 1)
  107. assert.Equal(t, c.people[1]["analyser.go"], 1)
  108. assert.Equal(t, c.people[1]["rbtree2.go"], 1)
  109. assert.Equal(t, len(c.people[2]), 1)
  110. assert.Equal(t, c.people[2]["file_test.go"], 1)
  111. assert.Equal(t, len(c.files["README.md"]), 3)
  112. assert.Equal(t, c.files["README.md"], map[string]int{
  113. "README.md": 2,
  114. "analyser.go": 2,
  115. "file_test.go": 1,
  116. })
  117. assert.Equal(t, c.files["LICENSE2"], map[string]int{
  118. "LICENSE2": 1,
  119. "file2.go": 1,
  120. "rbtree2.go": 1,
  121. })
  122. assert.Equal(t, c.files["file2.go"], map[string]int{
  123. "LICENSE2": 1,
  124. "file2.go": 1,
  125. "rbtree2.go": 1,
  126. })
  127. assert.Equal(t, c.files["rbtree2.go"], map[string]int{
  128. "LICENSE2": 1,
  129. "file2.go": 1,
  130. "rbtree2.go": 1,
  131. })
  132. assert.Equal(t, c.files["analyser.go"], map[string]int{
  133. "analyser.go": 2,
  134. "README.md": 2,
  135. "file_test.go": 1,
  136. })
  137. assert.Equal(t, c.files["file_test.go"], map[string]int{
  138. "file_test.go": 2,
  139. "README.md": 1,
  140. "analyser.go": 1,
  141. })
  142. assert.Equal(t, c.peopleCommits[0], 2)
  143. assert.Equal(t, c.peopleCommits[1], 1)
  144. assert.Equal(t, c.peopleCommits[2], 1)
  145. cr := c.Finalize().(CouplesResult)
  146. assert.Equal(t, len(cr.Files), 3)
  147. assert.Equal(t, cr.Files[0], "README.md")
  148. assert.Equal(t, cr.Files[1], "analyser.go")
  149. assert.Equal(t, cr.Files[2], "file_test.go")
  150. assert.Equal(t, len(cr.PeopleFiles[0]), 3)
  151. assert.Equal(t, cr.PeopleFiles[0][0], 0)
  152. assert.Equal(t, cr.PeopleFiles[0][1], 1)
  153. assert.Equal(t, cr.PeopleFiles[0][2], 2)
  154. assert.Equal(t, len(cr.PeopleFiles[1]), 2)
  155. assert.Equal(t, cr.PeopleFiles[1][0], 0)
  156. assert.Equal(t, cr.PeopleFiles[1][1], 1)
  157. assert.Equal(t, len(cr.PeopleFiles[2]), 1)
  158. assert.Equal(t, cr.PeopleFiles[2][0], 2)
  159. assert.Equal(t, len(cr.PeopleMatrix[0]), 3)
  160. assert.Equal(t, cr.PeopleMatrix[0][0], int64(7))
  161. assert.Equal(t, cr.PeopleMatrix[0][1], int64(3))
  162. assert.Equal(t, cr.PeopleMatrix[0][2], int64(1))
  163. assert.Equal(t, len(cr.PeopleMatrix[1]), 2)
  164. assert.Equal(t, cr.PeopleMatrix[1][0], int64(3))
  165. assert.Equal(t, cr.PeopleMatrix[1][1], int64(3))
  166. assert.Equal(t, len(cr.PeopleMatrix[2]), 2)
  167. assert.Equal(t, cr.PeopleMatrix[2][0], int64(1))
  168. assert.Equal(t, cr.PeopleMatrix[2][2], int64(1))
  169. assert.Equal(t, len(cr.FilesMatrix), 3)
  170. assert.Equal(t, len(cr.FilesMatrix[0]), 3)
  171. assert.Equal(t, cr.FilesMatrix[0][2], int64(1))
  172. assert.Equal(t, cr.FilesMatrix[0][0], int64(2))
  173. assert.Equal(t, cr.FilesMatrix[0][1], int64(2))
  174. assert.Equal(t, len(cr.FilesMatrix[1]), 3)
  175. assert.Equal(t, cr.FilesMatrix[1][2], int64(1))
  176. assert.Equal(t, cr.FilesMatrix[1][0], int64(2))
  177. assert.Equal(t, cr.FilesMatrix[1][1], int64(2))
  178. assert.Equal(t, len(cr.FilesMatrix[2]), 3)
  179. assert.Equal(t, cr.FilesMatrix[2][0], int64(1))
  180. assert.Equal(t, cr.FilesMatrix[2][1], int64(1))
  181. assert.Equal(t, cr.FilesMatrix[2][2], int64(3))
  182. }
  183. func TestCouplesFork(t *testing.T) {
  184. couples1 := fixtureCouples()
  185. clones := couples1.Fork(1)
  186. assert.Len(t, clones, 1)
  187. couples2 := clones[0].(*CouplesAnalysis)
  188. assert.True(t, couples1 != couples2)
  189. assert.Equal(t, *couples1, *couples2)
  190. couples1.Merge([]core.PipelineItem{couples2})
  191. }
  192. func TestCouplesSerialize(t *testing.T) {
  193. c := fixtureCouples()
  194. result := CouplesResult {
  195. PeopleMatrix: []map[int]int64{
  196. {0:7, 1:3, 2:1}, {0:3, 1:3}, {0:1, 2:1}, {},
  197. },
  198. PeopleFiles: [][]int{
  199. {0, 1, 2}, {1, 2}, {0}, {},
  200. },
  201. FilesMatrix: []map[int]int64{
  202. {1:1, 2:1, 0:3}, {1:2, 2:2, 0:1}, {2:2, 0:1, 1:2},
  203. },
  204. Files: []string{"five", "one", "three"},
  205. reversedPeopleDict: []string{"p1", "p2", "p3"},
  206. }
  207. buffer := &bytes.Buffer{}
  208. c.Serialize(result, false, buffer)
  209. assert.Equal(t, buffer.String(), ` files_coocc:
  210. index:
  211. - "five"
  212. - "one"
  213. - "three"
  214. matrix:
  215. - {0: 3, 1: 1, 2: 1}
  216. - {0: 1, 1: 2, 2: 2}
  217. - {0: 1, 1: 2, 2: 2}
  218. people_coocc:
  219. index:
  220. - "p1"
  221. - "p2"
  222. - "p3"
  223. matrix:
  224. - {0: 7, 1: 3, 2: 1}
  225. - {0: 3, 1: 3}
  226. - {0: 1, 2: 1}
  227. - {}
  228. author_files:
  229. - "p3":
  230. - "five"
  231. - "p2":
  232. - "one"
  233. - "three"
  234. - "p1":
  235. - "five"
  236. - "one"
  237. - "three"
  238. `)
  239. buffer = &bytes.Buffer{}
  240. c.Serialize(result, true, buffer)
  241. msg := pb.CouplesAnalysisResults{}
  242. proto.Unmarshal(buffer.Bytes(), &msg)
  243. assert.Len(t, msg.PeopleFiles, 3)
  244. tmp1 := [...]int32{0, 1, 2}
  245. assert.Equal(t, msg.PeopleFiles[0].Files, tmp1[:])
  246. tmp2 := [...]int32{1, 2}
  247. assert.Equal(t, msg.PeopleFiles[1].Files, tmp2[:])
  248. tmp3 := [...]int32{0}
  249. assert.Equal(t, msg.PeopleFiles[2].Files, tmp3[:])
  250. assert.Equal(t, msg.PeopleCouples.Index, result.reversedPeopleDict)
  251. assert.Equal(t, msg.PeopleCouples.Matrix.NumberOfRows, int32(4))
  252. assert.Equal(t, msg.PeopleCouples.Matrix.NumberOfColumns, int32(4))
  253. data := [...]int64{7, 3, 1, 3, 3, 1, 1}
  254. assert.Equal(t, msg.PeopleCouples.Matrix.Data, data[:])
  255. indices := [...]int32{0, 1, 2, 0, 1, 0, 2}
  256. assert.Equal(t, msg.PeopleCouples.Matrix.Indices, indices[:])
  257. indptr := [...]int64{0, 3, 5, 7, 7}
  258. assert.Equal(t, msg.PeopleCouples.Matrix.Indptr, indptr[:])
  259. files := [...]string{"five", "one", "three"}
  260. assert.Equal(t, msg.FileCouples.Index, files[:])
  261. assert.Equal(t, msg.FileCouples.Matrix.NumberOfRows, int32(3))
  262. assert.Equal(t, msg.FileCouples.Matrix.NumberOfColumns, int32(3))
  263. data2 := [...]int64{3, 1, 1, 1, 2, 2, 1, 2, 2}
  264. assert.Equal(t, msg.FileCouples.Matrix.Data, data2[:])
  265. indices2 := [...]int32{0, 1, 2, 0, 1, 2, 0, 1, 2}
  266. assert.Equal(t, msg.FileCouples.Matrix.Indices, indices2[:])
  267. indptr2 := [...]int64{0, 3, 6, 9}
  268. assert.Equal(t, msg.FileCouples.Matrix.Indptr, indptr2[:])
  269. }
  270. func TestCouplesDeserialize(t *testing.T) {
  271. allBuffer, err := ioutil.ReadFile(path.Join("..", "internal", "test_data", "couples.pb"))
  272. assert.Nil(t, err)
  273. message := pb.AnalysisResults{}
  274. err = proto.Unmarshal(allBuffer, &message)
  275. assert.Nil(t, err)
  276. couples := CouplesAnalysis{}
  277. iresult, err := couples.Deserialize(message.Contents[couples.Name()])
  278. assert.Nil(t, err)
  279. result := iresult.(CouplesResult)
  280. assert.Len(t, result.reversedPeopleDict, 2)
  281. assert.Len(t, result.PeopleFiles, 2)
  282. assert.Len(t, result.PeopleMatrix, 3)
  283. assert.Len(t, result.Files, 74)
  284. assert.Len(t, result.FilesMatrix, 74)
  285. }
  286. func TestCouplesMerge(t *testing.T) {
  287. r1, r2 := CouplesResult{}, CouplesResult{}
  288. people1 := [...]string{"one", "two"}
  289. people2 := [...]string{"two", "three"}
  290. r1.reversedPeopleDict = people1[:]
  291. r2.reversedPeopleDict = people2[:]
  292. r1.Files = people1[:]
  293. r2.Files = people2[:]
  294. r1.PeopleFiles = make([][]int, 2)
  295. r1.PeopleFiles[0] = make([]int, 2)
  296. r1.PeopleFiles[0][0] = 0
  297. r1.PeopleFiles[0][1] = 1
  298. r1.PeopleFiles[1] = make([]int, 1)
  299. r1.PeopleFiles[1][0] = 0
  300. r2.PeopleFiles = make([][]int, 2)
  301. r2.PeopleFiles[0] = make([]int, 1)
  302. r2.PeopleFiles[0][0] = 1
  303. r2.PeopleFiles[1] = make([]int, 2)
  304. r2.PeopleFiles[1][0] = 0
  305. r2.PeopleFiles[1][1] = 1
  306. r1.FilesMatrix = make([]map[int]int64, 2)
  307. r1.FilesMatrix[0] = map[int]int64{}
  308. r1.FilesMatrix[1] = map[int]int64{}
  309. r1.FilesMatrix[0][1] = 100
  310. r1.FilesMatrix[1][0] = 100
  311. r2.FilesMatrix = make([]map[int]int64, 2)
  312. r2.FilesMatrix[0] = map[int]int64{}
  313. r2.FilesMatrix[1] = map[int]int64{}
  314. r2.FilesMatrix[0][1] = 200
  315. r2.FilesMatrix[1][0] = 200
  316. r1.PeopleMatrix = make([]map[int]int64, 3)
  317. r1.PeopleMatrix[0] = map[int]int64{}
  318. r1.PeopleMatrix[1] = map[int]int64{}
  319. r1.PeopleMatrix[2] = map[int]int64{}
  320. r1.PeopleMatrix[0][1] = 100
  321. r1.PeopleMatrix[1][0] = 100
  322. r1.PeopleMatrix[2][0] = 300
  323. r1.PeopleMatrix[2][1] = 400
  324. r2.PeopleMatrix = make([]map[int]int64, 3)
  325. r2.PeopleMatrix[0] = map[int]int64{}
  326. r2.PeopleMatrix[1] = map[int]int64{}
  327. r2.PeopleMatrix[2] = map[int]int64{}
  328. r2.PeopleMatrix[0][1] = 10
  329. r2.PeopleMatrix[1][0] = 10
  330. r2.PeopleMatrix[2][0] = 30
  331. r2.PeopleMatrix[2][1] = 40
  332. couples := CouplesAnalysis{}
  333. merged := couples.MergeResults(r1, r2, nil, nil).(CouplesResult)
  334. mergedPeople := [...]string{"one", "two", "three"}
  335. assert.Equal(t, merged.reversedPeopleDict, mergedPeople[:])
  336. assert.Equal(t, merged.Files, mergedPeople[:])
  337. assert.Len(t, merged.PeopleFiles, 3)
  338. assert.Equal(t, merged.PeopleFiles[0], getSlice(0, 1))
  339. assert.Equal(t, merged.PeopleFiles[1], getSlice(0, 2))
  340. assert.Equal(t, merged.PeopleFiles[2], getSlice(1, 2))
  341. assert.Len(t, merged.PeopleMatrix, 4)
  342. assert.Equal(t, merged.PeopleMatrix[0], getCouplesMap(1, 100))
  343. assert.Equal(t, merged.PeopleMatrix[1], getCouplesMap(0, 100, 2, 10))
  344. assert.Equal(t, merged.PeopleMatrix[2], getCouplesMap(1, 10))
  345. assert.Equal(t, merged.PeopleMatrix[3], getCouplesMap(0, 300, 1, 430, 2, 40))
  346. assert.Len(t, merged.FilesMatrix, 3)
  347. assert.Equal(t, merged.FilesMatrix[0], getCouplesMap(1, 100))
  348. assert.Equal(t, merged.FilesMatrix[1], getCouplesMap(0, 100, 2, 200))
  349. assert.Equal(t, merged.FilesMatrix[2], getCouplesMap(1, 200))
  350. }
  351. func TestCouplesCurrentFiles(t *testing.T) {
  352. c := fixtureCouples()
  353. c.lastCommit, _ = test.Repository.CommitObject(gitplumbing.NewHash(
  354. "cce947b98a050c6d356bc6ba95030254914027b1"))
  355. files := c.currentFiles()
  356. assert.Equal(t, files, map[string]bool{".gitignore": true, "LICENSE": true})
  357. }
  358. func TestCouplesPropagateRenames(t *testing.T) {
  359. c := fixtureCouples()
  360. c.files["one"] = map[string]int{
  361. "one": 1,
  362. "two": 2,
  363. "three": 3,
  364. }
  365. c.files["two"] = map[string]int{
  366. "one": 2,
  367. "two": 10,
  368. "three": 1,
  369. "four": 7,
  370. }
  371. c.files["three"] = map[string]int{
  372. "one": 3,
  373. "two": 1,
  374. "three": 3,
  375. "four": 2,
  376. }
  377. c.files["four"] = map[string]int{
  378. "two": 7,
  379. "three": 3,
  380. "four": 1,
  381. }
  382. c.PeopleNumber = 1
  383. c.people = make([]map[string]int, 1)
  384. c.people[0] = map[string]int{}
  385. c.people[0]["one"] = 1
  386. c.people[0]["two"] = 2
  387. c.people[0]["three"] = 3
  388. c.people[0]["four"] = 4
  389. *c.renames = []rename{{ToName: "four", FromName: "one"}}
  390. files, people := c.propagateRenames(map[string]bool{"two": true, "three": true, "four": true})
  391. assert.Len(t, files, 3)
  392. assert.Len(t, people, 1)
  393. assert.Equal(t, files["two"], map[string]int{"two": 10, "three": 1, "four": 9})
  394. assert.Equal(t, files["three"], map[string]int{"two": 1, "three": 3, "four": 6})
  395. assert.Equal(t, files["four"], map[string]int{"two": 9, "three": 6, "four": 2})
  396. assert.Equal(t, people[0], map[string]int{"two": 2, "three": 3, "four": 5})
  397. }
  398. func getSlice(vals ...int) []int {
  399. return vals
  400. }
  401. func getCouplesMap(vals ...int) map[int]int64 {
  402. res := map[int]int64{}
  403. for i := 0; i < len(vals); i += 2 {
  404. res[vals[i]] = int64(vals[i+1])
  405. }
  406. return res
  407. }