uast_test.go 16 KB

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