2015-01-23 23:15:10 +01:00
|
|
|
// Copyright 2014 Google Inc. All rights reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2014-05-28 01:34:41 +02:00
|
|
|
package blueprint
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
2015-10-10 02:31:27 +02:00
|
|
|
type Walker interface {
|
|
|
|
Walk() bool
|
|
|
|
}
|
|
|
|
|
2014-05-28 01:34:41 +02:00
|
|
|
type fooModule struct {
|
|
|
|
properties struct {
|
|
|
|
Foo string
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-06 18:10:40 +02:00
|
|
|
func newFooModule() (Module, []interface{}) {
|
2014-05-28 01:34:41 +02:00
|
|
|
m := &fooModule{}
|
2014-10-06 18:10:40 +02:00
|
|
|
return m, []interface{}{&m.properties}
|
2014-05-28 01:34:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *fooModule) GenerateBuildActions(ModuleContext) {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *fooModule) Foo() string {
|
|
|
|
return f.properties.Foo
|
|
|
|
}
|
|
|
|
|
2015-10-10 02:31:27 +02:00
|
|
|
func (f *fooModule) Walk() bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2014-05-28 01:34:41 +02:00
|
|
|
type barModule struct {
|
|
|
|
properties struct {
|
|
|
|
Bar bool
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-06 18:10:40 +02:00
|
|
|
func newBarModule() (Module, []interface{}) {
|
2014-05-28 01:34:41 +02:00
|
|
|
m := &barModule{}
|
2014-10-06 18:10:40 +02:00
|
|
|
return m, []interface{}{&m.properties}
|
2014-05-28 01:34:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *barModule) GenerateBuildActions(ModuleContext) {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *barModule) Bar() bool {
|
|
|
|
return b.properties.Bar
|
|
|
|
}
|
|
|
|
|
2015-10-10 02:31:27 +02:00
|
|
|
func (b *barModule) Walk() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2014-05-28 01:34:41 +02:00
|
|
|
func TestContextParse(t *testing.T) {
|
|
|
|
ctx := NewContext()
|
2014-10-06 18:10:40 +02:00
|
|
|
ctx.RegisterModuleType("foo_module", newFooModule)
|
|
|
|
ctx.RegisterModuleType("bar_module", newBarModule)
|
2014-05-28 01:34:41 +02:00
|
|
|
|
|
|
|
r := bytes.NewBufferString(`
|
|
|
|
foo_module {
|
|
|
|
name: "MyFooModule",
|
|
|
|
deps: ["MyBarModule"],
|
|
|
|
}
|
|
|
|
|
|
|
|
bar_module {
|
|
|
|
name: "MyBarModule",
|
|
|
|
}
|
|
|
|
`)
|
|
|
|
|
2015-04-21 01:50:54 +02:00
|
|
|
_, _, _, errs := ctx.parse(".", "Blueprint", r, nil)
|
2014-05-28 01:34:41 +02:00
|
|
|
if len(errs) > 0 {
|
|
|
|
t.Errorf("unexpected parse errors:")
|
|
|
|
for _, err := range errs {
|
|
|
|
t.Errorf(" %s", err)
|
|
|
|
}
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
2015-10-29 23:32:56 +01:00
|
|
|
errs = ctx.ResolveDependencies(nil)
|
2014-05-28 01:34:41 +02:00
|
|
|
if len(errs) > 0 {
|
|
|
|
t.Errorf("unexpected dep errors:")
|
|
|
|
for _, err := range errs {
|
|
|
|
t.Errorf(" %s", err)
|
|
|
|
}
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
}
|
2015-10-10 02:31:27 +02:00
|
|
|
|
|
|
|
// |---B===D - represents a non-walkable edge
|
|
|
|
// A = represents a walkable edge
|
|
|
|
// |===C---E===G
|
|
|
|
// | | A should not be visited because it's the root node.
|
|
|
|
// |===F===| B, D and E should not be walked.
|
|
|
|
func TestWalkDeps(t *testing.T) {
|
|
|
|
ctx := NewContext()
|
2016-06-03 00:30:20 +02:00
|
|
|
ctx.MockFileSystem(map[string][]byte{
|
|
|
|
"Blueprints": []byte(`
|
|
|
|
foo_module {
|
|
|
|
name: "A",
|
|
|
|
deps: ["B", "C"],
|
|
|
|
}
|
|
|
|
|
|
|
|
bar_module {
|
|
|
|
name: "B",
|
|
|
|
deps: ["D"],
|
|
|
|
}
|
|
|
|
|
|
|
|
foo_module {
|
|
|
|
name: "C",
|
|
|
|
deps: ["E", "F"],
|
|
|
|
}
|
|
|
|
|
|
|
|
foo_module {
|
|
|
|
name: "D",
|
|
|
|
}
|
|
|
|
|
|
|
|
bar_module {
|
|
|
|
name: "E",
|
|
|
|
deps: ["G"],
|
|
|
|
}
|
|
|
|
|
|
|
|
foo_module {
|
|
|
|
name: "F",
|
|
|
|
deps: ["G"],
|
|
|
|
}
|
|
|
|
|
|
|
|
foo_module {
|
|
|
|
name: "G",
|
|
|
|
}
|
|
|
|
`),
|
|
|
|
})
|
|
|
|
|
2015-10-10 02:31:27 +02:00
|
|
|
ctx.RegisterModuleType("foo_module", newFooModule)
|
|
|
|
ctx.RegisterModuleType("bar_module", newBarModule)
|
2016-06-03 00:30:20 +02:00
|
|
|
_, errs := ctx.ParseBlueprintsFiles("Blueprints")
|
|
|
|
if len(errs) > 0 {
|
|
|
|
t.Errorf("unexpected parse errors:")
|
|
|
|
for _, err := range errs {
|
|
|
|
t.Errorf(" %s", err)
|
|
|
|
}
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
|
|
|
errs = ctx.ResolveDependencies(nil)
|
|
|
|
if len(errs) > 0 {
|
|
|
|
t.Errorf("unexpected dep errors:")
|
|
|
|
for _, err := range errs {
|
|
|
|
t.Errorf(" %s", err)
|
|
|
|
}
|
|
|
|
t.FailNow()
|
|
|
|
}
|
2015-10-10 02:31:27 +02:00
|
|
|
|
|
|
|
var output string
|
|
|
|
topModule := ctx.moduleGroups["A"].modules[0]
|
|
|
|
ctx.walkDeps(topModule,
|
|
|
|
func(module, parent Module) bool {
|
|
|
|
if module.(Walker).Walk() {
|
|
|
|
output += ctx.ModuleName(module)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
if output != "CFG" {
|
|
|
|
t.Fatalf("unexpected walkDeps behaviour: %s\nshould be: CFG", output)
|
|
|
|
}
|
|
|
|
}
|