blob: dca9b87fddc3dcc7f18d5a3a72e5664f9275020d [file] [log] [blame]
// Copyright 2015 The Bazel Authors. 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 com.google.devtools.build.lib.analysis;
import static com.google.common.truth.Truth.assertThat;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ListMultimap;
import com.google.common.testing.EqualsTester;
import com.google.common.testing.NullPointerTester;
import com.google.devtools.build.lib.analysis.DependencyResolver.Dependency;
import com.google.devtools.build.lib.analysis.config.BuildConfiguration;
import com.google.devtools.build.lib.analysis.config.ConfigMatchingProvider;
import com.google.devtools.build.lib.analysis.util.AnalysisTestCase;
import com.google.devtools.build.lib.analysis.util.TestAspects;
import com.google.devtools.build.lib.analysis.util.TestAspects.AspectRequiringRule;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.packages.AspectDefinition;
import com.google.devtools.build.lib.packages.AspectFactory;
import com.google.devtools.build.lib.packages.AspectParameters;
import com.google.devtools.build.lib.packages.Attribute;
import com.google.devtools.build.lib.packages.NoSuchThingException;
import com.google.devtools.build.lib.packages.Target;
import com.google.devtools.build.lib.testutil.TestRuleClassProvider;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import javax.annotation.Nullable;
/**
* Tests for {@link DependencyResolver}.
*
* <p>These use custom rules so that all usual and unusual cases related to aspect processing can
* be tested.
*
* <p>It would be nicer is we didn't have a Skyframe executor, if we didn't have that, we'd need a
* way to create a configuration, a package manager and a whole lot of other things, so it's just
* easier this way.
*/
@RunWith(JUnit4.class)
public class DependencyResolverTest extends AnalysisTestCase {
private DependencyResolver dependencyResolver;
@Override
@Before
public void setUp() throws Exception {
super.setUp();
dependencyResolver = new DependencyResolver() {
@Override
protected void invalidVisibilityReferenceHook(TargetAndConfiguration node, Label label) {
throw new IllegalStateException();
}
@Override
protected void invalidPackageGroupReferenceHook(TargetAndConfiguration node, Label label) {
throw new IllegalStateException();
}
@Nullable
@Override
protected Target getTarget(Label label) throws NoSuchThingException {
try {
return packageManager.getTarget(reporter, label);
} catch (InterruptedException e) {
throw new IllegalStateException(e);
}
}
};
}
@Override
@After
public void tearDown() throws Exception {
super.tearDown();
}
private void pkg(String name, String... contents) throws Exception {
scratch.file("" + name + "/BUILD", contents);
}
@SafeVarargs
private final void setRules(RuleDefinition... rules) throws Exception {
ConfiguredRuleClassProvider.Builder builder =
new ConfiguredRuleClassProvider.Builder();
TestRuleClassProvider.addStandardRules(builder);
for (RuleDefinition rule : rules) {
builder.addRuleDefinition(rule);
}
useRuleClassProvider(builder.build());
update();
}
private ListMultimap<Attribute, Dependency> dependentNodeMap(
String targetName, Class<? extends ConfiguredAspectFactory> aspect) throws Exception {
AspectDefinition aspectDefinition = aspect == null
? null
: AspectFactory.Util.create(aspect).getDefinition();
Target target = packageManager.getTarget(reporter, Label.parseAbsolute(targetName));
return dependencyResolver.dependentNodeMap(
new TargetAndConfiguration(target, getTargetConfiguration()),
getHostConfiguration(),
aspectDefinition,
AspectParameters.EMPTY,
ImmutableSet.<ConfigMatchingProvider>of());
}
@SafeVarargs
private final void assertDep(
ListMultimap<Attribute, Dependency> dependentNodeMap,
String attrName,
String dep,
AspectWithParameters... aspects) {
Attribute attr = null;
for (Attribute candidate : dependentNodeMap.keySet()) {
if (candidate.getName().equals(attrName)) {
attr = candidate;
break;
}
}
assertNotNull("Attribute '" + attrName + "' not found", attr);
Dependency dependency = null;
for (Dependency candidate : dependentNodeMap.get(attr)) {
if (candidate.getLabel().toString().equals(dep)) {
dependency = candidate;
break;
}
}
assertNotNull("Dependency '" + dep + "' on attribute '" + attrName + "' not found", dependency);
assertThat(dependency.getAspects()).containsExactly((Object[]) aspects);
}
@Test
public void hasAspectsRequiredByRule() throws Exception {
setRules(new AspectRequiringRule(), new TestAspects.BaseRule());
pkg("a",
"aspect(name='a', foo=[':b'])",
"aspect(name='b', foo=[])");
ListMultimap<Attribute, Dependency> map = dependentNodeMap("//a:a", null);
assertDep(map, "foo", "//a:b", new AspectWithParameters(TestAspects.SimpleAspect.class));
}
@Test
public void hasAspectsRequiredByAspect() throws Exception {
setRules(new TestAspects.BaseRule(), new TestAspects.SimpleRule());
pkg("a",
"simple(name='a', foo=[':b'])",
"simple(name='b', foo=[])");
ListMultimap<Attribute, Dependency> map =
dependentNodeMap("//a:a", TestAspects.AttributeAspect.class);
assertDep(map, "foo", "//a:b", new AspectWithParameters(TestAspects.AttributeAspect.class));
}
@Test
public void hasAspectDependencies() throws Exception {
setRules(new TestAspects.BaseRule());
pkg("a", "base(name='a')");
pkg("extra", "base(name='extra')");
ListMultimap<Attribute, Dependency> map =
dependentNodeMap("//a:a", TestAspects.ExtraAttributeAspect.class);
assertDep(map, "$dep", "//extra:extra");
}
@Test
public void constructorsForDependencyPassNullableTester() throws Exception {
update();
new NullPointerTester()
.setDefault(Label.class, Label.parseAbsolute("//a"))
.setDefault(BuildConfiguration.class, getTargetConfiguration())
.setDefault(ImmutableSet.class, ImmutableSet.of())
.testAllPublicConstructors(Dependency.class);
}
@Test
public void equalsOnDependencyPassesEqualsTester() throws Exception {
update();
Label a = Label.parseAbsolute("//a");
Label aExplicit = Label.parseAbsolute("//a:a");
Label b = Label.parseAbsolute("//b");
BuildConfiguration host = getHostConfiguration();
BuildConfiguration target = getTargetConfiguration();
ImmutableSet<AspectWithParameters> twoAspects = ImmutableSet.of(
new AspectWithParameters(TestAspects.SimpleAspect.class),
new AspectWithParameters(TestAspects.AttributeAspect.class));
ImmutableSet<AspectWithParameters> inverseAspects = ImmutableSet.of(
new AspectWithParameters(TestAspects.AttributeAspect.class),
new AspectWithParameters(TestAspects.SimpleAspect.class));
ImmutableSet<AspectWithParameters> differentAspects = ImmutableSet.of(
new AspectWithParameters(TestAspects.AttributeAspect.class),
new AspectWithParameters(TestAspects.ErrorAspect.class));
new EqualsTester()
.addEqualityGroup(
// base set: //a, host configuration, normal aspect set
new Dependency(a, host, twoAspects),
new Dependency(aExplicit, host, twoAspects),
new Dependency(a, host, inverseAspects),
new Dependency(aExplicit, host, inverseAspects))
.addEqualityGroup(
// base set but with label //b
new Dependency(b, host, twoAspects),
new Dependency(b, host, inverseAspects))
.addEqualityGroup(
// base set but with target configuration
new Dependency(a, target, twoAspects),
new Dependency(aExplicit, target, twoAspects),
new Dependency(a, target, inverseAspects),
new Dependency(aExplicit, target, inverseAspects))
.addEqualityGroup(
// base set but with null configuration
new Dependency(a, (BuildConfiguration) null, twoAspects),
new Dependency(aExplicit, (BuildConfiguration) null, twoAspects),
new Dependency(a, (BuildConfiguration) null, inverseAspects),
new Dependency(aExplicit, (BuildConfiguration) null, inverseAspects))
.addEqualityGroup(
// base set but with different aspects
new Dependency(a, host, differentAspects),
new Dependency(aExplicit, host, differentAspects))
.addEqualityGroup(
// base set but with label //b and target configuration
new Dependency(b, target, twoAspects),
new Dependency(b, target, inverseAspects))
.addEqualityGroup(
// base set but with label //b and null configuration
new Dependency(b, (BuildConfiguration) null, twoAspects),
new Dependency(b, (BuildConfiguration) null, inverseAspects))
.addEqualityGroup(
// base set but with label //b and different aspects
new Dependency(b, host, differentAspects))
.addEqualityGroup(
// base set but with target configuration and different aspects
new Dependency(a, target, differentAspects),
new Dependency(aExplicit, target, differentAspects))
.addEqualityGroup(
// base set but with null configuration and different aspects
new Dependency(a, (BuildConfiguration) null, differentAspects),
new Dependency(aExplicit, (BuildConfiguration) null, differentAspects))
.addEqualityGroup(
// inverse of base set: //b, target configuration, different aspects
new Dependency(b, target, differentAspects))
.addEqualityGroup(
// inverse of base set: //b, null configuration, different aspects
new Dependency(b, (BuildConfiguration) null, differentAspects))
.testEquals();
}
}