platform_build_soong/cc/cc_test_only_property_test.go
Ronald Braunstein 6a08d4915a Add test-only for test-per-src cc modules
When testing test-only validations, I noticed that some cc_test modules
where not reporting that they were test-only.

This happened because test-per-src variations were short-circuiting
writing out the provider.

The `all_teams` target is using the first variation for a module, but
it possibly should see if any variation is test-only.
For now, just making this fix to cc.

Test: m all_teams  && gqui from  "flatten(out/soong/ownership/all_teams.pb, teams)" proto build/soong/android/team_proto/team.proto:AllTeams ' where teams.kind = "cc_test" and teams.target_name="libnativebridge-tests"'
Test: go test ./cc
Change-Id: I6d44a521f5f2457527049399509d979559d7dc17
2024-04-15 23:21:23 +00:00

219 lines
7.2 KiB
Go

// Copyright 2024 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 cc
import (
"android/soong/android"
"android/soong/android/team_proto"
"log"
"strings"
"testing"
"github.com/google/blueprint"
"google.golang.org/protobuf/proto"
)
func TestTestOnlyProvider(t *testing.T) {
t.Parallel()
ctx := android.GroupFixturePreparers(
prepareForCcTest,
android.FixtureRegisterWithContext(func(ctx android.RegistrationContext) {
ctx.RegisterModuleType("cc_test_host", TestHostFactory)
}),
).RunTestWithBp(t, `
// These should be test-only
cc_fuzz { name: "cc-fuzz" }
cc_test { name: "cc-test", gtest:false }
cc_benchmark { name: "cc-benchmark" }
cc_library { name: "cc-library-forced",
test_only: true }
cc_test_library {name: "cc-test-library", gtest: false}
cc_test_host {name: "cc-test-host", gtest: false}
// These should not be.
cc_genrule { name: "cc_genrule", cmd: "echo foo", out: ["out"] }
cc_library { name: "cc_library" }
cc_library { name: "cc_library_false", test_only: false }
cc_library_static { name: "cc_static" }
cc_library_shared { name: "cc_library_shared" }
cc_object { name: "cc-object" }
`)
// Visit all modules and ensure only the ones that should
// marked as test-only are marked as test-only.
actualTestOnly := []string{}
ctx.VisitAllModules(func(m blueprint.Module) {
if provider, ok := android.OtherModuleProvider(ctx.TestContext.OtherModuleProviderAdaptor(), m, android.TestOnlyProviderKey); ok {
if provider.TestOnly {
actualTestOnly = append(actualTestOnly, m.Name())
}
}
})
expectedTestOnlyModules := []string{
"cc-test",
"cc-library-forced",
"cc-fuzz",
"cc-benchmark",
"cc-test-library",
"cc-test-host",
}
notEqual, left, right := android.ListSetDifference(expectedTestOnlyModules, actualTestOnly)
if notEqual {
t.Errorf("test-only: Expected but not found: %v, Found but not expected: %v", left, right)
}
}
func TestTestOnlyValueWithTestPerSrcProp(t *testing.T) {
t.Parallel()
ctx := android.GroupFixturePreparers(
prepareForCcTest,
).RunTestWithBp(t, `
// These should be test-only
cc_test { name: "cc-test",
gtest: false,
test_per_src: true,
srcs: ["foo_test.cpp"],
test_options: { unit_test: false, },
}
`)
// Ensure all variation of test-per-src tests are marked test-only.
ctx.VisitAllModules(func(m blueprint.Module) {
testOnly := false
if provider, ok := android.OtherModuleProvider(ctx.TestContext.OtherModuleProviderAdaptor(), m, android.TestOnlyProviderKey); ok {
if provider.TestOnly {
testOnly = true
}
}
if module, ok := m.(*Module); ok {
if testModule, ok := module.installer.(*testBinary); ok {
if !testOnly && *testModule.Properties.Test_per_src {
t.Errorf("%v is not test-only but should be", m)
}
}
}
})
}
func TestTestOnlyInTeamsProto(t *testing.T) {
t.Parallel()
ctx := android.GroupFixturePreparers(
android.PrepareForTestWithTeamBuildComponents,
prepareForCcTest,
android.FixtureRegisterWithContext(func(ctx android.RegistrationContext) {
ctx.RegisterParallelSingletonType("all_teams", android.AllTeamsFactory)
ctx.RegisterModuleType("cc_test_host", TestHostFactory)
}),
).RunTestWithBp(t, `
package { default_team: "someteam"}
// These should be test-only
cc_fuzz { name: "cc-fuzz" }
cc_test { name: "cc-test", gtest:false }
cc_benchmark { name: "cc-benchmark" }
cc_library { name: "cc-library-forced",
test_only: true }
cc_test_library {name: "cc-test-library", gtest: false}
cc_test_host {name: "cc-test-host", gtest: false}
// These should not be.
cc_genrule { name: "cc_genrule", cmd: "echo foo", out: ["out"] }
cc_library { name: "cc_library" }
cc_library_static { name: "cc_static" }
cc_library_shared { name: "cc_library_shared" }
cc_object { name: "cc-object" }
team {
name: "someteam",
trendy_team_id: "cool_team",
}
`)
var teams *team_proto.AllTeams
teams = getTeamProtoOutput(t, ctx)
// map of module name -> trendy team name.
actualTrueModules := []string{}
for _, teamProto := range teams.Teams {
if Bool(teamProto.TestOnly) {
actualTrueModules = append(actualTrueModules, teamProto.GetTargetName())
}
}
expectedTestOnlyModules := []string{
"cc-test",
"cc-library-forced",
"cc-fuzz",
"cc-benchmark",
"cc-test-library",
"cc-test-host",
}
notEqual, left, right := android.ListSetDifference(expectedTestOnlyModules, actualTrueModules)
if notEqual {
t.Errorf("test-only: Expected but not found: %v, Found but not expected: %v", left, right)
}
}
// Don't allow setting test-only on things that are always tests or never tests.
func TestInvalidTestOnlyTargets(t *testing.T) {
testCases := []string{
` cc_test { name: "cc-test", test_only: true, gtest: false, srcs: ["foo.cc"], } `,
` cc_binary { name: "cc-binary", test_only: true, srcs: ["foo.cc"], } `,
` cc_test_library {name: "cc-test-library", test_only: true, gtest: false} `,
` cc_test_host {name: "cc-test-host", test_only: true, gtest: false} `,
` cc_defaults {name: "cc-defaults", test_only: true} `,
}
for i, bp := range testCases {
ctx := android.GroupFixturePreparers(
prepareForCcTest,
android.FixtureRegisterWithContext(func(ctx android.RegistrationContext) {
ctx.RegisterModuleType("cc_test_host", TestHostFactory)
})).
ExtendWithErrorHandler(android.FixtureIgnoreErrors).
RunTestWithBp(t, bp)
if len(ctx.Errs) == 0 {
t.Errorf("Expected err setting test_only in testcase #%d", i)
}
if len(ctx.Errs) > 1 {
t.Errorf("Too many errs: [%s] %v", bp, ctx.Errs)
}
if len(ctx.Errs) == 1 {
if !strings.Contains(ctx.Errs[0].Error(), "unrecognized property \"test_only\"") {
t.Errorf("ERR: %s bad bp: %s", ctx.Errs[0], bp)
}
}
}
}
func getTeamProtoOutput(t *testing.T, ctx *android.TestResult) *team_proto.AllTeams {
teams := new(team_proto.AllTeams)
config := ctx.SingletonForTests("all_teams")
allOutputs := config.AllOutputs()
protoPath := allOutputs[0]
out := config.MaybeOutput(protoPath)
outProto := []byte(android.ContentFromFileRuleForTests(t, ctx.TestContext, out))
if err := proto.Unmarshal(outProto, teams); err != nil {
log.Fatalln("Failed to parse teams proto:", err)
}
return teams
}