couples_test.go 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598
  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. gitplumbing "gopkg.in/src-d/go-git.v4/plumbing"
  11. "gopkg.in/src-d/go-git.v4/plumbing/object"
  12. "gopkg.in/src-d/hercules.v10/internal/core"
  13. "gopkg.in/src-d/hercules.v10/internal/pb"
  14. "gopkg.in/src-d/hercules.v10/internal/plumbing"
  15. "gopkg.in/src-d/hercules.v10/internal/plumbing/identity"
  16. "gopkg.in/src-d/hercules.v10/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. logger := core.NewLogger()
  33. assert.NoError(t, c.Configure(map[string]interface{}{
  34. core.ConfigLogger: logger,
  35. }))
  36. assert.Equal(t, logger, c.l)
  37. }
  38. func TestCouplesRegistration(t *testing.T) {
  39. summoned := core.Registry.Summon((&CouplesAnalysis{}).Name())
  40. assert.Len(t, summoned, 1)
  41. assert.Equal(t, summoned[0].Name(), "Couples")
  42. leaves := core.Registry.GetLeaves()
  43. matched := false
  44. for _, tp := range leaves {
  45. if tp.Flag() == (&CouplesAnalysis{}).Flag() {
  46. matched = true
  47. break
  48. }
  49. }
  50. assert.True(t, matched)
  51. }
  52. func generateChanges(names ...string) object.Changes {
  53. changes := make(object.Changes, 0, len(names))
  54. for _, name := range names {
  55. action := name[:1]
  56. name = name[1:]
  57. var change object.Change
  58. if action == "+" {
  59. change = object.Change{
  60. From: object.ChangeEntry{},
  61. To: object.ChangeEntry{Name: name},
  62. }
  63. } else if action == "-" {
  64. change = object.Change{
  65. From: object.ChangeEntry{Name: name},
  66. To: object.ChangeEntry{},
  67. }
  68. } else if action == "=" {
  69. change = object.Change{
  70. From: object.ChangeEntry{Name: name},
  71. To: object.ChangeEntry{Name: name},
  72. }
  73. } else {
  74. if action != ">" {
  75. panic("Invalid action.")
  76. }
  77. parts := strings.Split(name, ">")
  78. change = object.Change{
  79. From: object.ChangeEntry{Name: parts[0]},
  80. To: object.ChangeEntry{Name: parts[1]},
  81. }
  82. }
  83. changes = append(changes, &change)
  84. }
  85. return changes
  86. }
  87. func TestCouplesConsumeFinalize(t *testing.T) {
  88. c := fixtureCouples()
  89. deps := map[string]interface{}{}
  90. deps[identity.DependencyAuthor] = 0
  91. deps[core.DependencyCommit], _ = test.Repository.CommitObject(gitplumbing.NewHash(
  92. "a3ee37f91f0d705ec9c41ae88426f0ae44b2fbc3"))
  93. deps[core.DependencyIsMerge] = false
  94. deps[plumbing.DependencyTreeChanges] = generateChanges("+LICENSE2", "+file2.go", "+rbtree2.go")
  95. c.Consume(deps)
  96. deps[plumbing.DependencyTreeChanges] = generateChanges("+README.md", "-LICENSE2", "=analyser.go", ">file2.go>file_test.go")
  97. c.Consume(deps)
  98. deps[identity.DependencyAuthor] = 1
  99. deps[plumbing.DependencyTreeChanges] = generateChanges("=README.md", "=analyser.go", "-rbtree2.go")
  100. c.Consume(deps)
  101. deps[identity.DependencyAuthor] = 2
  102. deps[plumbing.DependencyTreeChanges] = generateChanges("=file_test.go")
  103. c.Consume(deps)
  104. assert.Equal(t, len(c.people[0]), 6)
  105. assert.Equal(t, c.people[0]["README.md"], 1)
  106. assert.Equal(t, c.people[0]["LICENSE2"], 2)
  107. assert.Equal(t, c.people[0]["analyser.go"], 1)
  108. assert.Equal(t, c.people[0]["file2.go"], 1)
  109. assert.Equal(t, c.people[0]["file_test.go"], 1)
  110. assert.Equal(t, c.people[0]["rbtree2.go"], 1)
  111. assert.Equal(t, len(c.people[1]), 3)
  112. assert.Equal(t, c.people[1]["README.md"], 1)
  113. assert.Equal(t, c.people[1]["analyser.go"], 1)
  114. assert.Equal(t, c.people[1]["rbtree2.go"], 1)
  115. assert.Equal(t, len(c.people[2]), 1)
  116. assert.Equal(t, c.people[2]["file_test.go"], 1)
  117. assert.Equal(t, len(c.files["README.md"]), 3)
  118. assert.Equal(t, c.files["README.md"], map[string]int{
  119. "README.md": 2,
  120. "analyser.go": 2,
  121. "file_test.go": 1,
  122. })
  123. assert.Equal(t, c.files["LICENSE2"], map[string]int{
  124. "LICENSE2": 1,
  125. "file2.go": 1,
  126. "rbtree2.go": 1,
  127. })
  128. assert.Equal(t, c.files["file2.go"], map[string]int{
  129. "LICENSE2": 1,
  130. "file2.go": 1,
  131. "rbtree2.go": 1,
  132. })
  133. assert.Equal(t, c.files["rbtree2.go"], map[string]int{
  134. "LICENSE2": 1,
  135. "file2.go": 1,
  136. "rbtree2.go": 1,
  137. })
  138. assert.Equal(t, c.files["analyser.go"], map[string]int{
  139. "analyser.go": 2,
  140. "README.md": 2,
  141. "file_test.go": 1,
  142. })
  143. assert.Equal(t, c.files["file_test.go"], map[string]int{
  144. "file_test.go": 2,
  145. "README.md": 1,
  146. "analyser.go": 1,
  147. })
  148. assert.Equal(t, c.peopleCommits[0], 2)
  149. assert.Equal(t, c.peopleCommits[1], 1)
  150. assert.Equal(t, c.peopleCommits[2], 1)
  151. cr := c.Finalize().(CouplesResult)
  152. assert.Equal(t, len(cr.Files), 3)
  153. assert.Equal(t, cr.Files[0], "README.md")
  154. assert.Equal(t, cr.Files[1], "analyser.go")
  155. assert.Equal(t, cr.Files[2], "file_test.go")
  156. assert.Equal(t, len(cr.FilesLines), 3)
  157. assert.Equal(t, cr.FilesLines[0], 15)
  158. assert.Equal(t, cr.FilesLines[1], 252)
  159. assert.Equal(t, cr.FilesLines[2], 238)
  160. assert.Equal(t, len(cr.PeopleFiles[0]), 3)
  161. assert.Equal(t, cr.PeopleFiles[0][0], 0)
  162. assert.Equal(t, cr.PeopleFiles[0][1], 1)
  163. assert.Equal(t, cr.PeopleFiles[0][2], 2)
  164. assert.Equal(t, len(cr.PeopleFiles[1]), 2)
  165. assert.Equal(t, cr.PeopleFiles[1][0], 0)
  166. assert.Equal(t, cr.PeopleFiles[1][1], 1)
  167. assert.Equal(t, len(cr.PeopleFiles[2]), 1)
  168. assert.Equal(t, cr.PeopleFiles[2][0], 2)
  169. assert.Equal(t, len(cr.PeopleMatrix[0]), 3)
  170. assert.Equal(t, cr.PeopleMatrix[0][0], int64(7))
  171. assert.Equal(t, cr.PeopleMatrix[0][1], int64(3))
  172. assert.Equal(t, cr.PeopleMatrix[0][2], int64(1))
  173. assert.Equal(t, len(cr.PeopleMatrix[1]), 2)
  174. assert.Equal(t, cr.PeopleMatrix[1][0], int64(3))
  175. assert.Equal(t, cr.PeopleMatrix[1][1], int64(3))
  176. assert.Equal(t, len(cr.PeopleMatrix[2]), 2)
  177. assert.Equal(t, cr.PeopleMatrix[2][0], int64(1))
  178. assert.Equal(t, cr.PeopleMatrix[2][2], int64(1))
  179. assert.Equal(t, len(cr.FilesMatrix), 3)
  180. assert.Equal(t, len(cr.FilesMatrix[0]), 3)
  181. assert.Equal(t, cr.FilesMatrix[0][2], int64(1))
  182. assert.Equal(t, cr.FilesMatrix[0][0], int64(2))
  183. assert.Equal(t, cr.FilesMatrix[0][1], int64(2))
  184. assert.Equal(t, len(cr.FilesMatrix[1]), 3)
  185. assert.Equal(t, cr.FilesMatrix[1][2], int64(1))
  186. assert.Equal(t, cr.FilesMatrix[1][0], int64(2))
  187. assert.Equal(t, cr.FilesMatrix[1][1], int64(2))
  188. assert.Equal(t, len(cr.FilesMatrix[2]), 3)
  189. assert.Equal(t, cr.FilesMatrix[2][0], int64(1))
  190. assert.Equal(t, cr.FilesMatrix[2][1], int64(1))
  191. assert.Equal(t, cr.FilesMatrix[2][2], int64(3))
  192. }
  193. func TestCouplesConsumeFinalizeMerge(t *testing.T) {
  194. c := fixtureCouples()
  195. deps := map[string]interface{}{}
  196. deps[identity.DependencyAuthor] = 0
  197. deps[core.DependencyCommit], _ = test.Repository.CommitObject(gitplumbing.NewHash(
  198. "a3ee37f91f0d705ec9c41ae88426f0ae44b2fbc3"))
  199. deps[core.DependencyIsMerge] = true
  200. deps[plumbing.DependencyTreeChanges] = generateChanges("+LICENSE2", "+file2.go")
  201. c.Consume(deps)
  202. deps[plumbing.DependencyTreeChanges] = generateChanges("+file2.go", "-LICENSE2", ">file2.go>file_test.go")
  203. c.Consume(deps)
  204. assert.Equal(t, len(c.people[0]), 3)
  205. assert.Equal(t, c.people[0]["LICENSE2"], 1)
  206. assert.Equal(t, c.people[0]["file2.go"], 1)
  207. assert.Equal(t, c.people[0]["file_test.go"], 1)
  208. for i := 1; i < 3; i++ {
  209. assert.Equal(t, len(c.people[i]), 0)
  210. }
  211. assert.Equal(t, c.files["LICENSE2"], map[string]int{
  212. "LICENSE2": 1,
  213. "file2.go": 1,
  214. })
  215. assert.Equal(t, c.files["file2.go"], map[string]int{
  216. "file2.go": 1,
  217. "LICENSE2": 1,
  218. })
  219. assert.Equal(t, c.files["file_test.go"], map[string]int{
  220. "file_test.go": 1,
  221. })
  222. }
  223. func TestCouplesConsumeFinalizeAuthorMissing(t *testing.T) {
  224. c := fixtureCouples()
  225. deps := map[string]interface{}{}
  226. deps[identity.DependencyAuthor] = 0
  227. deps[core.DependencyCommit], _ = test.Repository.CommitObject(gitplumbing.NewHash(
  228. "a3ee37f91f0d705ec9c41ae88426f0ae44b2fbc3"))
  229. deps[core.DependencyIsMerge] = false
  230. deps[plumbing.DependencyTreeChanges] = generateChanges("+LICENSE2", "+file2.go", "+rbtree2.go")
  231. c.Consume(deps)
  232. deps[plumbing.DependencyTreeChanges] = generateChanges("+README.md", "-LICENSE2", "=analyser.go", ">file2.go>file_test.go")
  233. c.Consume(deps)
  234. deps[identity.DependencyAuthor] = 1
  235. deps[plumbing.DependencyTreeChanges] = generateChanges("=README.md", "=analyser.go", "-rbtree2.go")
  236. c.Consume(deps)
  237. deps[identity.DependencyAuthor] = identity.AuthorMissing
  238. deps[plumbing.DependencyTreeChanges] = generateChanges("=file_test.go")
  239. c.Consume(deps)
  240. assert.Equal(t, len(c.people[0]), 6)
  241. assert.Equal(t, c.people[0]["README.md"], 1)
  242. assert.Equal(t, c.people[0]["LICENSE2"], 2)
  243. assert.Equal(t, c.people[0]["analyser.go"], 1)
  244. assert.Equal(t, c.people[0]["file2.go"], 1)
  245. assert.Equal(t, c.people[0]["file_test.go"], 1)
  246. assert.Equal(t, c.people[0]["rbtree2.go"], 1)
  247. assert.Equal(t, len(c.people[1]), 3)
  248. assert.Equal(t, c.people[1]["README.md"], 1)
  249. assert.Equal(t, c.people[1]["analyser.go"], 1)
  250. assert.Equal(t, c.people[1]["rbtree2.go"], 1)
  251. assert.Equal(t, len(c.people[2]), 0)
  252. assert.Equal(t, len(c.files["README.md"]), 3)
  253. assert.Equal(t, c.files["README.md"], map[string]int{
  254. "README.md": 2,
  255. "analyser.go": 2,
  256. "file_test.go": 1,
  257. })
  258. assert.Equal(t, c.files["LICENSE2"], map[string]int{
  259. "LICENSE2": 1,
  260. "file2.go": 1,
  261. "rbtree2.go": 1,
  262. })
  263. assert.Equal(t, c.files["file2.go"], map[string]int{
  264. "LICENSE2": 1,
  265. "file2.go": 1,
  266. "rbtree2.go": 1,
  267. })
  268. assert.Equal(t, c.files["rbtree2.go"], map[string]int{
  269. "LICENSE2": 1,
  270. "file2.go": 1,
  271. "rbtree2.go": 1,
  272. })
  273. assert.Equal(t, c.files["analyser.go"], map[string]int{
  274. "analyser.go": 2,
  275. "README.md": 2,
  276. "file_test.go": 1,
  277. })
  278. assert.Equal(t, c.files["file_test.go"], map[string]int{
  279. "file_test.go": 2,
  280. "README.md": 1,
  281. "analyser.go": 1,
  282. })
  283. assert.Equal(t, c.peopleCommits[0], 2)
  284. assert.Equal(t, c.peopleCommits[1], 1)
  285. assert.Equal(t, c.peopleCommits[2], 0)
  286. cr := c.Finalize().(CouplesResult)
  287. assert.Equal(t, len(cr.Files), 3)
  288. assert.Equal(t, cr.Files[0], "README.md")
  289. assert.Equal(t, cr.Files[1], "analyser.go")
  290. assert.Equal(t, cr.Files[2], "file_test.go")
  291. assert.Equal(t, len(cr.FilesLines), 3)
  292. assert.Equal(t, cr.FilesLines[0], 15)
  293. assert.Equal(t, cr.FilesLines[1], 252)
  294. assert.Equal(t, cr.FilesLines[2], 238)
  295. assert.Equal(t, len(cr.PeopleFiles[0]), 3)
  296. assert.Equal(t, cr.PeopleFiles[0][0], 0)
  297. assert.Equal(t, cr.PeopleFiles[0][1], 1)
  298. assert.Equal(t, cr.PeopleFiles[0][2], 2)
  299. assert.Equal(t, len(cr.PeopleFiles[1]), 2)
  300. assert.Equal(t, cr.PeopleFiles[1][0], 0)
  301. assert.Equal(t, cr.PeopleFiles[1][1], 1)
  302. assert.Equal(t, len(cr.PeopleFiles[2]), 0)
  303. assert.Equal(t, len(cr.PeopleMatrix[0]), 3)
  304. assert.Equal(t, cr.PeopleMatrix[0][0], int64(7))
  305. assert.Equal(t, cr.PeopleMatrix[0][1], int64(3))
  306. assert.Equal(t, cr.PeopleMatrix[0][2], int64(0))
  307. assert.Equal(t, len(cr.PeopleMatrix[1]), 2)
  308. assert.Equal(t, cr.PeopleMatrix[1][0], int64(3))
  309. assert.Equal(t, cr.PeopleMatrix[1][1], int64(3))
  310. assert.Equal(t, len(cr.PeopleMatrix[2]), 0)
  311. assert.Equal(t, len(cr.FilesMatrix), 3)
  312. assert.Equal(t, len(cr.FilesMatrix[0]), 3)
  313. assert.Equal(t, cr.FilesMatrix[0][2], int64(1))
  314. assert.Equal(t, cr.FilesMatrix[0][0], int64(2))
  315. assert.Equal(t, cr.FilesMatrix[0][1], int64(2))
  316. assert.Equal(t, len(cr.FilesMatrix[1]), 3)
  317. assert.Equal(t, cr.FilesMatrix[1][2], int64(1))
  318. assert.Equal(t, cr.FilesMatrix[1][0], int64(2))
  319. assert.Equal(t, cr.FilesMatrix[1][1], int64(2))
  320. assert.Equal(t, len(cr.FilesMatrix[2]), 3)
  321. assert.Equal(t, cr.FilesMatrix[2][0], int64(1))
  322. assert.Equal(t, cr.FilesMatrix[2][1], int64(1))
  323. assert.Equal(t, cr.FilesMatrix[2][2], int64(3))
  324. }
  325. func TestCouplesConsumeManyFiles(t *testing.T) {
  326. c := fixtureCouples()
  327. deps := map[string]interface{}{}
  328. deps[identity.DependencyAuthor] = 0
  329. deps[core.DependencyCommit], _ = test.Repository.CommitObject(gitplumbing.NewHash(
  330. "a3ee37f91f0d705ec9c41ae88426f0ae44b2fbc3"))
  331. deps[core.DependencyIsMerge] = false
  332. changes := make(object.Changes, CouplesMaximumMeaningfulContextSize+1)
  333. for i := 0; i < len(changes); i++ {
  334. changes[i] = &object.Change{
  335. From: object.ChangeEntry{},
  336. To: object.ChangeEntry{Name: string(i)},
  337. }
  338. }
  339. deps[plumbing.DependencyTreeChanges] = changes
  340. _, err := c.Consume(deps)
  341. assert.Nil(t, err)
  342. assert.Equal(t, len(c.people[0]), len(changes))
  343. assert.Len(t, c.files, 0)
  344. }
  345. func TestCouplesFork(t *testing.T) {
  346. couples1 := fixtureCouples()
  347. clones := couples1.Fork(1)
  348. assert.Len(t, clones, 1)
  349. couples2 := clones[0].(*CouplesAnalysis)
  350. assert.True(t, couples1 != couples2)
  351. assert.Equal(t, *couples1, *couples2)
  352. couples1.Merge([]core.PipelineItem{couples2})
  353. }
  354. func TestCouplesSerialize(t *testing.T) {
  355. c := fixtureCouples()
  356. result := CouplesResult{
  357. PeopleMatrix: []map[int]int64{
  358. {0: 7, 1: 3, 2: 1}, {0: 3, 1: 3}, {0: 1, 2: 1}, {},
  359. },
  360. PeopleFiles: [][]int{
  361. {0, 1, 2}, {1, 2}, {0}, {},
  362. },
  363. FilesMatrix: []map[int]int64{
  364. {1: 1, 2: 1, 0: 3}, {1: 2, 2: 2, 0: 1}, {2: 2, 0: 1, 1: 2},
  365. },
  366. Files: []string{"five", "one", "three"},
  367. FilesLines: []int{9, 8, 7},
  368. reversedPeopleDict: []string{"p1", "p2", "p3"},
  369. }
  370. buffer := &bytes.Buffer{}
  371. assert.Nil(t, c.Serialize(result, false, buffer))
  372. assert.Equal(t, buffer.String(), ` files_coocc:
  373. index:
  374. - "five"
  375. - "one"
  376. - "three"
  377. lines:
  378. - 9
  379. - 8
  380. - 7
  381. matrix:
  382. - {0: 3, 1: 1, 2: 1}
  383. - {0: 1, 1: 2, 2: 2}
  384. - {0: 1, 1: 2, 2: 2}
  385. people_coocc:
  386. index:
  387. - "p1"
  388. - "p2"
  389. - "p3"
  390. matrix:
  391. - {0: 7, 1: 3, 2: 1}
  392. - {0: 3, 1: 3}
  393. - {0: 1, 2: 1}
  394. - {}
  395. author_files:
  396. - "p3":
  397. - "five"
  398. - "p2":
  399. - "one"
  400. - "three"
  401. - "p1":
  402. - "five"
  403. - "one"
  404. - "three"
  405. `)
  406. buffer = &bytes.Buffer{}
  407. assert.Nil(t, c.Serialize(result, true, buffer))
  408. msg := pb.CouplesAnalysisResults{}
  409. assert.Nil(t, proto.Unmarshal(buffer.Bytes(), &msg))
  410. assert.Equal(t, msg.FilesLines, []int32{9, 8, 7})
  411. assert.Len(t, msg.PeopleFiles, 3)
  412. tmp1 := [...]int32{0, 1, 2}
  413. assert.Equal(t, msg.PeopleFiles[0].Files, tmp1[:])
  414. tmp2 := [...]int32{1, 2}
  415. assert.Equal(t, msg.PeopleFiles[1].Files, tmp2[:])
  416. tmp3 := [...]int32{0}
  417. assert.Equal(t, msg.PeopleFiles[2].Files, tmp3[:])
  418. assert.Equal(t, msg.PeopleCouples.Index, result.reversedPeopleDict)
  419. assert.Equal(t, msg.PeopleCouples.Matrix.NumberOfRows, int32(4))
  420. assert.Equal(t, msg.PeopleCouples.Matrix.NumberOfColumns, int32(4))
  421. data := [...]int64{7, 3, 1, 3, 3, 1, 1}
  422. assert.Equal(t, msg.PeopleCouples.Matrix.Data, data[:])
  423. indices := [...]int32{0, 1, 2, 0, 1, 0, 2}
  424. assert.Equal(t, msg.PeopleCouples.Matrix.Indices, indices[:])
  425. indptr := [...]int64{0, 3, 5, 7, 7}
  426. assert.Equal(t, msg.PeopleCouples.Matrix.Indptr, indptr[:])
  427. files := [...]string{"five", "one", "three"}
  428. assert.Equal(t, msg.FileCouples.Index, files[:])
  429. assert.Equal(t, msg.FileCouples.Matrix.NumberOfRows, int32(3))
  430. assert.Equal(t, msg.FileCouples.Matrix.NumberOfColumns, int32(3))
  431. data2 := [...]int64{3, 1, 1, 1, 2, 2, 1, 2, 2}
  432. assert.Equal(t, msg.FileCouples.Matrix.Data, data2[:])
  433. indices2 := [...]int32{0, 1, 2, 0, 1, 2, 0, 1, 2}
  434. assert.Equal(t, msg.FileCouples.Matrix.Indices, indices2[:])
  435. indptr2 := [...]int64{0, 3, 6, 9}
  436. assert.Equal(t, msg.FileCouples.Matrix.Indptr, indptr2[:])
  437. }
  438. func TestCouplesDeserialize(t *testing.T) {
  439. message, err := ioutil.ReadFile(path.Join("..", "internal", "test_data", "couples.pb"))
  440. assert.Nil(t, err)
  441. couples := CouplesAnalysis{}
  442. iresult, err := couples.Deserialize(message)
  443. assert.Nil(t, err)
  444. result := iresult.(CouplesResult)
  445. assert.Len(t, result.reversedPeopleDict, 2)
  446. assert.Len(t, result.PeopleFiles, 2)
  447. assert.Len(t, result.PeopleMatrix, 3)
  448. assert.Len(t, result.Files, 74)
  449. assert.Len(t, result.FilesLines, 74)
  450. for _, v := range result.FilesLines {
  451. assert.True(t, v > 0)
  452. }
  453. assert.Len(t, result.FilesMatrix, 74)
  454. }
  455. func TestCouplesMerge(t *testing.T) {
  456. r1, r2 := CouplesResult{}, CouplesResult{}
  457. people1 := [...]string{"one", "two"}
  458. people2 := [...]string{"two", "three"}
  459. r1.reversedPeopleDict = people1[:]
  460. r2.reversedPeopleDict = people2[:]
  461. r1.Files = people1[:]
  462. r2.Files = people2[:]
  463. r1.FilesLines = []int{1, 2}
  464. r2.FilesLines = []int{2, 3}
  465. r1.PeopleFiles = make([][]int, 2)
  466. r1.PeopleFiles[0] = make([]int, 2)
  467. r1.PeopleFiles[0][0] = 0
  468. r1.PeopleFiles[0][1] = 1
  469. r1.PeopleFiles[1] = make([]int, 1)
  470. r1.PeopleFiles[1][0] = 0
  471. r2.PeopleFiles = make([][]int, 2)
  472. r2.PeopleFiles[0] = make([]int, 1)
  473. r2.PeopleFiles[0][0] = 1
  474. r2.PeopleFiles[1] = make([]int, 2)
  475. r2.PeopleFiles[1][0] = 0
  476. r2.PeopleFiles[1][1] = 1
  477. r1.FilesMatrix = make([]map[int]int64, 2)
  478. r1.FilesMatrix[0] = map[int]int64{}
  479. r1.FilesMatrix[1] = map[int]int64{}
  480. r1.FilesMatrix[0][1] = 100
  481. r1.FilesMatrix[1][0] = 100
  482. r2.FilesMatrix = make([]map[int]int64, 2)
  483. r2.FilesMatrix[0] = map[int]int64{}
  484. r2.FilesMatrix[1] = map[int]int64{}
  485. r2.FilesMatrix[0][1] = 200
  486. r2.FilesMatrix[1][0] = 200
  487. r1.PeopleMatrix = make([]map[int]int64, 3)
  488. r1.PeopleMatrix[0] = map[int]int64{}
  489. r1.PeopleMatrix[1] = map[int]int64{}
  490. r1.PeopleMatrix[2] = map[int]int64{}
  491. r1.PeopleMatrix[0][1] = 100
  492. r1.PeopleMatrix[1][0] = 100
  493. r1.PeopleMatrix[2][0] = 300
  494. r1.PeopleMatrix[2][1] = 400
  495. r2.PeopleMatrix = make([]map[int]int64, 3)
  496. r2.PeopleMatrix[0] = map[int]int64{}
  497. r2.PeopleMatrix[1] = map[int]int64{}
  498. r2.PeopleMatrix[2] = map[int]int64{}
  499. r2.PeopleMatrix[0][1] = 10
  500. r2.PeopleMatrix[1][0] = 10
  501. r2.PeopleMatrix[2][0] = 30
  502. r2.PeopleMatrix[2][1] = 40
  503. couples := CouplesAnalysis{}
  504. merged := couples.MergeResults(r1, r2, nil, nil).(CouplesResult)
  505. mergedPeople := [...]string{"one", "two", "three"}
  506. assert.Equal(t, merged.reversedPeopleDict, mergedPeople[:])
  507. assert.Equal(t, merged.Files, mergedPeople[:])
  508. assert.Equal(t, merged.FilesLines, []int{1, 4, 3})
  509. assert.Len(t, merged.PeopleFiles, 3)
  510. assert.Equal(t, merged.PeopleFiles[0], getSlice(0, 1))
  511. assert.Equal(t, merged.PeopleFiles[1], getSlice(0, 2))
  512. assert.Equal(t, merged.PeopleFiles[2], getSlice(1, 2))
  513. assert.Len(t, merged.PeopleMatrix, 4)
  514. assert.Equal(t, merged.PeopleMatrix[0], getCouplesMap(1, 100))
  515. assert.Equal(t, merged.PeopleMatrix[1], getCouplesMap(0, 100, 2, 10))
  516. assert.Equal(t, merged.PeopleMatrix[2], getCouplesMap(1, 10))
  517. assert.Equal(t, merged.PeopleMatrix[3], getCouplesMap(0, 300, 1, 430, 2, 40))
  518. assert.Len(t, merged.FilesMatrix, 3)
  519. assert.Equal(t, merged.FilesMatrix[0], getCouplesMap(1, 100))
  520. assert.Equal(t, merged.FilesMatrix[1], getCouplesMap(0, 100, 2, 200))
  521. assert.Equal(t, merged.FilesMatrix[2], getCouplesMap(1, 200))
  522. }
  523. func TestCouplesCurrentFiles(t *testing.T) {
  524. c := fixtureCouples()
  525. c.lastCommit, _ = test.Repository.CommitObject(gitplumbing.NewHash(
  526. "cce947b98a050c6d356bc6ba95030254914027b1"))
  527. files := c.currentFiles()
  528. assert.Equal(t, files, map[string]bool{".gitignore": true, "LICENSE": true})
  529. }
  530. func TestCouplesPropagateRenames(t *testing.T) {
  531. c := fixtureCouples()
  532. c.files["one"] = map[string]int{
  533. "one": 1,
  534. "two": 2,
  535. "three": 3,
  536. }
  537. c.files["two"] = map[string]int{
  538. "one": 2,
  539. "two": 10,
  540. "three": 1,
  541. "four": 7,
  542. }
  543. c.files["three"] = map[string]int{
  544. "one": 3,
  545. "two": 1,
  546. "three": 3,
  547. "four": 2,
  548. }
  549. c.files["four"] = map[string]int{
  550. "two": 7,
  551. "three": 3,
  552. "four": 1,
  553. }
  554. c.PeopleNumber = 1
  555. c.people = make([]map[string]int, 1)
  556. c.people[0] = map[string]int{}
  557. c.people[0]["one"] = 1
  558. c.people[0]["two"] = 2
  559. c.people[0]["three"] = 3
  560. c.people[0]["four"] = 4
  561. *c.renames = []rename{{ToName: "four", FromName: "one"}}
  562. files, people := c.propagateRenames(map[string]bool{"two": true, "three": true, "four": true})
  563. assert.Len(t, files, 3)
  564. assert.Len(t, people, 1)
  565. assert.Equal(t, files["two"], map[string]int{"two": 10, "three": 1, "four": 9})
  566. assert.Equal(t, files["three"], map[string]int{"two": 1, "three": 3, "four": 6})
  567. assert.Equal(t, files["four"], map[string]int{"two": 9, "three": 6, "four": 2})
  568. assert.Equal(t, people[0], map[string]int{"two": 2, "three": 3, "four": 5})
  569. }
  570. func getSlice(vals ...int) []int {
  571. return vals
  572. }
  573. func getCouplesMap(vals ...int) map[int]int64 {
  574. res := map[int]int64{}
  575. for i := 0; i < len(vals); i += 2 {
  576. res[vals[i]] = int64(vals[i+1])
  577. }
  578. return res
  579. }