file_history_test.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  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 fixtureFileHistory() *FileHistoryAnalysis {
  17. fh := FileHistoryAnalysis{}
  18. fh.Initialize(test.Repository)
  19. return &fh
  20. }
  21. func TestFileHistoryMeta(t *testing.T) {
  22. fh := fixtureFileHistory()
  23. assert.Equal(t, fh.Name(), "FileHistoryAnalysis")
  24. assert.Equal(t, len(fh.Provides()), 0)
  25. assert.Equal(t, len(fh.Requires()), 3)
  26. assert.Equal(t, fh.Requires()[0], items.DependencyTreeChanges)
  27. assert.Equal(t, fh.Requires()[1], items.DependencyLineStats)
  28. assert.Equal(t, fh.Requires()[2], identity.DependencyAuthor)
  29. assert.Len(t, fh.ListConfigurationOptions(), 0)
  30. assert.Nil(t, fh.Configure(nil))
  31. logger := core.NewLogger()
  32. assert.NoError(t, fh.Configure(map[string]interface{}{
  33. core.ConfigLogger: logger,
  34. }))
  35. assert.Equal(t, logger, fh.l)
  36. }
  37. func TestFileHistoryRegistration(t *testing.T) {
  38. summoned := core.Registry.Summon((&FileHistoryAnalysis{}).Name())
  39. assert.Len(t, summoned, 1)
  40. assert.Equal(t, summoned[0].Name(), "FileHistoryAnalysis")
  41. leaves := core.Registry.GetLeaves()
  42. matched := false
  43. for _, tp := range leaves {
  44. if tp.Flag() == (&FileHistoryAnalysis{}).Flag() {
  45. matched = true
  46. break
  47. }
  48. }
  49. assert.True(t, matched)
  50. }
  51. func TestFileHistoryConsume(t *testing.T) {
  52. fh, deps := bakeFileHistoryForSerialization(t)
  53. validate := func() {
  54. assert.Len(t, fh.files, 3)
  55. assert.Equal(t, fh.files["cmd/hercules/main.go"].People,
  56. map[int]items.LineStats{1: ls(0, 207, 0)})
  57. assert.Equal(t, fh.files[".travis.yml"].People, map[int]items.LineStats{1: ls(12, 0, 0)})
  58. assert.Equal(t, fh.files["analyser.go"].People, map[int]items.LineStats{1: ls(628, 9, 67)})
  59. assert.Len(t, fh.files["analyser.go"].Hashes, 2)
  60. assert.Equal(t, fh.files["analyser.go"].Hashes[0], plumbing.NewHash(
  61. "ffffffffffffffffffffffffffffffffffffffff"))
  62. assert.Equal(t, fh.files["analyser.go"].Hashes[1], plumbing.NewHash(
  63. "2b1ed978194a94edeabbca6de7ff3b5771d4d665"))
  64. assert.Len(t, fh.files[".travis.yml"].Hashes, 1)
  65. assert.Equal(t, fh.files[".travis.yml"].Hashes[0], plumbing.NewHash(
  66. "2b1ed978194a94edeabbca6de7ff3b5771d4d665"))
  67. assert.Len(t, fh.files["cmd/hercules/main.go"].Hashes, 2)
  68. assert.Equal(t, fh.files["cmd/hercules/main.go"].Hashes[0], plumbing.NewHash(
  69. "0000000000000000000000000000000000000000"))
  70. assert.Equal(t, fh.files["cmd/hercules/main.go"].Hashes[1], plumbing.NewHash(
  71. "2b1ed978194a94edeabbca6de7ff3b5771d4d665"))
  72. }
  73. validate()
  74. res := fh.Finalize().(FileHistoryResult)
  75. assert.Equal(t, 2, len(res.Files))
  76. for key, val := range res.Files {
  77. assert.Equal(t, val, *fh.files[key])
  78. }
  79. deps[core.DependencyIsMerge] = true
  80. cres, err := fh.Consume(deps)
  81. assert.Nil(t, cres)
  82. assert.Nil(t, err)
  83. validate()
  84. fh.lastCommit = &object.Commit{}
  85. assert.Panics(t, func() { fh.Finalize() })
  86. }
  87. func TestFileHistoryFork(t *testing.T) {
  88. fh1 := fixtureFileHistory()
  89. clones := fh1.Fork(1)
  90. assert.Len(t, clones, 1)
  91. fh2 := clones[0].(*FileHistoryAnalysis)
  92. assert.True(t, fh1 == fh2)
  93. fh1.Merge([]core.PipelineItem{fh2})
  94. }
  95. func TestFileHistorySerializeText(t *testing.T) {
  96. fh, _ := bakeFileHistoryForSerialization(t)
  97. res := fh.Finalize().(FileHistoryResult)
  98. buffer := &bytes.Buffer{}
  99. assert.Nil(t, fh.Serialize(res, false, buffer))
  100. assert.Equal(t, buffer.String(), ` - .travis.yml:
  101. commits: ["2b1ed978194a94edeabbca6de7ff3b5771d4d665"]
  102. people: {1:[12,0,0]}
  103. - cmd/hercules/main.go:
  104. commits: ["0000000000000000000000000000000000000000","2b1ed978194a94edeabbca6de7ff3b5771d4d665"]
  105. people: {1:[0,207,0]}
  106. `)
  107. }
  108. func TestFileHistorySerializeBinary(t *testing.T) {
  109. fh, _ := bakeFileHistoryForSerialization(t)
  110. res := fh.Finalize().(FileHistoryResult)
  111. buffer := &bytes.Buffer{}
  112. assert.Nil(t, fh.Serialize(res, true, buffer))
  113. msg := pb.FileHistoryResultMessage{}
  114. assert.Nil(t, proto.Unmarshal(buffer.Bytes(), &msg))
  115. assert.Len(t, msg.Files, 2)
  116. assert.Len(t, msg.Files[".travis.yml"].Commits, 1)
  117. assert.Equal(t, msg.Files[".travis.yml"].Commits[0], "2b1ed978194a94edeabbca6de7ff3b5771d4d665")
  118. assert.Len(t, msg.Files["cmd/hercules/main.go"].Commits, 2)
  119. assert.Equal(t, msg.Files["cmd/hercules/main.go"].Commits[0],
  120. "0000000000000000000000000000000000000000")
  121. assert.Equal(t, msg.Files["cmd/hercules/main.go"].Commits[1],
  122. "2b1ed978194a94edeabbca6de7ff3b5771d4d665")
  123. assert.Equal(t, msg.Files[".travis.yml"].ChangesByDeveloper,
  124. map[int32]*pb.LineStats{1: {Added: 12, Removed: 0, Changed: 0}})
  125. assert.Equal(t, msg.Files["cmd/hercules/main.go"].ChangesByDeveloper,
  126. map[int32]*pb.LineStats{1: {Added: 0, Removed: 207, Changed: 0}})
  127. }
  128. func bakeFileHistoryForSerialization(t *testing.T) (*FileHistoryAnalysis, map[string]interface{}) {
  129. fh := fixtureFileHistory()
  130. deps := map[string]interface{}{}
  131. cache := map[plumbing.Hash]*items.CachedBlob{}
  132. AddHash(t, cache, "291286b4ac41952cbd1389fda66420ec03c1a9fe")
  133. AddHash(t, cache, "c29112dbd697ad9b401333b80c18a63951bc18d9")
  134. AddHash(t, cache, "baa64828831d174f40140e4b3cfa77d1e917a2c1")
  135. AddHash(t, cache, "dc248ba2b22048cc730c571a748e8ffcf7085ab9")
  136. deps[items.DependencyBlobCache] = cache
  137. changes := make(object.Changes, 3)
  138. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  139. "a1eb2ea76eb7f9bfbde9b243861474421000eb96"))
  140. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  141. "994eac1cd07235bb9815e547a75c84265dea00f5"))
  142. changes[0] = &object.Change{From: object.ChangeEntry{
  143. Name: "analyser.go",
  144. Tree: treeFrom,
  145. TreeEntry: object.TreeEntry{
  146. Name: "analyser.go",
  147. Mode: 0100644,
  148. Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),
  149. },
  150. }, To: object.ChangeEntry{
  151. Name: "analyser.go",
  152. Tree: treeTo,
  153. TreeEntry: object.TreeEntry{
  154. Name: "analyser.go",
  155. Mode: 0100644,
  156. Hash: plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1"),
  157. },
  158. }}
  159. changes[1] = &object.Change{To: object.ChangeEntry{}, From: object.ChangeEntry{
  160. Name: "cmd/hercules/main.go",
  161. Tree: treeTo,
  162. TreeEntry: object.TreeEntry{
  163. Name: "cmd/hercules/main.go",
  164. Mode: 0100644,
  165. Hash: plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"),
  166. },
  167. },
  168. }
  169. changes[2] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  170. Name: ".travis.yml",
  171. Tree: treeTo,
  172. TreeEntry: object.TreeEntry{
  173. Name: ".travis.yml",
  174. Mode: 0100644,
  175. Hash: plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"),
  176. },
  177. },
  178. }
  179. deps[items.DependencyTreeChanges] = changes
  180. commit, _ := test.Repository.CommitObject(plumbing.NewHash(
  181. "2b1ed978194a94edeabbca6de7ff3b5771d4d665"))
  182. deps[core.DependencyCommit] = commit
  183. deps[core.DependencyIsMerge] = false
  184. deps[identity.DependencyAuthor] = 1
  185. fd := fixtures.FileDiff()
  186. result, err := fd.Consume(deps)
  187. assert.Nil(t, err)
  188. deps[items.DependencyFileDiff] = result[items.DependencyFileDiff]
  189. lineStats, err := (&items.LinesStatsCalculator{}).Consume(deps)
  190. assert.Nil(t, err)
  191. deps[items.DependencyLineStats] = lineStats[items.DependencyLineStats]
  192. fh.files["cmd/hercules/main.go"] = &FileHistory{Hashes: []plumbing.Hash{plumbing.NewHash(
  193. "0000000000000000000000000000000000000000")}}
  194. fh.files["analyser.go"] = &FileHistory{Hashes: []plumbing.Hash{plumbing.NewHash(
  195. "ffffffffffffffffffffffffffffffffffffffff")}}
  196. cres, err := fh.Consume(deps)
  197. assert.Nil(t, cres)
  198. assert.Nil(t, err)
  199. return fh, deps
  200. }