2017-07-14 05:43:27 +08:00
|
|
|
// Copyright 2017 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"
|
2017-11-18 05:29:40 +08:00
|
|
|
"path/filepath"
|
2018-03-12 16:34:26 +08:00
|
|
|
"regexp"
|
2017-07-14 05:43:27 +08:00
|
|
|
"strings"
|
2018-03-12 16:29:17 +08:00
|
|
|
"testing"
|
2017-07-14 05:43:27 +08:00
|
|
|
|
|
|
|
"github.com/google/blueprint"
|
|
|
|
)
|
|
|
|
|
|
|
|
func NewTestContext() *TestContext {
|
2017-11-30 08:47:17 +08:00
|
|
|
namespaceExportFilter := func(namespace *Namespace) bool {
|
|
|
|
return true
|
|
|
|
}
|
2017-12-02 09:10:33 +08:00
|
|
|
|
|
|
|
nameResolver := NewNameResolver(namespaceExportFilter)
|
|
|
|
ctx := &TestContext{
|
|
|
|
Context: blueprint.NewContext(),
|
|
|
|
NameResolver: nameResolver,
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.SetNameInterface(nameResolver)
|
2017-11-30 08:47:17 +08:00
|
|
|
|
|
|
|
return ctx
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
|
|
|
|
2017-09-16 08:33:55 +08:00
|
|
|
func NewTestArchContext() *TestContext {
|
|
|
|
ctx := NewTestContext()
|
|
|
|
ctx.preDeps = append(ctx.preDeps, registerArchMutator)
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
|
2017-07-14 05:43:27 +08:00
|
|
|
type TestContext struct {
|
|
|
|
*blueprint.Context
|
|
|
|
preArch, preDeps, postDeps []RegisterMutatorFunc
|
2017-12-02 09:10:33 +08:00
|
|
|
NameResolver *NameResolver
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *TestContext) PreArchMutators(f RegisterMutatorFunc) {
|
|
|
|
ctx.preArch = append(ctx.preArch, f)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *TestContext) PreDepsMutators(f RegisterMutatorFunc) {
|
|
|
|
ctx.preDeps = append(ctx.preDeps, f)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *TestContext) PostDepsMutators(f RegisterMutatorFunc) {
|
|
|
|
ctx.postDeps = append(ctx.postDeps, f)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *TestContext) Register() {
|
|
|
|
registerMutators(ctx.Context, ctx.preArch, ctx.preDeps, ctx.postDeps)
|
|
|
|
|
2017-11-29 15:55:23 +08:00
|
|
|
ctx.RegisterSingletonType("env", SingletonFactoryAdaptor(EnvSingleton))
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *TestContext) ModuleForTests(name, variant string) TestingModule {
|
|
|
|
var module Module
|
|
|
|
ctx.VisitAllModules(func(m blueprint.Module) {
|
|
|
|
if ctx.ModuleName(m) == name && ctx.ModuleSubDir(m) == variant {
|
|
|
|
module = m.(Module)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
if module == nil {
|
2017-09-28 08:05:30 +08:00
|
|
|
// find all the modules that do exist
|
|
|
|
allModuleNames := []string{}
|
|
|
|
ctx.VisitAllModules(func(m blueprint.Module) {
|
|
|
|
allModuleNames = append(allModuleNames, m.(Module).Name()+"("+ctx.ModuleSubDir(m)+")")
|
|
|
|
})
|
|
|
|
|
|
|
|
panic(fmt.Errorf("failed to find module %q variant %q."+
|
|
|
|
"\nall modules: %v", name, variant, allModuleNames))
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return TestingModule{module}
|
|
|
|
}
|
|
|
|
|
2018-07-11 09:49:27 +08:00
|
|
|
func (ctx *TestContext) ModuleVariantsForTests(name string) []string {
|
|
|
|
var variants []string
|
|
|
|
ctx.VisitAllModules(func(m blueprint.Module) {
|
|
|
|
if ctx.ModuleName(m) == name {
|
|
|
|
variants = append(variants, ctx.ModuleSubDir(m))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return variants
|
|
|
|
}
|
|
|
|
|
2017-11-18 05:29:40 +08:00
|
|
|
// MockFileSystem causes the Context to replace all reads with accesses to the provided map of
|
|
|
|
// filenames to contents stored as a byte slice.
|
|
|
|
func (ctx *TestContext) MockFileSystem(files map[string][]byte) {
|
|
|
|
// no module list file specified; find every file named Blueprints or Android.bp
|
|
|
|
pathsToParse := []string{}
|
|
|
|
for candidate := range files {
|
|
|
|
base := filepath.Base(candidate)
|
|
|
|
if base == "Blueprints" || base == "Android.bp" {
|
|
|
|
pathsToParse = append(pathsToParse, candidate)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(pathsToParse) < 1 {
|
|
|
|
panic(fmt.Sprintf("No Blueprint or Android.bp files found in mock filesystem: %v\n", files))
|
|
|
|
}
|
|
|
|
files[blueprint.MockModuleListFile] = []byte(strings.Join(pathsToParse, "\n"))
|
|
|
|
|
|
|
|
ctx.Context.MockFileSystem(files)
|
|
|
|
}
|
|
|
|
|
2019-01-06 14:09:19 +08:00
|
|
|
// TestingModule is wrapper around an android.Module that provides methods to find information about individual
|
|
|
|
// ctx.Build parameters for verification in tests.
|
2017-07-14 05:43:27 +08:00
|
|
|
type TestingModule struct {
|
|
|
|
module Module
|
|
|
|
}
|
|
|
|
|
2019-01-06 14:09:19 +08:00
|
|
|
// Module returns the Module wrapped by the TestingModule.
|
2017-07-14 05:43:27 +08:00
|
|
|
func (m TestingModule) Module() Module {
|
|
|
|
return m.module
|
|
|
|
}
|
|
|
|
|
2019-01-06 14:09:19 +08:00
|
|
|
// MaybeRule finds a call to ctx.Build with BuildParams.Rule set to a rule with the given name. Returns an empty
|
|
|
|
// BuildParams if no rule is found.
|
|
|
|
func (m TestingModule) MaybeRule(rule string) BuildParams {
|
2017-07-14 05:43:27 +08:00
|
|
|
for _, p := range m.module.BuildParamsForTests() {
|
|
|
|
if strings.Contains(p.Rule.String(), rule) {
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
}
|
2019-01-06 14:09:19 +08:00
|
|
|
return BuildParams{}
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
|
|
|
|
2019-01-06 14:09:19 +08:00
|
|
|
// Rule finds a call to ctx.Build with BuildParams.Rule set to a rule with the given name. Panics if no rule is found.
|
|
|
|
func (m TestingModule) Rule(rule string) BuildParams {
|
|
|
|
p := m.MaybeRule(rule)
|
|
|
|
if p.Rule == nil {
|
|
|
|
panic(fmt.Errorf("couldn't find rule %q", rule))
|
|
|
|
}
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
// MaybeDescription finds a call to ctx.Build with BuildParams.Description set to a the given string. Returns an empty
|
|
|
|
// BuildParams if no rule is found.
|
|
|
|
func (m TestingModule) MaybeDescription(desc string) BuildParams {
|
2017-10-20 04:06:22 +08:00
|
|
|
for _, p := range m.module.BuildParamsForTests() {
|
|
|
|
if p.Description == desc {
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
}
|
2019-01-06 14:09:19 +08:00
|
|
|
return BuildParams{}
|
2017-10-20 04:06:22 +08:00
|
|
|
}
|
|
|
|
|
2019-01-06 14:09:19 +08:00
|
|
|
// Description finds a call to ctx.Build with BuildParams.Description set to a the given string. Panics if no rule is
|
|
|
|
// found.
|
|
|
|
func (m TestingModule) Description(desc string) BuildParams {
|
|
|
|
p := m.MaybeDescription(desc)
|
|
|
|
if p.Rule == nil {
|
|
|
|
panic(fmt.Errorf("couldn't find description %q", desc))
|
|
|
|
}
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m TestingModule) maybeOutput(file string) (BuildParams, []string) {
|
2017-12-06 01:26:15 +08:00
|
|
|
var searchedOutputs []string
|
2017-07-14 05:43:27 +08:00
|
|
|
for _, p := range m.module.BuildParamsForTests() {
|
|
|
|
outputs := append(WritablePaths(nil), p.Outputs...)
|
|
|
|
if p.Output != nil {
|
|
|
|
outputs = append(outputs, p.Output)
|
|
|
|
}
|
|
|
|
for _, f := range outputs {
|
2017-12-01 12:13:19 +08:00
|
|
|
if f.String() == file || f.Rel() == file {
|
2019-01-06 14:09:19 +08:00
|
|
|
return p, nil
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
2017-12-06 01:26:15 +08:00
|
|
|
searchedOutputs = append(searchedOutputs, f.Rel())
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
|
|
|
}
|
2019-01-06 14:09:19 +08:00
|
|
|
return BuildParams{}, searchedOutputs
|
|
|
|
}
|
|
|
|
|
|
|
|
// MaybeOutput finds a call to ctx.Build with a BuildParams.Output or BuildParams.Outputspath whose String() or Rel()
|
|
|
|
// value matches the provided string. Returns an empty BuildParams if no rule is found.
|
|
|
|
func (m TestingModule) MaybeOutput(file string) BuildParams {
|
|
|
|
p, _ := m.maybeOutput(file)
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
// Output finds a call to ctx.Build with a BuildParams.Output or BuildParams.Outputspath whose String() or Rel()
|
|
|
|
// value matches the provided string. Panics if no rule is found.
|
|
|
|
func (m TestingModule) Output(file string) BuildParams {
|
|
|
|
p, searchedOutputs := m.maybeOutput(file)
|
|
|
|
if p.Rule == nil {
|
|
|
|
panic(fmt.Errorf("couldn't find output %q.\nall outputs: %v",
|
|
|
|
file, searchedOutputs))
|
|
|
|
}
|
|
|
|
return p
|
2017-07-14 05:43:27 +08:00
|
|
|
}
|
2018-03-12 16:29:17 +08:00
|
|
|
|
|
|
|
func FailIfErrored(t *testing.T, errs []error) {
|
|
|
|
t.Helper()
|
|
|
|
if len(errs) > 0 {
|
|
|
|
for _, err := range errs {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
}
|
2018-03-12 16:34:26 +08:00
|
|
|
|
|
|
|
func FailIfNoMatchingErrors(t *testing.T, pattern string, errs []error) {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
matcher, err := regexp.Compile(pattern)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("failed to compile regular expression %q because %s", pattern, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
found := false
|
|
|
|
for _, err := range errs {
|
|
|
|
if matcher.FindStringIndex(err.Error()) != nil {
|
|
|
|
found = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
t.Errorf("missing the expected error %q (checked %d error(s))", pattern, len(errs))
|
|
|
|
for i, err := range errs {
|
|
|
|
t.Errorf("errs[%d] = %s", i, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|