devs_test.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. package leaves
  2. import (
  3. "bytes"
  4. "testing"
  5. "github.com/gogo/protobuf/proto"
  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.v9/internal/core"
  10. "gopkg.in/src-d/hercules.v9/internal/pb"
  11. items "gopkg.in/src-d/hercules.v9/internal/plumbing"
  12. "gopkg.in/src-d/hercules.v9/internal/plumbing/identity"
  13. "gopkg.in/src-d/hercules.v9/internal/test"
  14. "gopkg.in/src-d/hercules.v9/internal/test/fixtures"
  15. )
  16. func fixtureDevs() *DevsAnalysis {
  17. d := DevsAnalysis{}
  18. d.Initialize(test.Repository)
  19. people := [...]string{"one@srcd", "two@srcd"}
  20. d.reversedPeopleDict = people[:]
  21. return &d
  22. }
  23. func TestDevsMeta(t *testing.T) {
  24. d := fixtureDevs()
  25. assert.Equal(t, d.Name(), "Devs")
  26. assert.Equal(t, len(d.Provides()), 0)
  27. assert.Equal(t, len(d.Requires()), 5)
  28. assert.Equal(t, d.Requires()[0], identity.DependencyAuthor)
  29. assert.Equal(t, d.Requires()[1], items.DependencyTreeChanges)
  30. assert.Equal(t, d.Requires()[2], items.DependencyDay)
  31. assert.Equal(t, d.Requires()[3], items.DependencyLanguages)
  32. assert.Equal(t, d.Requires()[4], items.DependencyLineStats)
  33. assert.Equal(t, d.Flag(), "devs")
  34. assert.Len(t, d.ListConfigurationOptions(), 1)
  35. assert.Equal(t, d.ListConfigurationOptions()[0].Name, ConfigDevsConsiderEmptyCommits)
  36. assert.Equal(t, d.ListConfigurationOptions()[0].Flag, "empty-commits")
  37. assert.Equal(t, d.ListConfigurationOptions()[0].Type, core.BoolConfigurationOption)
  38. assert.Equal(t, d.ListConfigurationOptions()[0].Default, false)
  39. assert.True(t, len(d.Description()) > 0)
  40. }
  41. func TestDevsRegistration(t *testing.T) {
  42. summoned := core.Registry.Summon((&DevsAnalysis{}).Name())
  43. assert.Len(t, summoned, 1)
  44. assert.Equal(t, summoned[0].Name(), "Devs")
  45. leaves := core.Registry.GetLeaves()
  46. matched := false
  47. for _, tp := range leaves {
  48. if tp.Flag() == (&DevsAnalysis{}).Flag() {
  49. matched = true
  50. break
  51. }
  52. }
  53. assert.True(t, matched)
  54. }
  55. func TestDevsConfigure(t *testing.T) {
  56. devs := DevsAnalysis{}
  57. facts := map[string]interface{}{}
  58. facts[ConfigDevsConsiderEmptyCommits] = true
  59. devs.Configure(facts)
  60. assert.Equal(t, devs.ConsiderEmptyCommits, true)
  61. }
  62. func TestDevsInitialize(t *testing.T) {
  63. d := fixtureDevs()
  64. assert.NotNil(t, d.days)
  65. }
  66. func TestDevsConsumeFinalize(t *testing.T) {
  67. devs := fixtureDevs()
  68. deps := map[string]interface{}{}
  69. // stage 1
  70. deps[identity.DependencyAuthor] = 0
  71. deps[items.DependencyDay] = 0
  72. cache := map[plumbing.Hash]*items.CachedBlob{}
  73. AddHash(t, cache, "291286b4ac41952cbd1389fda66420ec03c1a9fe")
  74. AddHash(t, cache, "c29112dbd697ad9b401333b80c18a63951bc18d9")
  75. AddHash(t, cache, "baa64828831d174f40140e4b3cfa77d1e917a2c1")
  76. AddHash(t, cache, "dc248ba2b22048cc730c571a748e8ffcf7085ab9")
  77. deps[items.DependencyBlobCache] = cache
  78. deps[items.DependencyLanguages] = map[plumbing.Hash]string{
  79. plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"): "Go",
  80. plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"): "Go",
  81. plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1"): "Go",
  82. plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"): "Go",
  83. }
  84. changes := make(object.Changes, 3)
  85. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  86. "a1eb2ea76eb7f9bfbde9b243861474421000eb96"))
  87. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  88. "994eac1cd07235bb9815e547a75c84265dea00f5"))
  89. changes[0] = &object.Change{From: object.ChangeEntry{
  90. Name: "analyser.go",
  91. Tree: treeFrom,
  92. TreeEntry: object.TreeEntry{
  93. Name: "analyser.go",
  94. Mode: 0100644,
  95. Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),
  96. },
  97. }, To: object.ChangeEntry{
  98. Name: "analyser.go",
  99. Tree: treeTo,
  100. TreeEntry: object.TreeEntry{
  101. Name: "analyser.go",
  102. Mode: 0100644,
  103. Hash: plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1"),
  104. },
  105. }}
  106. changes[1] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  107. Name: "cmd/hercules/main.go",
  108. Tree: treeTo,
  109. TreeEntry: object.TreeEntry{
  110. Name: "cmd/hercules/main.go",
  111. Mode: 0100644,
  112. Hash: plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"),
  113. },
  114. },
  115. }
  116. changes[2] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  117. Name: ".travis.yml",
  118. Tree: treeTo,
  119. TreeEntry: object.TreeEntry{
  120. Name: ".travis.yml",
  121. Mode: 0100644,
  122. Hash: plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"),
  123. },
  124. },
  125. }
  126. deps[items.DependencyTreeChanges] = changes
  127. fd := fixtures.FileDiff()
  128. result, err := fd.Consume(deps)
  129. assert.Nil(t, err)
  130. deps[items.DependencyFileDiff] = result[items.DependencyFileDiff]
  131. deps[core.DependencyCommit], _ = test.Repository.CommitObject(plumbing.NewHash(
  132. "cce947b98a050c6d356bc6ba95030254914027b1"))
  133. deps[core.DependencyIsMerge] = false
  134. lsc := &items.LinesStatsCalculator{}
  135. lscres, err := lsc.Consume(deps)
  136. assert.Nil(t, err)
  137. deps[items.DependencyLineStats] = lscres[items.DependencyLineStats]
  138. result, err = devs.Consume(deps)
  139. assert.Nil(t, result)
  140. assert.Nil(t, err)
  141. assert.Len(t, devs.days, 1)
  142. day := devs.days[0]
  143. assert.Len(t, day, 1)
  144. dev := day[0]
  145. assert.Equal(t, dev.Commits, 1)
  146. assert.Equal(t, dev.Added, 847)
  147. assert.Equal(t, dev.Removed, 9)
  148. assert.Equal(t, dev.Changed, 67)
  149. assert.Equal(t, dev.Languages["Go"].Added, 847)
  150. assert.Equal(t, dev.Languages["Go"].Removed, 9)
  151. assert.Equal(t, dev.Languages["Go"].Changed, 67)
  152. deps[core.DependencyIsMerge] = true
  153. lscres, err = lsc.Consume(deps)
  154. assert.Nil(t, err)
  155. deps[items.DependencyLineStats] = lscres[items.DependencyLineStats]
  156. result, err = devs.Consume(deps)
  157. assert.Nil(t, result)
  158. assert.Nil(t, err)
  159. assert.Len(t, devs.days, 1)
  160. day = devs.days[0]
  161. assert.Len(t, day, 1)
  162. dev = day[0]
  163. assert.Equal(t, dev.Commits, 2)
  164. assert.Equal(t, dev.Added, 847)
  165. assert.Equal(t, dev.Removed, 9)
  166. assert.Equal(t, dev.Changed, 67)
  167. assert.Equal(t, dev.Languages["Go"].Added, 847)
  168. assert.Equal(t, dev.Languages["Go"].Removed, 9)
  169. assert.Equal(t, dev.Languages["Go"].Changed, 67)
  170. deps[core.DependencyIsMerge] = false
  171. deps[identity.DependencyAuthor] = 1
  172. lscres, err = lsc.Consume(deps)
  173. assert.Nil(t, err)
  174. deps[items.DependencyLineStats] = lscres[items.DependencyLineStats]
  175. result, err = devs.Consume(deps)
  176. assert.Nil(t, result)
  177. assert.Nil(t, err)
  178. assert.Len(t, devs.days, 1)
  179. day = devs.days[0]
  180. assert.Len(t, day, 2)
  181. for i := 0; i < 2; i++ {
  182. dev = day[i]
  183. if i == 0 {
  184. assert.Equal(t, dev.Commits, 2)
  185. } else {
  186. assert.Equal(t, dev.Commits, 1)
  187. }
  188. assert.Equal(t, dev.Added, 847)
  189. assert.Equal(t, dev.Removed, 9)
  190. assert.Equal(t, dev.Changed, 67)
  191. assert.Equal(t, dev.Languages["Go"].Added, 847)
  192. assert.Equal(t, dev.Languages["Go"].Removed, 9)
  193. assert.Equal(t, dev.Languages["Go"].Changed, 67)
  194. }
  195. result, err = devs.Consume(deps)
  196. assert.Nil(t, result)
  197. assert.Nil(t, err)
  198. assert.Len(t, devs.days, 1)
  199. day = devs.days[0]
  200. assert.Len(t, day, 2)
  201. dev = day[0]
  202. assert.Equal(t, dev.Commits, 2)
  203. assert.Equal(t, dev.Added, 847)
  204. assert.Equal(t, dev.Removed, 9)
  205. assert.Equal(t, dev.Changed, 67)
  206. assert.Equal(t, dev.Languages["Go"].Added, 847)
  207. assert.Equal(t, dev.Languages["Go"].Removed, 9)
  208. assert.Equal(t, dev.Languages["Go"].Changed, 67)
  209. dev = day[1]
  210. assert.Equal(t, dev.Commits, 2)
  211. assert.Equal(t, dev.Added, 847*2)
  212. assert.Equal(t, dev.Removed, 9*2)
  213. assert.Equal(t, dev.Changed, 67*2)
  214. assert.Equal(t, dev.Languages["Go"].Added, 847*2)
  215. assert.Equal(t, dev.Languages["Go"].Removed, 9*2)
  216. assert.Equal(t, dev.Languages["Go"].Changed, 67*2)
  217. deps[items.DependencyDay] = 1
  218. result, err = devs.Consume(deps)
  219. assert.Nil(t, result)
  220. assert.Nil(t, err)
  221. assert.Len(t, devs.days, 2)
  222. day = devs.days[0]
  223. assert.Len(t, day, 2)
  224. dev = day[0]
  225. assert.Equal(t, dev.Commits, 2)
  226. assert.Equal(t, dev.Added, 847)
  227. assert.Equal(t, dev.Removed, 9)
  228. assert.Equal(t, dev.Changed, 67)
  229. assert.Equal(t, dev.Languages["Go"].Added, 847)
  230. assert.Equal(t, dev.Languages["Go"].Removed, 9)
  231. assert.Equal(t, dev.Languages["Go"].Changed, 67)
  232. dev = day[1]
  233. assert.Equal(t, dev.Commits, 2)
  234. assert.Equal(t, dev.Added, 847*2)
  235. assert.Equal(t, dev.Removed, 9*2)
  236. assert.Equal(t, dev.Changed, 67*2)
  237. assert.Equal(t, dev.Languages["Go"].Added, 847*2)
  238. assert.Equal(t, dev.Languages["Go"].Removed, 9*2)
  239. assert.Equal(t, dev.Languages["Go"].Changed, 67*2)
  240. day = devs.days[1]
  241. assert.Len(t, day, 1)
  242. dev = day[1]
  243. assert.Equal(t, dev.Commits, 1)
  244. assert.Equal(t, dev.Added, 847)
  245. assert.Equal(t, dev.Removed, 9)
  246. assert.Equal(t, dev.Changed, 67)
  247. assert.Equal(t, dev.Languages["Go"].Added, 847)
  248. assert.Equal(t, dev.Languages["Go"].Removed, 9)
  249. assert.Equal(t, dev.Languages["Go"].Changed, 67)
  250. }
  251. func ls(added, removed, changed int) items.LineStats {
  252. return items.LineStats{Added: added, Removed: removed, Changed: changed}
  253. }
  254. func TestDevsFinalize(t *testing.T) {
  255. devs := fixtureDevs()
  256. devs.days[1] = map[int]*DevDay{}
  257. devs.days[1][1] = &DevDay{10, ls(20, 30, 40), nil}
  258. x := devs.Finalize().(DevsResult)
  259. assert.Equal(t, x.Days, devs.days)
  260. assert.Equal(t, x.reversedPeopleDict, devs.reversedPeopleDict)
  261. }
  262. func TestDevsFork(t *testing.T) {
  263. devs := fixtureDevs()
  264. clone := devs.Fork(1)[0].(*DevsAnalysis)
  265. assert.True(t, devs == clone)
  266. }
  267. func TestDevsSerialize(t *testing.T) {
  268. devs := fixtureDevs()
  269. devs.days[1] = map[int]*DevDay{}
  270. devs.days[1][0] = &DevDay{10, ls(20, 30, 40), map[string]items.LineStats{"Go": ls(2, 3, 4)}}
  271. devs.days[1][1] = &DevDay{1, ls(2, 3, 4), map[string]items.LineStats{"Go": ls(25, 35, 45)}}
  272. devs.days[10] = map[int]*DevDay{}
  273. devs.days[10][0] = &DevDay{11, ls(21, 31, 41), map[string]items.LineStats{"": ls(12, 13, 14)}}
  274. devs.days[10][identity.AuthorMissing] = &DevDay{
  275. 100, ls(200, 300, 400), map[string]items.LineStats{"Go": ls(32, 33, 34)}}
  276. res := devs.Finalize().(DevsResult)
  277. buffer := &bytes.Buffer{}
  278. err := devs.Serialize(res, false, buffer)
  279. assert.Nil(t, err)
  280. assert.Equal(t, ` days:
  281. 1:
  282. 0: [10, 20, 30, 40, {Go: [2, 3, 4]}]
  283. 1: [1, 2, 3, 4, {Go: [25, 35, 45]}]
  284. 10:
  285. 0: [11, 21, 31, 41, {none: [12, 13, 14]}]
  286. -1: [100, 200, 300, 400, {Go: [32, 33, 34]}]
  287. people:
  288. - "one@srcd"
  289. - "two@srcd"
  290. `, buffer.String())
  291. buffer = &bytes.Buffer{}
  292. err = devs.Serialize(res, true, buffer)
  293. assert.Nil(t, err)
  294. msg := pb.DevsAnalysisResults{}
  295. assert.Nil(t, proto.Unmarshal(buffer.Bytes(), &msg))
  296. assert.Equal(t, msg.DevIndex, devs.reversedPeopleDict)
  297. assert.Len(t, msg.Days, 2)
  298. assert.Len(t, msg.Days[1].Devs, 2)
  299. assert.Equal(t, msg.Days[1].Devs[0], &pb.DevDay{
  300. Commits: 10, Stats: &pb.LineStats{Added: 20, Removed: 30, Changed: 40},
  301. Languages: map[string]*pb.LineStats{"Go": {Added: 2, Removed: 3, Changed: 4}}})
  302. assert.Equal(t, msg.Days[1].Devs[1], &pb.DevDay{
  303. Commits: 1, Stats: &pb.LineStats{Added: 2, Removed: 3, Changed: 4},
  304. Languages: map[string]*pb.LineStats{"Go": {Added: 25, Removed: 35, Changed: 45}}})
  305. assert.Len(t, msg.Days[10].Devs, 2)
  306. assert.Equal(t, msg.Days[10].Devs[0], &pb.DevDay{
  307. Commits: 11, Stats: &pb.LineStats{Added: 21, Removed: 31, Changed: 41},
  308. Languages: map[string]*pb.LineStats{"": {Added: 12, Removed: 13, Changed: 14}}})
  309. assert.Equal(t, msg.Days[10].Devs[-1], &pb.DevDay{
  310. Commits: 100, Stats: &pb.LineStats{Added: 200, Removed: 300, Changed: 400},
  311. Languages: map[string]*pb.LineStats{"Go": {Added: 32, Removed: 33, Changed: 34}}})
  312. }
  313. func TestDevsDeserialize(t *testing.T) {
  314. devs := fixtureDevs()
  315. devs.days[1] = map[int]*DevDay{}
  316. devs.days[1][0] = &DevDay{10, ls(20, 30, 40), map[string]items.LineStats{"Go": ls(12, 13, 14)}}
  317. devs.days[1][1] = &DevDay{1, ls(2, 3, 4), map[string]items.LineStats{"Go": ls(22, 23, 24)}}
  318. devs.days[10] = map[int]*DevDay{}
  319. devs.days[10][0] = &DevDay{11, ls(21, 31, 41), map[string]items.LineStats{"Go": ls(32, 33, 34)}}
  320. devs.days[10][identity.AuthorMissing] = &DevDay{
  321. 100, ls(200, 300, 400), map[string]items.LineStats{"Go": ls(42, 43, 44)}}
  322. res := devs.Finalize().(DevsResult)
  323. buffer := &bytes.Buffer{}
  324. err := devs.Serialize(res, true, buffer)
  325. assert.Nil(t, err)
  326. rawres2, err := devs.Deserialize(buffer.Bytes())
  327. assert.Nil(t, err)
  328. res2 := rawres2.(DevsResult)
  329. assert.Equal(t, res, res2)
  330. }
  331. func TestDevsMergeResults(t *testing.T) {
  332. people1 := [...]string{"1@srcd", "2@srcd"}
  333. people2 := [...]string{"3@srcd", "1@srcd"}
  334. r1 := DevsResult{
  335. Days: map[int]map[int]*DevDay{},
  336. reversedPeopleDict: people1[:],
  337. }
  338. r1.Days[1] = map[int]*DevDay{}
  339. r1.Days[1][0] = &DevDay{10, ls(20, 30, 40), map[string]items.LineStats{"Go": ls(12, 13, 14)}}
  340. r1.Days[1][1] = &DevDay{1, ls(2, 3, 4), map[string]items.LineStats{"Go": ls(22, 23, 24)}}
  341. r1.Days[10] = map[int]*DevDay{}
  342. r1.Days[10][0] = &DevDay{11, ls(21, 31, 41), nil}
  343. r1.Days[10][identity.AuthorMissing] = &DevDay{
  344. 100, ls(200, 300, 400), map[string]items.LineStats{"Go": ls(32, 33, 34)}}
  345. r1.Days[11] = map[int]*DevDay{}
  346. r1.Days[11][1] = &DevDay{10, ls(20, 30, 40), map[string]items.LineStats{"Go": ls(42, 43, 44)}}
  347. r2 := DevsResult{
  348. Days: map[int]map[int]*DevDay{},
  349. reversedPeopleDict: people2[:],
  350. }
  351. r2.Days[1] = map[int]*DevDay{}
  352. r2.Days[1][0] = &DevDay{10, ls(20, 30, 40), map[string]items.LineStats{"Go": ls(12, 13, 14)}}
  353. r2.Days[1][1] = &DevDay{1, ls(2, 3, 4), map[string]items.LineStats{"Go": ls(22, 23, 24)}}
  354. r2.Days[2] = map[int]*DevDay{}
  355. r2.Days[2][0] = &DevDay{11, ls(21, 31, 41), map[string]items.LineStats{"Go": ls(32, 33, 34)}}
  356. r2.Days[2][identity.AuthorMissing] = &DevDay{
  357. 100, ls(200, 300, 400), map[string]items.LineStats{"Go": ls(42, 43, 44)}}
  358. r2.Days[10] = map[int]*DevDay{}
  359. r2.Days[10][0] = &DevDay{11, ls(21, 31, 41), map[string]items.LineStats{"Go": ls(52, 53, 54)}}
  360. r2.Days[10][identity.AuthorMissing] = &DevDay{
  361. 100, ls(200, 300, 400), map[string]items.LineStats{"Go": ls(62, 63, 64)}}
  362. devs := fixtureDevs()
  363. rm := devs.MergeResults(r1, r2, nil, nil).(DevsResult)
  364. peoplerm := [...]string{"1@srcd", "2@srcd", "3@srcd"}
  365. assert.Equal(t, rm.reversedPeopleDict, peoplerm[:])
  366. assert.Len(t, rm.Days, 4)
  367. assert.Equal(t, rm.Days[11], map[int]*DevDay{
  368. 1: {10, ls(20, 30, 40), map[string]items.LineStats{"Go": ls(42, 43, 44)}}})
  369. assert.Equal(t, rm.Days[2], map[int]*DevDay{
  370. identity.AuthorMissing: {100, ls(200, 300, 400), map[string]items.LineStats{"Go": ls(42, 43, 44)}},
  371. 2: {11, ls(21, 31, 41), map[string]items.LineStats{"Go": ls(32, 33, 34)}},
  372. })
  373. assert.Equal(t, rm.Days[1], map[int]*DevDay{
  374. 0: {11, ls(22, 33, 44), map[string]items.LineStats{"Go": ls(34, 36, 38)}},
  375. 1: {1, ls(2, 3, 4), map[string]items.LineStats{"Go": ls(22, 23, 24)}},
  376. 2: {10, ls(20, 30, 40), map[string]items.LineStats{"Go": ls(12, 13, 14)}},
  377. })
  378. assert.Equal(t, rm.Days[10], map[int]*DevDay{
  379. 0: {11, ls(21, 31, 41), map[string]items.LineStats{}},
  380. 2: {11, ls(21, 31, 41), map[string]items.LineStats{"Go": ls(52, 53, 54)}},
  381. identity.AuthorMissing: {
  382. 100 * 2, ls(200*2, 300*2, 400*2), map[string]items.LineStats{"Go": ls(94, 96, 98)}},
  383. })
  384. }