devs_test.go 15 KB

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