| // Copyright 2015 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. | 
 |  | 
 | package android | 
 |  | 
 | import ( | 
 | 	"fmt" | 
 | 	"reflect" | 
 | 	"strings" | 
 | 	"testing" | 
 |  | 
 | 	"github.com/google/blueprint" | 
 | ) | 
 |  | 
 | type mutatorTestModule struct { | 
 | 	ModuleBase | 
 | 	props struct { | 
 | 		Deps_missing_deps    []string | 
 | 		Mutator_missing_deps []string | 
 | 	} | 
 |  | 
 | 	missingDeps []string | 
 | } | 
 |  | 
 | func mutatorTestModuleFactory() Module { | 
 | 	module := &mutatorTestModule{} | 
 | 	module.AddProperties(&module.props) | 
 | 	InitAndroidModule(module) | 
 | 	return module | 
 | } | 
 |  | 
 | func (m *mutatorTestModule) GenerateAndroidBuildActions(ctx ModuleContext) { | 
 | 	ctx.Build(pctx, BuildParams{ | 
 | 		Rule:   Touch, | 
 | 		Output: PathForModuleOut(ctx, "output"), | 
 | 	}) | 
 |  | 
 | 	m.missingDeps = ctx.GetMissingDependencies() | 
 | } | 
 |  | 
 | func (m *mutatorTestModule) DepsMutator(ctx BottomUpMutatorContext) { | 
 | 	ctx.AddDependency(ctx.Module(), nil, m.props.Deps_missing_deps...) | 
 | } | 
 |  | 
 | func addMissingDependenciesMutator(ctx TopDownMutatorContext) { | 
 | 	ctx.AddMissingDependencies(ctx.Module().(*mutatorTestModule).props.Mutator_missing_deps) | 
 | } | 
 |  | 
 | func TestMutatorAddMissingDependencies(t *testing.T) { | 
 | 	bp := ` | 
 | 		test { | 
 | 			name: "foo", | 
 | 			deps_missing_deps: ["regular_missing_dep"], | 
 | 			mutator_missing_deps: ["added_missing_dep"], | 
 | 		} | 
 | 	` | 
 |  | 
 | 	result := GroupFixturePreparers( | 
 | 		PrepareForTestWithAllowMissingDependencies, | 
 | 		FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
 | 			ctx.RegisterModuleType("test", mutatorTestModuleFactory) | 
 | 			ctx.PreDepsMutators(func(ctx RegisterMutatorsContext) { | 
 | 				ctx.TopDown("add_missing_dependencies", addMissingDependenciesMutator) | 
 | 			}) | 
 | 		}), | 
 | 		FixtureWithRootAndroidBp(bp), | 
 | 	).RunTest(t) | 
 |  | 
 | 	foo := result.ModuleForTests("foo", "").Module().(*mutatorTestModule) | 
 |  | 
 | 	AssertDeepEquals(t, "foo missing deps", []string{"added_missing_dep", "regular_missing_dep"}, foo.missingDeps) | 
 | } | 
 |  | 
 | func TestModuleString(t *testing.T) { | 
 | 	bp := ` | 
 | 		test { | 
 | 			name: "foo", | 
 | 		} | 
 | 	` | 
 |  | 
 | 	var moduleStrings []string | 
 |  | 
 | 	GroupFixturePreparers( | 
 | 		FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
 |  | 
 | 			ctx.PreArchMutators(func(ctx RegisterMutatorsContext) { | 
 | 				ctx.BottomUp("pre_arch", func(ctx BottomUpMutatorContext) { | 
 | 					moduleStrings = append(moduleStrings, ctx.Module().String()) | 
 | 					ctx.CreateVariations("a", "b") | 
 | 				}) | 
 | 				ctx.TopDown("rename_top_down", func(ctx TopDownMutatorContext) { | 
 | 					moduleStrings = append(moduleStrings, ctx.Module().String()) | 
 | 					ctx.Rename(ctx.Module().base().Name() + "_renamed1") | 
 | 				}) | 
 | 			}) | 
 |  | 
 | 			ctx.PreDepsMutators(func(ctx RegisterMutatorsContext) { | 
 | 				ctx.BottomUp("pre_deps", func(ctx BottomUpMutatorContext) { | 
 | 					moduleStrings = append(moduleStrings, ctx.Module().String()) | 
 | 					ctx.CreateVariations("c", "d") | 
 | 				}) | 
 | 			}) | 
 |  | 
 | 			ctx.PostDepsMutators(func(ctx RegisterMutatorsContext) { | 
 | 				ctx.BottomUp("post_deps", func(ctx BottomUpMutatorContext) { | 
 | 					moduleStrings = append(moduleStrings, ctx.Module().String()) | 
 | 					ctx.CreateLocalVariations("e", "f") | 
 | 				}) | 
 | 				ctx.BottomUp("rename_bottom_up", func(ctx BottomUpMutatorContext) { | 
 | 					moduleStrings = append(moduleStrings, ctx.Module().String()) | 
 | 					ctx.Rename(ctx.Module().base().Name() + "_renamed2") | 
 | 				}) | 
 | 				ctx.BottomUp("final", func(ctx BottomUpMutatorContext) { | 
 | 					moduleStrings = append(moduleStrings, ctx.Module().String()) | 
 | 				}) | 
 | 			}) | 
 |  | 
 | 			ctx.RegisterModuleType("test", mutatorTestModuleFactory) | 
 | 		}), | 
 | 		FixtureWithRootAndroidBp(bp), | 
 | 	).RunTest(t) | 
 |  | 
 | 	want := []string{ | 
 | 		// Initial name. | 
 | 		"foo{}", | 
 |  | 
 | 		// After pre_arch (reversed because rename_top_down is TopDown so it visits in reverse order). | 
 | 		"foo{pre_arch:b}", | 
 | 		"foo{pre_arch:a}", | 
 |  | 
 | 		// After rename_top_down. | 
 | 		"foo_renamed1{pre_arch:a}", | 
 | 		"foo_renamed1{pre_arch:b}", | 
 |  | 
 | 		// After pre_deps. | 
 | 		"foo_renamed1{pre_arch:a,pre_deps:c}", | 
 | 		"foo_renamed1{pre_arch:a,pre_deps:d}", | 
 | 		"foo_renamed1{pre_arch:b,pre_deps:c}", | 
 | 		"foo_renamed1{pre_arch:b,pre_deps:d}", | 
 |  | 
 | 		// After post_deps. | 
 | 		"foo_renamed1{pre_arch:a,pre_deps:c,post_deps:e}", | 
 | 		"foo_renamed1{pre_arch:a,pre_deps:c,post_deps:f}", | 
 | 		"foo_renamed1{pre_arch:a,pre_deps:d,post_deps:e}", | 
 | 		"foo_renamed1{pre_arch:a,pre_deps:d,post_deps:f}", | 
 | 		"foo_renamed1{pre_arch:b,pre_deps:c,post_deps:e}", | 
 | 		"foo_renamed1{pre_arch:b,pre_deps:c,post_deps:f}", | 
 | 		"foo_renamed1{pre_arch:b,pre_deps:d,post_deps:e}", | 
 | 		"foo_renamed1{pre_arch:b,pre_deps:d,post_deps:f}", | 
 |  | 
 | 		// After rename_bottom_up. | 
 | 		"foo_renamed2{pre_arch:a,pre_deps:c,post_deps:e}", | 
 | 		"foo_renamed2{pre_arch:a,pre_deps:c,post_deps:f}", | 
 | 		"foo_renamed2{pre_arch:a,pre_deps:d,post_deps:e}", | 
 | 		"foo_renamed2{pre_arch:a,pre_deps:d,post_deps:f}", | 
 | 		"foo_renamed2{pre_arch:b,pre_deps:c,post_deps:e}", | 
 | 		"foo_renamed2{pre_arch:b,pre_deps:c,post_deps:f}", | 
 | 		"foo_renamed2{pre_arch:b,pre_deps:d,post_deps:e}", | 
 | 		"foo_renamed2{pre_arch:b,pre_deps:d,post_deps:f}", | 
 | 	} | 
 |  | 
 | 	AssertDeepEquals(t, "module String() values", want, moduleStrings) | 
 | } | 
 |  | 
 | func TestFinalDepsPhase(t *testing.T) { | 
 | 	bp := ` | 
 | 		test { | 
 | 			name: "common_dep_1", | 
 | 		} | 
 | 		test { | 
 | 			name: "common_dep_2", | 
 | 		} | 
 | 		test { | 
 | 			name: "foo", | 
 | 		} | 
 | 	` | 
 |  | 
 | 	finalGot := map[string]int{} | 
 |  | 
 | 	GroupFixturePreparers( | 
 | 		FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
 | 			dep1Tag := struct { | 
 | 				blueprint.BaseDependencyTag | 
 | 			}{} | 
 | 			dep2Tag := struct { | 
 | 				blueprint.BaseDependencyTag | 
 | 			}{} | 
 |  | 
 | 			ctx.PostDepsMutators(func(ctx RegisterMutatorsContext) { | 
 | 				ctx.BottomUp("far_deps_1", func(ctx BottomUpMutatorContext) { | 
 | 					if !strings.HasPrefix(ctx.ModuleName(), "common_dep") { | 
 | 						ctx.AddFarVariationDependencies([]blueprint.Variation{}, dep1Tag, "common_dep_1") | 
 | 					} | 
 | 				}) | 
 | 				ctx.BottomUp("variant", func(ctx BottomUpMutatorContext) { | 
 | 					ctx.CreateLocalVariations("a", "b") | 
 | 				}) | 
 | 			}) | 
 |  | 
 | 			ctx.FinalDepsMutators(func(ctx RegisterMutatorsContext) { | 
 | 				ctx.BottomUp("far_deps_2", func(ctx BottomUpMutatorContext) { | 
 | 					if !strings.HasPrefix(ctx.ModuleName(), "common_dep") { | 
 | 						ctx.AddFarVariationDependencies([]blueprint.Variation{}, dep2Tag, "common_dep_2") | 
 | 					} | 
 | 				}) | 
 | 				ctx.BottomUp("final", func(ctx BottomUpMutatorContext) { | 
 | 					finalGot[ctx.Module().String()] += 1 | 
 | 					ctx.VisitDirectDeps(func(mod Module) { | 
 | 						finalGot[fmt.Sprintf("%s -> %s", ctx.Module().String(), mod)] += 1 | 
 | 					}) | 
 | 				}) | 
 | 			}) | 
 |  | 
 | 			ctx.RegisterModuleType("test", mutatorTestModuleFactory) | 
 | 		}), | 
 | 		FixtureWithRootAndroidBp(bp), | 
 | 	).RunTest(t) | 
 |  | 
 | 	finalWant := map[string]int{ | 
 | 		"common_dep_1{variant:a}":                   1, | 
 | 		"common_dep_1{variant:b}":                   1, | 
 | 		"common_dep_2{variant:a}":                   1, | 
 | 		"common_dep_2{variant:b}":                   1, | 
 | 		"foo{variant:a}":                            1, | 
 | 		"foo{variant:a} -> common_dep_1{variant:a}": 1, | 
 | 		"foo{variant:a} -> common_dep_2{variant:a}": 1, | 
 | 		"foo{variant:b}":                            1, | 
 | 		"foo{variant:b} -> common_dep_1{variant:b}": 1, | 
 | 		"foo{variant:b} -> common_dep_2{variant:a}": 1, | 
 | 	} | 
 |  | 
 | 	AssertDeepEquals(t, "final", finalWant, finalGot) | 
 | } | 
 |  | 
 | func TestNoCreateVariationsInFinalDeps(t *testing.T) { | 
 | 	checkErr := func() { | 
 | 		if err := recover(); err == nil || !strings.Contains(fmt.Sprintf("%s", err), "not allowed in FinalDepsMutators") { | 
 | 			panic("Expected FinalDepsMutators consistency check to fail") | 
 | 		} | 
 | 	} | 
 |  | 
 | 	GroupFixturePreparers( | 
 | 		FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
 | 			ctx.FinalDepsMutators(func(ctx RegisterMutatorsContext) { | 
 | 				ctx.BottomUp("vars", func(ctx BottomUpMutatorContext) { | 
 | 					defer checkErr() | 
 | 					ctx.CreateVariations("a", "b") | 
 | 				}) | 
 | 				ctx.BottomUp("local_vars", func(ctx BottomUpMutatorContext) { | 
 | 					defer checkErr() | 
 | 					ctx.CreateLocalVariations("a", "b") | 
 | 				}) | 
 | 			}) | 
 |  | 
 | 			ctx.RegisterModuleType("test", mutatorTestModuleFactory) | 
 | 		}), | 
 | 		FixtureWithRootAndroidBp(`test {name: "foo"}`), | 
 | 	).RunTest(t) | 
 | } | 
 |  | 
 | func TestConvertApexAvailableToTags(t *testing.T) { | 
 | 	input := []string{ | 
 | 		"com.android.adbd", | 
 | 		"//apex_available:platform", | 
 | 	} | 
 | 	actual := ConvertApexAvailableToTags(input) | 
 | 	expected := []string{ | 
 | 		"apex_available=com.android.adbd", | 
 | 		"apex_available=//apex_available:platform", | 
 | 	} | 
 | 	if !reflect.DeepEqual(actual, expected) { | 
 | 		t.Errorf("Expected: %v, actual: %v", expected, actual) | 
 | 	} | 
 |  | 
 | 	if ConvertApexAvailableToTags(nil) != nil { | 
 | 		t.Errorf("Expected providing nil to return nil") | 
 | 	} | 
 | } |