uast_test.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520
  1. // +build !disable_babelfish
  2. package uast
  3. import (
  4. "bytes"
  5. "fmt"
  6. "io/ioutil"
  7. "os"
  8. "path"
  9. "testing"
  10. "github.com/gogo/protobuf/proto"
  11. "github.com/stretchr/testify/assert"
  12. "gopkg.in/bblfsh/sdk.v2/uast"
  13. "gopkg.in/bblfsh/sdk.v2/uast/nodes"
  14. "gopkg.in/src-d/go-git.v4/plumbing"
  15. "gopkg.in/src-d/go-git.v4/plumbing/object"
  16. "gopkg.in/src-d/hercules.v10/internal/core"
  17. "gopkg.in/src-d/hercules.v10/internal/pb"
  18. items "gopkg.in/src-d/hercules.v10/internal/plumbing"
  19. "gopkg.in/src-d/hercules.v10/internal/test"
  20. )
  21. func fixtureUASTExtractor() *Extractor {
  22. exr := Extractor{Endpoint: "0.0.0.0:9432"}
  23. err := exr.Initialize(test.Repository)
  24. if err != nil {
  25. panic(err)
  26. }
  27. return &exr
  28. }
  29. func AddHash(t *testing.T, cache map[plumbing.Hash]*items.CachedBlob, hash string) {
  30. objhash := plumbing.NewHash(hash)
  31. blob, err := test.Repository.BlobObject(objhash)
  32. assert.Nil(t, err)
  33. cb := &items.CachedBlob{Blob: *blob}
  34. err = cb.Cache()
  35. assert.Nil(t, err)
  36. cache[objhash] = cb
  37. }
  38. func TestUASTExtractorMeta(t *testing.T) {
  39. exr := fixtureUASTExtractor()
  40. assert.Equal(t, exr.Name(), "UAST")
  41. assert.Equal(t, len(exr.Provides()), 1)
  42. assert.Equal(t, exr.Provides()[0], DependencyUasts)
  43. assert.Equal(t, len(exr.Requires()), 2)
  44. assert.Equal(t, exr.Requires()[0], items.DependencyTreeChanges)
  45. assert.Equal(t, exr.Requires()[1], items.DependencyBlobCache)
  46. opts := exr.ListConfigurationOptions()
  47. assert.Len(t, opts, 5)
  48. assert.Equal(t, opts[0].Name, ConfigUASTEndpoint)
  49. assert.Equal(t, opts[1].Name, ConfigUASTTimeout)
  50. assert.Equal(t, opts[2].Name, ConfigUASTPoolSize)
  51. assert.Equal(t, opts[3].Name, ConfigUASTFailOnErrors)
  52. assert.Equal(t, opts[4].Name, ConfigUASTIgnoreMissingDrivers)
  53. feats := exr.Features()
  54. assert.Len(t, feats, 1)
  55. assert.Equal(t, feats[0], FeatureUast)
  56. }
  57. func TestUASTExtractorConfiguration(t *testing.T) {
  58. exr := fixtureUASTExtractor()
  59. facts := map[string]interface{}{}
  60. exr.Configure(facts)
  61. facts[ConfigUASTEndpoint] = "localhost:9432"
  62. facts[ConfigUASTTimeout] = 15
  63. facts[ConfigUASTPoolSize] = 7
  64. facts[ConfigUASTFailOnErrors] = true
  65. facts[ConfigUASTIgnoreMissingDrivers] = []string{"test"}
  66. exr.Configure(facts)
  67. assert.Equal(t, exr.Endpoint, facts[ConfigUASTEndpoint])
  68. assert.NotNil(t, exr.Context)
  69. assert.Equal(t, exr.PoolSize, facts[ConfigUASTPoolSize])
  70. assert.Equal(t, exr.FailOnErrors, true)
  71. assert.Equal(t, exr.IgnoredMissingDrivers, map[string]bool{"test": true})
  72. }
  73. func TestUASTExtractorRegistration(t *testing.T) {
  74. summoned := core.Registry.Summon((&Extractor{}).Name())
  75. assert.Len(t, summoned, 1)
  76. assert.Equal(t, summoned[0].Name(), "UAST")
  77. summoned = core.Registry.Summon((&Extractor{}).Provides()[0])
  78. assert.Len(t, summoned, 1)
  79. assert.Equal(t, summoned[0].Name(), "UAST")
  80. }
  81. func TestUASTExtractorNoBabelfish(t *testing.T) {
  82. exr := Extractor{Endpoint: "0.0.0.0:56934"}
  83. err := exr.Initialize(test.Repository)
  84. assert.NotNil(t, err)
  85. }
  86. func TestUASTExtractorConsume(t *testing.T) {
  87. exr := fixtureUASTExtractor()
  88. changes := make(object.Changes, 4)
  89. // 2b1ed978194a94edeabbca6de7ff3b5771d4d665
  90. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  91. "96c6ece9b2f3c7c51b83516400d278dea5605100"))
  92. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  93. "251f2094d7b523d5bcc60e663b6cf38151bf8844"))
  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("baa64828831d174f40140e4b3cfa77d1e917a2c1"),
  101. },
  102. }, To: object.ChangeEntry{},
  103. }
  104. changes[1] = &object.Change{From: object.ChangeEntry{
  105. Name: "cmd/hercules/main.go",
  106. Tree: treeFrom,
  107. TreeEntry: object.TreeEntry{
  108. Name: "cmd/hercules/main.go",
  109. Mode: 0100644,
  110. Hash: plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"),
  111. },
  112. }, To: object.ChangeEntry{
  113. Name: "cmd/hercules/main.go",
  114. Tree: treeTo,
  115. TreeEntry: object.TreeEntry{
  116. Name: "cmd/hercules/main.go",
  117. Mode: 0100644,
  118. Hash: plumbing.NewHash("f7d918ec500e2f925ecde79b51cc007bac27de72"),
  119. },
  120. },
  121. }
  122. changes[2] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  123. Name: "linux.png",
  124. Tree: treeTo,
  125. TreeEntry: object.TreeEntry{
  126. Name: "linux.png",
  127. Mode: 0100644,
  128. Hash: plumbing.NewHash("81f2b6d1fa5357f90e9dead150cd515720897545"),
  129. },
  130. },
  131. }
  132. changes[3] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  133. Name: "README.md",
  134. Tree: treeTo,
  135. TreeEntry: object.TreeEntry{
  136. Name: "README.md",
  137. Mode: 0100644,
  138. Hash: plumbing.NewHash("5248c86995f6d60eb57730da18b5e020a4341863"),
  139. },
  140. },
  141. }
  142. cache := map[plumbing.Hash]*items.CachedBlob{}
  143. for _, hash := range []string{
  144. "baa64828831d174f40140e4b3cfa77d1e917a2c1",
  145. "5d78f57d732aed825764347ec6f3ab74d50d0619",
  146. "c29112dbd697ad9b401333b80c18a63951bc18d9",
  147. "f7d918ec500e2f925ecde79b51cc007bac27de72",
  148. "81f2b6d1fa5357f90e9dead150cd515720897545",
  149. "5248c86995f6d60eb57730da18b5e020a4341863",
  150. } {
  151. AddHash(t, cache, hash)
  152. }
  153. deps := map[string]interface{}{}
  154. deps[items.DependencyBlobCache] = cache
  155. deps[items.DependencyTreeChanges] = changes
  156. deps[core.DependencyCommit], _ = test.Repository.CommitObject(
  157. plumbing.NewHash("2b1ed978194a94edeabbca6de7ff3b5771d4d665"))
  158. res, err := exr.Consume(deps)
  159. assert.Len(t, res[DependencyUasts], 1)
  160. assert.Nil(t, err)
  161. res, err = exr.Consume(deps)
  162. assert.Len(t, res[DependencyUasts], 1)
  163. assert.Nil(t, err)
  164. exr.FailOnErrors = true
  165. res, err = exr.Consume(deps)
  166. assert.Nil(t, res)
  167. assert.NotNil(t, err)
  168. exr.FailOnErrors = false
  169. hash := plumbing.NewHash("5d78f57d732aed825764347ec6f3ab74d50d0619")
  170. changes[1] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  171. Name: "labours.py",
  172. Tree: treeTo,
  173. TreeEntry: object.TreeEntry{
  174. Name: "labours.py",
  175. Mode: 0100644,
  176. Hash: hash,
  177. },
  178. },
  179. }
  180. deps[items.DependencyTreeChanges] = changes[:2]
  181. res, err = exr.Consume(deps)
  182. assert.Nil(t, err)
  183. uasts := res[DependencyUasts].(map[plumbing.Hash]nodes.Node)
  184. assert.Equal(t, len(uasts), 1)
  185. assert.Equal(t, len(uasts[hash].(nodes.Object)["body"].(nodes.Array)), 24)
  186. exr.IgnoredMissingDrivers = map[string]bool{}
  187. changes[2] = changes[3]
  188. deps[items.DependencyTreeChanges] = changes[:3]
  189. res, err = exr.Consume(deps)
  190. assert.Nil(t, err)
  191. exr.FailOnErrors = true
  192. res, err = exr.Consume(deps)
  193. assert.Nil(t, res)
  194. assert.NotNil(t, err)
  195. exr.FailOnErrors = false
  196. }
  197. func TestUASTExtractorFork(t *testing.T) {
  198. exr1 := fixtureUASTExtractor()
  199. clones := exr1.Fork(1)
  200. assert.Len(t, clones, 1)
  201. exr2 := clones[0].(*Extractor)
  202. assert.True(t, exr1 == exr2)
  203. exr1.Merge([]core.PipelineItem{exr2})
  204. }
  205. func fixtureUASTChanges() *Changes {
  206. ch := Changes{}
  207. ch.Configure(nil)
  208. ch.Initialize(test.Repository)
  209. return &ch
  210. }
  211. func TestUASTChangesMeta(t *testing.T) {
  212. ch := fixtureUASTChanges()
  213. assert.Equal(t, ch.Name(), "UASTChanges")
  214. assert.Equal(t, len(ch.Provides()), 1)
  215. assert.Equal(t, ch.Provides()[0], DependencyUastChanges)
  216. assert.Equal(t, len(ch.Requires()), 2)
  217. assert.Equal(t, ch.Requires()[0], DependencyUasts)
  218. assert.Equal(t, ch.Requires()[1], items.DependencyTreeChanges)
  219. opts := ch.ListConfigurationOptions()
  220. assert.Len(t, opts, 0)
  221. }
  222. func TestUASTChangesRegistration(t *testing.T) {
  223. summoned := core.Registry.Summon((&Changes{}).Name())
  224. assert.Len(t, summoned, 1)
  225. assert.Equal(t, summoned[0].Name(), "UASTChanges")
  226. summoned = core.Registry.Summon((&Changes{}).Provides()[0])
  227. assert.True(t, len(summoned) >= 1)
  228. matched := false
  229. for _, tp := range summoned {
  230. matched = matched || tp.Name() == "UASTChanges"
  231. }
  232. assert.True(t, matched)
  233. }
  234. func newNodeWithType(name string) nodes.Node {
  235. return nodes.Object{
  236. uast.KeyType: nodes.String(name),
  237. uast.KeyToken: nodes.String("my_token"),
  238. }
  239. }
  240. func TestUASTChangesConsume(t *testing.T) {
  241. var uastsArray []nodes.Node
  242. uasts := map[plumbing.Hash]nodes.Node{}
  243. hash := plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe")
  244. uasts[hash] = newNodeWithType("uno")
  245. uastsArray = append(uastsArray, uasts[hash])
  246. hash = plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9")
  247. uasts[hash] = newNodeWithType("dos")
  248. uastsArray = append(uastsArray, uasts[hash])
  249. hash = plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1")
  250. uasts[hash] = newNodeWithType("tres")
  251. uastsArray = append(uastsArray, uasts[hash])
  252. hash = plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9")
  253. uasts[hash] = newNodeWithType("quatro")
  254. uastsArray = append(uastsArray, uasts[hash])
  255. changes := make(object.Changes, 3)
  256. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  257. "a1eb2ea76eb7f9bfbde9b243861474421000eb96"))
  258. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  259. "994eac1cd07235bb9815e547a75c84265dea00f5"))
  260. changes[0] = &object.Change{From: object.ChangeEntry{
  261. Name: "analyser.go",
  262. Tree: treeFrom,
  263. TreeEntry: object.TreeEntry{
  264. Name: "analyser.go",
  265. Mode: 0100644,
  266. Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),
  267. },
  268. }, To: object.ChangeEntry{
  269. Name: "analyser.go",
  270. Tree: treeTo,
  271. TreeEntry: object.TreeEntry{
  272. Name: "analyser.go",
  273. Mode: 0100644,
  274. Hash: plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1"),
  275. },
  276. }}
  277. changes[1] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{
  278. Name: "cmd/hercules/main.go",
  279. Tree: treeTo,
  280. TreeEntry: object.TreeEntry{
  281. Name: "cmd/hercules/main.go",
  282. Mode: 0100644,
  283. Hash: plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"),
  284. },
  285. },
  286. }
  287. changes[2] = &object.Change{To: object.ChangeEntry{}, From: object.ChangeEntry{
  288. Name: ".travis.yml",
  289. Tree: treeTo,
  290. TreeEntry: object.TreeEntry{
  291. Name: ".travis.yml",
  292. Mode: 0100644,
  293. Hash: plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"),
  294. },
  295. },
  296. }
  297. deps := map[string]interface{}{}
  298. deps[DependencyUasts] = uasts
  299. deps[items.DependencyTreeChanges] = changes
  300. ch := fixtureUASTChanges()
  301. ch.cache[changes[0].From.TreeEntry.Hash] = uastsArray[3]
  302. ch.cache[changes[2].From.TreeEntry.Hash] = uastsArray[0]
  303. resultMap, err := ch.Consume(deps)
  304. assert.Nil(t, err)
  305. result := resultMap[DependencyUastChanges].([]Change)
  306. assert.Len(t, result, 3)
  307. assert.Equal(t, result[0].Change, changes[0])
  308. assert.Equal(t, result[0].Before, uastsArray[3])
  309. assert.Equal(t, result[0].After, uastsArray[2])
  310. assert.Equal(t, result[1].Change, changes[1])
  311. assert.Nil(t, result[1].Before)
  312. assert.Equal(t, result[1].After, uastsArray[1])
  313. assert.Equal(t, result[2].Change, changes[2])
  314. assert.Equal(t, result[2].Before, uastsArray[0])
  315. assert.Nil(t, result[2].After)
  316. }
  317. func TestUASTChangesFork(t *testing.T) {
  318. changes1 := fixtureUASTChanges()
  319. changes1.cache[plumbing.ZeroHash] = nil
  320. clones := changes1.Fork(1)
  321. assert.Len(t, clones, 1)
  322. changes2 := clones[0].(*Changes)
  323. assert.False(t, changes1 == changes2)
  324. assert.Equal(t, changes1.cache, changes2.cache)
  325. delete(changes1.cache, plumbing.ZeroHash)
  326. assert.Len(t, changes2.cache, 1)
  327. changes1.Merge([]core.PipelineItem{changes2})
  328. }
  329. func fixtureUASTChangesSaver() *ChangesSaver {
  330. ch := ChangesSaver{}
  331. ch.Initialize(test.Repository)
  332. return &ch
  333. }
  334. func TestUASTChangesSaverMeta(t *testing.T) {
  335. chs := fixtureUASTChangesSaver()
  336. assert.Equal(t, chs.Name(), "UASTChangesSaver")
  337. assert.True(t, len(chs.Description()) > 0)
  338. assert.Equal(t, len(chs.Provides()), 0)
  339. assert.Equal(t, len(chs.Requires()), 1)
  340. assert.Equal(t, chs.Requires()[0], DependencyUastChanges)
  341. opts := chs.ListConfigurationOptions()
  342. assert.Len(t, opts, 1)
  343. assert.Equal(t, opts[0].Name, ConfigUASTChangesSaverOutputPath)
  344. assert.Equal(t, chs.Flag(), "dump-uast-changes")
  345. }
  346. func TestUASTChangesSaverConfiguration(t *testing.T) {
  347. facts := map[string]interface{}{}
  348. chs := fixtureUASTChangesSaver()
  349. chs.Configure(facts)
  350. assert.Empty(t, chs.OutputPath)
  351. facts[ConfigUASTChangesSaverOutputPath] = "libre"
  352. chs.Configure(facts)
  353. assert.Equal(t, chs.OutputPath, "libre")
  354. }
  355. func TestUASTChangesSaverRegistration(t *testing.T) {
  356. summoned := core.Registry.Summon((&ChangesSaver{}).Name())
  357. assert.Len(t, summoned, 1)
  358. assert.Equal(t, summoned[0].Name(), "UASTChangesSaver")
  359. leaves := core.Registry.GetLeaves()
  360. matched := false
  361. for _, tp := range leaves {
  362. if tp.Flag() == (&ChangesSaver{}).Flag() {
  363. matched = true
  364. break
  365. }
  366. }
  367. assert.True(t, matched)
  368. }
  369. func TestUASTChangesSaverPayload(t *testing.T) {
  370. chs := fixtureUASTChangesSaver()
  371. deps := map[string]interface{}{}
  372. changes := make([]Change, 1)
  373. deps[DependencyUastChanges] = changes
  374. deps[core.DependencyCommit], _ = test.Repository.CommitObject(
  375. plumbing.NewHash("2b1ed978194a94edeabbca6de7ff3b5771d4d665"))
  376. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  377. "a1eb2ea76eb7f9bfbde9b243861474421000eb96"))
  378. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  379. "994eac1cd07235bb9815e547a75c84265dea00f5"))
  380. changes[0] = Change{Before: nodes.Object{}, After: nodes.Object{},
  381. Change: &object.Change{From: object.ChangeEntry{
  382. Name: "analyser.go",
  383. Tree: treeFrom,
  384. TreeEntry: object.TreeEntry{
  385. Name: "analyser.go",
  386. Mode: 0100644,
  387. Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),
  388. },
  389. }, To: object.ChangeEntry{
  390. Name: "analyser.go",
  391. Tree: treeTo,
  392. TreeEntry: object.TreeEntry{
  393. Name: "analyser.go",
  394. Mode: 0100644,
  395. Hash: plumbing.NewHash("334cde09da4afcb74f8d2b3e6fd6cce61228b485"),
  396. },
  397. }}}
  398. chs.Consume(deps)
  399. res := chs.Finalize()
  400. tmpdir, err := ioutil.TempDir("", "hercules-test-")
  401. assert.Nil(t, err)
  402. defer os.RemoveAll(tmpdir)
  403. chs.OutputPath = tmpdir
  404. buffer := &bytes.Buffer{}
  405. chs.Serialize(res, true, buffer)
  406. pbResults := &pb.UASTChangesSaverResults{}
  407. proto.Unmarshal(buffer.Bytes(), pbResults)
  408. assert.Len(t, pbResults.Changes, 1)
  409. assert.Equal(t, pbResults.Changes[0].FileName, "analyser.go")
  410. assert.Equal(t, pbResults.Changes[0].SrcAfter,
  411. path.Join(tmpdir, "0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.src"))
  412. assert.Equal(t, pbResults.Changes[0].SrcBefore,
  413. path.Join(tmpdir, "0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.src"))
  414. assert.Equal(t, pbResults.Changes[0].UastAfter,
  415. path.Join(tmpdir, "0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.pb"))
  416. assert.Equal(t, pbResults.Changes[0].UastBefore,
  417. path.Join(tmpdir, "0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.pb"))
  418. checkFiles := func() {
  419. files, err := ioutil.ReadDir(tmpdir)
  420. assert.Nil(t, err)
  421. assert.Len(t, files, 4)
  422. names := map[string]int{
  423. "0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.src": 1,
  424. "0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.src": 1,
  425. "0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.pb": 1,
  426. "0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.pb": 1,
  427. }
  428. matches := 0
  429. for _, fi := range files {
  430. matches += names[fi.Name()]
  431. os.Remove(fi.Name())
  432. }
  433. assert.Equal(t, matches, len(names))
  434. }
  435. checkFiles()
  436. buffer.Truncate(0)
  437. chs.Serialize(res, false, buffer)
  438. assert.Equal(t, buffer.String(), fmt.Sprintf(` - {file: analyser.go, src0: %s/0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.src, src1: %s/0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.src, uast0: %s/0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.pb, uast1: %s/0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.pb}
  439. `, tmpdir, tmpdir, tmpdir, tmpdir))
  440. checkFiles()
  441. }
  442. func TestUASTChangesSaverConsumeMerge(t *testing.T) {
  443. chs := fixtureUASTChangesSaver()
  444. deps := map[string]interface{}{}
  445. changes := make([]Change, 1)
  446. deps[DependencyUastChanges] = changes
  447. deps[core.DependencyCommit], _ = test.Repository.CommitObject(
  448. plumbing.NewHash("2b1ed978194a94edeabbca6de7ff3b5771d4d665"))
  449. treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(
  450. "a1eb2ea76eb7f9bfbde9b243861474421000eb96"))
  451. treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(
  452. "994eac1cd07235bb9815e547a75c84265dea00f5"))
  453. changes[0] = Change{Before: nodes.Object{}, After: nodes.Object{},
  454. Change: &object.Change{From: object.ChangeEntry{
  455. Name: "analyser.go",
  456. Tree: treeFrom,
  457. TreeEntry: object.TreeEntry{
  458. Name: "analyser.go",
  459. Mode: 0100644,
  460. Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),
  461. },
  462. }, To: object.ChangeEntry{
  463. Name: "analyser.go",
  464. Tree: treeTo,
  465. TreeEntry: object.TreeEntry{
  466. Name: "analyser.go",
  467. Mode: 0100644,
  468. Hash: plumbing.NewHash("334cde09da4afcb74f8d2b3e6fd6cce61228b485"),
  469. },
  470. }}}
  471. deps[core.DependencyCommit], _ = test.Repository.CommitObject(
  472. plumbing.NewHash("cce947b98a050c6d356bc6ba95030254914027b1"))
  473. chs.Consume(deps)
  474. assert.Len(t, chs.result, 1)
  475. chs.Consume(deps)
  476. assert.Len(t, chs.result, 2)
  477. deps[core.DependencyCommit], _ = test.Repository.CommitObject(
  478. plumbing.NewHash("dd9dd084d5851d7dc4399fc7dbf3d8292831ebc5"))
  479. chs.Consume(deps)
  480. assert.Len(t, chs.result, 3)
  481. chs.Consume(deps)
  482. assert.Len(t, chs.result, 3)
  483. }
  484. func TestUASTChangesSaverFork(t *testing.T) {
  485. saver1 := fixtureUASTChangesSaver()
  486. clones := saver1.Fork(1)
  487. assert.Len(t, clones, 1)
  488. saver2 := clones[0].(*ChangesSaver)
  489. assert.True(t, saver1 == saver2)
  490. saver1.Merge([]core.PipelineItem{saver2})
  491. }