| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520 | // +build !disable_babelfishpackage uastimport (	"bytes"	"fmt"	"io/ioutil"	"os"	"path"	"testing"	"github.com/gogo/protobuf/proto"	"github.com/stretchr/testify/assert"	"gopkg.in/bblfsh/sdk.v2/uast"	"gopkg.in/bblfsh/sdk.v2/uast/nodes"	"gopkg.in/src-d/go-git.v4/plumbing"	"gopkg.in/src-d/go-git.v4/plumbing/object"	"gopkg.in/src-d/hercules.v9/internal/core"	"gopkg.in/src-d/hercules.v9/internal/pb"	items "gopkg.in/src-d/hercules.v9/internal/plumbing"	"gopkg.in/src-d/hercules.v9/internal/test")func fixtureUASTExtractor() *Extractor {	exr := Extractor{Endpoint: "0.0.0.0:9432"}	err := exr.Initialize(test.Repository)	if err != nil {		panic(err)	}	return &exr}func AddHash(t *testing.T, cache map[plumbing.Hash]*items.CachedBlob, hash string) {	objhash := plumbing.NewHash(hash)	blob, err := test.Repository.BlobObject(objhash)	assert.Nil(t, err)	cb := &items.CachedBlob{Blob: *blob}	err = cb.Cache()	assert.Nil(t, err)	cache[objhash] = cb}func TestUASTExtractorMeta(t *testing.T) {	exr := fixtureUASTExtractor()	assert.Equal(t, exr.Name(), "UAST")	assert.Equal(t, len(exr.Provides()), 1)	assert.Equal(t, exr.Provides()[0], DependencyUasts)	assert.Equal(t, len(exr.Requires()), 2)	assert.Equal(t, exr.Requires()[0], items.DependencyTreeChanges)	assert.Equal(t, exr.Requires()[1], items.DependencyBlobCache)	opts := exr.ListConfigurationOptions()	assert.Len(t, opts, 5)	assert.Equal(t, opts[0].Name, ConfigUASTEndpoint)	assert.Equal(t, opts[1].Name, ConfigUASTTimeout)	assert.Equal(t, opts[2].Name, ConfigUASTPoolSize)	assert.Equal(t, opts[3].Name, ConfigUASTFailOnErrors)	assert.Equal(t, opts[4].Name, ConfigUASTIgnoreMissingDrivers)	feats := exr.Features()	assert.Len(t, feats, 1)	assert.Equal(t, feats[0], FeatureUast)}func TestUASTExtractorConfiguration(t *testing.T) {	exr := fixtureUASTExtractor()	facts := map[string]interface{}{}	exr.Configure(facts)	facts[ConfigUASTEndpoint] = "localhost:9432"	facts[ConfigUASTTimeout] = 15	facts[ConfigUASTPoolSize] = 7	facts[ConfigUASTFailOnErrors] = true	facts[ConfigUASTIgnoreMissingDrivers] = []string{"test"}	exr.Configure(facts)	assert.Equal(t, exr.Endpoint, facts[ConfigUASTEndpoint])	assert.NotNil(t, exr.Context)	assert.Equal(t, exr.PoolSize, facts[ConfigUASTPoolSize])	assert.Equal(t, exr.FailOnErrors, true)	assert.Equal(t, exr.IgnoredMissingDrivers, map[string]bool{"test": true})}func TestUASTExtractorRegistration(t *testing.T) {	summoned := core.Registry.Summon((&Extractor{}).Name())	assert.Len(t, summoned, 1)	assert.Equal(t, summoned[0].Name(), "UAST")	summoned = core.Registry.Summon((&Extractor{}).Provides()[0])	assert.Len(t, summoned, 1)	assert.Equal(t, summoned[0].Name(), "UAST")}func TestUASTExtractorNoBabelfish(t *testing.T) {	exr := Extractor{Endpoint: "0.0.0.0:56934"}	err := exr.Initialize(test.Repository)	assert.NotNil(t, err)}func TestUASTExtractorConsume(t *testing.T) {	exr := fixtureUASTExtractor()	changes := make(object.Changes, 4)	// 2b1ed978194a94edeabbca6de7ff3b5771d4d665	treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(		"96c6ece9b2f3c7c51b83516400d278dea5605100"))	treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(		"251f2094d7b523d5bcc60e663b6cf38151bf8844"))	changes[0] = &object.Change{From: object.ChangeEntry{		Name: "analyser.go",		Tree: treeFrom,		TreeEntry: object.TreeEntry{			Name: "analyser.go",			Mode: 0100644,			Hash: plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1"),		},	}, To: object.ChangeEntry{},	}	changes[1] = &object.Change{From: object.ChangeEntry{		Name: "cmd/hercules/main.go",		Tree: treeFrom,		TreeEntry: object.TreeEntry{			Name: "cmd/hercules/main.go",			Mode: 0100644,			Hash: plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"),		},	}, To: object.ChangeEntry{		Name: "cmd/hercules/main.go",		Tree: treeTo,		TreeEntry: object.TreeEntry{			Name: "cmd/hercules/main.go",			Mode: 0100644,			Hash: plumbing.NewHash("f7d918ec500e2f925ecde79b51cc007bac27de72"),		},	},	}	changes[2] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{		Name: "linux.png",		Tree: treeTo,		TreeEntry: object.TreeEntry{			Name: "linux.png",			Mode: 0100644,			Hash: plumbing.NewHash("81f2b6d1fa5357f90e9dead150cd515720897545"),		},	},	}	changes[3] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{		Name: "README.md",		Tree: treeTo,		TreeEntry: object.TreeEntry{			Name: "README.md",			Mode: 0100644,			Hash: plumbing.NewHash("5248c86995f6d60eb57730da18b5e020a4341863"),		},	},	}	cache := map[plumbing.Hash]*items.CachedBlob{}	for _, hash := range []string{		"baa64828831d174f40140e4b3cfa77d1e917a2c1",		"5d78f57d732aed825764347ec6f3ab74d50d0619",		"c29112dbd697ad9b401333b80c18a63951bc18d9",		"f7d918ec500e2f925ecde79b51cc007bac27de72",		"81f2b6d1fa5357f90e9dead150cd515720897545",		"5248c86995f6d60eb57730da18b5e020a4341863",	} {		AddHash(t, cache, hash)	}	deps := map[string]interface{}{}	deps[items.DependencyBlobCache] = cache	deps[items.DependencyTreeChanges] = changes	deps[core.DependencyCommit], _ = test.Repository.CommitObject(		plumbing.NewHash("2b1ed978194a94edeabbca6de7ff3b5771d4d665"))	res, err := exr.Consume(deps)	assert.Len(t, res[DependencyUasts], 1)	assert.Nil(t, err)	res, err = exr.Consume(deps)	assert.Len(t, res[DependencyUasts], 1)	assert.Nil(t, err)	exr.FailOnErrors = true	res, err = exr.Consume(deps)	assert.Nil(t, res)	assert.NotNil(t, err)	exr.FailOnErrors = false	hash := plumbing.NewHash("5d78f57d732aed825764347ec6f3ab74d50d0619")	changes[1] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{		Name: "labours.py",		Tree: treeTo,		TreeEntry: object.TreeEntry{			Name: "labours.py",			Mode: 0100644,			Hash: hash,		},	},	}	deps[items.DependencyTreeChanges] = changes[:2]	res, err = exr.Consume(deps)	assert.Nil(t, err)	uasts := res[DependencyUasts].(map[plumbing.Hash]nodes.Node)	assert.Equal(t, len(uasts), 1)	assert.Equal(t, len(uasts[hash].(nodes.Object)["body"].(nodes.Array)), 24)	exr.IgnoredMissingDrivers = map[string]bool{}	changes[2] = changes[3]	deps[items.DependencyTreeChanges] = changes[:3]	res, err = exr.Consume(deps)	assert.Nil(t, err)	exr.FailOnErrors = true	res, err = exr.Consume(deps)	assert.Nil(t, res)	assert.NotNil(t, err)	exr.FailOnErrors = false}func TestUASTExtractorFork(t *testing.T) {	exr1 := fixtureUASTExtractor()	clones := exr1.Fork(1)	assert.Len(t, clones, 1)	exr2 := clones[0].(*Extractor)	assert.True(t, exr1 == exr2)	exr1.Merge([]core.PipelineItem{exr2})}func fixtureUASTChanges() *Changes {	ch := Changes{}	ch.Configure(nil)	ch.Initialize(test.Repository)	return &ch}func TestUASTChangesMeta(t *testing.T) {	ch := fixtureUASTChanges()	assert.Equal(t, ch.Name(), "UASTChanges")	assert.Equal(t, len(ch.Provides()), 1)	assert.Equal(t, ch.Provides()[0], DependencyUastChanges)	assert.Equal(t, len(ch.Requires()), 2)	assert.Equal(t, ch.Requires()[0], DependencyUasts)	assert.Equal(t, ch.Requires()[1], items.DependencyTreeChanges)	opts := ch.ListConfigurationOptions()	assert.Len(t, opts, 0)}func TestUASTChangesRegistration(t *testing.T) {	summoned := core.Registry.Summon((&Changes{}).Name())	assert.Len(t, summoned, 1)	assert.Equal(t, summoned[0].Name(), "UASTChanges")	summoned = core.Registry.Summon((&Changes{}).Provides()[0])	assert.True(t, len(summoned) >= 1)	matched := false	for _, tp := range summoned {		matched = matched || tp.Name() == "UASTChanges"	}	assert.True(t, matched)}func newNodeWithType(name string) nodes.Node {	return nodes.Object{		uast.KeyType:  nodes.String(name),		uast.KeyToken: nodes.String("my_token"),	}}func TestUASTChangesConsume(t *testing.T) {	var uastsArray []nodes.Node	uasts := map[plumbing.Hash]nodes.Node{}	hash := plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe")	uasts[hash] = newNodeWithType("uno")	uastsArray = append(uastsArray, uasts[hash])	hash = plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9")	uasts[hash] = newNodeWithType("dos")	uastsArray = append(uastsArray, uasts[hash])	hash = plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1")	uasts[hash] = newNodeWithType("tres")	uastsArray = append(uastsArray, uasts[hash])	hash = plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9")	uasts[hash] = newNodeWithType("quatro")	uastsArray = append(uastsArray, uasts[hash])	changes := make(object.Changes, 3)	treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(		"a1eb2ea76eb7f9bfbde9b243861474421000eb96"))	treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(		"994eac1cd07235bb9815e547a75c84265dea00f5"))	changes[0] = &object.Change{From: object.ChangeEntry{		Name: "analyser.go",		Tree: treeFrom,		TreeEntry: object.TreeEntry{			Name: "analyser.go",			Mode: 0100644,			Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),		},	}, To: object.ChangeEntry{		Name: "analyser.go",		Tree: treeTo,		TreeEntry: object.TreeEntry{			Name: "analyser.go",			Mode: 0100644,			Hash: plumbing.NewHash("baa64828831d174f40140e4b3cfa77d1e917a2c1"),		},	}}	changes[1] = &object.Change{From: object.ChangeEntry{}, To: object.ChangeEntry{		Name: "cmd/hercules/main.go",		Tree: treeTo,		TreeEntry: object.TreeEntry{			Name: "cmd/hercules/main.go",			Mode: 0100644,			Hash: plumbing.NewHash("c29112dbd697ad9b401333b80c18a63951bc18d9"),		},	},	}	changes[2] = &object.Change{To: object.ChangeEntry{}, From: object.ChangeEntry{		Name: ".travis.yml",		Tree: treeTo,		TreeEntry: object.TreeEntry{			Name: ".travis.yml",			Mode: 0100644,			Hash: plumbing.NewHash("291286b4ac41952cbd1389fda66420ec03c1a9fe"),		},	},	}	deps := map[string]interface{}{}	deps[DependencyUasts] = uasts	deps[items.DependencyTreeChanges] = changes	ch := fixtureUASTChanges()	ch.cache[changes[0].From.TreeEntry.Hash] = uastsArray[3]	ch.cache[changes[2].From.TreeEntry.Hash] = uastsArray[0]	resultMap, err := ch.Consume(deps)	assert.Nil(t, err)	result := resultMap[DependencyUastChanges].([]Change)	assert.Len(t, result, 3)	assert.Equal(t, result[0].Change, changes[0])	assert.Equal(t, result[0].Before, uastsArray[3])	assert.Equal(t, result[0].After, uastsArray[2])	assert.Equal(t, result[1].Change, changes[1])	assert.Nil(t, result[1].Before)	assert.Equal(t, result[1].After, uastsArray[1])	assert.Equal(t, result[2].Change, changes[2])	assert.Equal(t, result[2].Before, uastsArray[0])	assert.Nil(t, result[2].After)}func TestUASTChangesFork(t *testing.T) {	changes1 := fixtureUASTChanges()	changes1.cache[plumbing.ZeroHash] = nil	clones := changes1.Fork(1)	assert.Len(t, clones, 1)	changes2 := clones[0].(*Changes)	assert.False(t, changes1 == changes2)	assert.Equal(t, changes1.cache, changes2.cache)	delete(changes1.cache, plumbing.ZeroHash)	assert.Len(t, changes2.cache, 1)	changes1.Merge([]core.PipelineItem{changes2})}func fixtureUASTChangesSaver() *ChangesSaver {	ch := ChangesSaver{}	ch.Initialize(test.Repository)	return &ch}func TestUASTChangesSaverMeta(t *testing.T) {	chs := fixtureUASTChangesSaver()	assert.Equal(t, chs.Name(), "UASTChangesSaver")	assert.True(t, len(chs.Description()) > 0)	assert.Equal(t, len(chs.Provides()), 0)	assert.Equal(t, len(chs.Requires()), 1)	assert.Equal(t, chs.Requires()[0], DependencyUastChanges)	opts := chs.ListConfigurationOptions()	assert.Len(t, opts, 1)	assert.Equal(t, opts[0].Name, ConfigUASTChangesSaverOutputPath)	assert.Equal(t, chs.Flag(), "dump-uast-changes")}func TestUASTChangesSaverConfiguration(t *testing.T) {	facts := map[string]interface{}{}	chs := fixtureUASTChangesSaver()	chs.Configure(facts)	assert.Empty(t, chs.OutputPath)	facts[ConfigUASTChangesSaverOutputPath] = "libre"	chs.Configure(facts)	assert.Equal(t, chs.OutputPath, "libre")}func TestUASTChangesSaverRegistration(t *testing.T) {	summoned := core.Registry.Summon((&ChangesSaver{}).Name())	assert.Len(t, summoned, 1)	assert.Equal(t, summoned[0].Name(), "UASTChangesSaver")	leaves := core.Registry.GetLeaves()	matched := false	for _, tp := range leaves {		if tp.Flag() == (&ChangesSaver{}).Flag() {			matched = true			break		}	}	assert.True(t, matched)}func TestUASTChangesSaverPayload(t *testing.T) {	chs := fixtureUASTChangesSaver()	deps := map[string]interface{}{}	changes := make([]Change, 1)	deps[DependencyUastChanges] = changes	deps[core.DependencyCommit], _ = test.Repository.CommitObject(		plumbing.NewHash("2b1ed978194a94edeabbca6de7ff3b5771d4d665"))	treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(		"a1eb2ea76eb7f9bfbde9b243861474421000eb96"))	treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(		"994eac1cd07235bb9815e547a75c84265dea00f5"))	changes[0] = Change{Before: nodes.Object{}, After: nodes.Object{},		Change: &object.Change{From: object.ChangeEntry{			Name: "analyser.go",			Tree: treeFrom,			TreeEntry: object.TreeEntry{				Name: "analyser.go",				Mode: 0100644,				Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),			},		}, To: object.ChangeEntry{			Name: "analyser.go",			Tree: treeTo,			TreeEntry: object.TreeEntry{				Name: "analyser.go",				Mode: 0100644,				Hash: plumbing.NewHash("334cde09da4afcb74f8d2b3e6fd6cce61228b485"),			},		}}}	chs.Consume(deps)	res := chs.Finalize()	tmpdir, err := ioutil.TempDir("", "hercules-test-")	assert.Nil(t, err)	defer os.RemoveAll(tmpdir)	chs.OutputPath = tmpdir	buffer := &bytes.Buffer{}	chs.Serialize(res, true, buffer)	pbResults := &pb.UASTChangesSaverResults{}	proto.Unmarshal(buffer.Bytes(), pbResults)	assert.Len(t, pbResults.Changes, 1)	assert.Equal(t, pbResults.Changes[0].FileName, "analyser.go")	assert.Equal(t, pbResults.Changes[0].SrcAfter,		path.Join(tmpdir, "0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.src"))	assert.Equal(t, pbResults.Changes[0].SrcBefore,		path.Join(tmpdir, "0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.src"))	assert.Equal(t, pbResults.Changes[0].UastAfter,		path.Join(tmpdir, "0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.pb"))	assert.Equal(t, pbResults.Changes[0].UastBefore,		path.Join(tmpdir, "0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.pb"))	checkFiles := func() {		files, err := ioutil.ReadDir(tmpdir)		assert.Nil(t, err)		assert.Len(t, files, 4)		names := map[string]int{			"0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.src":  1,			"0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.src": 1,			"0_0_after_334cde09da4afcb74f8d2b3e6fd6cce61228b485.pb":   1,			"0_0_before_dc248ba2b22048cc730c571a748e8ffcf7085ab9.pb":  1,		}		matches := 0		for _, fi := range files {			matches += names[fi.Name()]			os.Remove(fi.Name())		}		assert.Equal(t, matches, len(names))	}	checkFiles()	buffer.Truncate(0)	chs.Serialize(res, false, buffer)	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}`, tmpdir, tmpdir, tmpdir, tmpdir))	checkFiles()}func TestUASTChangesSaverConsumeMerge(t *testing.T) {	chs := fixtureUASTChangesSaver()	deps := map[string]interface{}{}	changes := make([]Change, 1)	deps[DependencyUastChanges] = changes	deps[core.DependencyCommit], _ = test.Repository.CommitObject(		plumbing.NewHash("2b1ed978194a94edeabbca6de7ff3b5771d4d665"))	treeFrom, _ := test.Repository.TreeObject(plumbing.NewHash(		"a1eb2ea76eb7f9bfbde9b243861474421000eb96"))	treeTo, _ := test.Repository.TreeObject(plumbing.NewHash(		"994eac1cd07235bb9815e547a75c84265dea00f5"))	changes[0] = Change{Before: nodes.Object{}, After: nodes.Object{},		Change: &object.Change{From: object.ChangeEntry{			Name: "analyser.go",			Tree: treeFrom,			TreeEntry: object.TreeEntry{				Name: "analyser.go",				Mode: 0100644,				Hash: plumbing.NewHash("dc248ba2b22048cc730c571a748e8ffcf7085ab9"),			},		}, To: object.ChangeEntry{			Name: "analyser.go",			Tree: treeTo,			TreeEntry: object.TreeEntry{				Name: "analyser.go",				Mode: 0100644,				Hash: plumbing.NewHash("334cde09da4afcb74f8d2b3e6fd6cce61228b485"),			},		}}}	deps[core.DependencyCommit], _ = test.Repository.CommitObject(		plumbing.NewHash("cce947b98a050c6d356bc6ba95030254914027b1"))	chs.Consume(deps)	assert.Len(t, chs.result, 1)	chs.Consume(deps)	assert.Len(t, chs.result, 2)	deps[core.DependencyCommit], _ = test.Repository.CommitObject(		plumbing.NewHash("dd9dd084d5851d7dc4399fc7dbf3d8292831ebc5"))	chs.Consume(deps)	assert.Len(t, chs.result, 3)	chs.Consume(deps)	assert.Len(t, chs.result, 3)}func TestUASTChangesSaverFork(t *testing.T) {	saver1 := fixtureUASTChangesSaver()	clones := saver1.Fork(1)	assert.Len(t, clones, 1)	saver2 := clones[0].(*ChangesSaver)	assert.True(t, saver1 == saver2)	saver1.Merge([]core.PipelineItem{saver2})}
 |