blob: 5deb1667d6cafa4740d87f46c3cedbc695317b76 [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.packages;
import static com.google.common.truth.Truth.assertThat;
import static com.google.common.truth.Truth.assertWithMessage;
import static com.google.devtools.build.lib.packages.Attribute.attr;
import static com.google.devtools.build.lib.packages.BuildType.LABEL;
import static com.google.devtools.build.lib.packages.BuildType.LABEL_LIST;
import static com.google.devtools.build.lib.packages.BuildType.OUTPUT_LIST;
import static com.google.devtools.build.lib.packages.ImplicitOutputsFunction.substitutePlaceholderIntoTemplate;
import static com.google.devtools.build.lib.packages.RuleClass.NO_EXTERNAL_BINDINGS;
import static com.google.devtools.build.lib.syntax.Type.BOOLEAN;
import static com.google.devtools.build.lib.syntax.Type.INTEGER;
import static com.google.devtools.build.lib.syntax.Type.STRING;
import static com.google.devtools.build.lib.syntax.Type.STRING_LIST;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertSame;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import com.google.common.base.Function;
import com.google.common.base.Predicate;
import com.google.common.base.Predicates;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import com.google.devtools.build.lib.analysis.config.BuildConfiguration;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.cmdline.LabelSyntaxException;
import com.google.devtools.build.lib.cmdline.PackageIdentifier;
import com.google.devtools.build.lib.events.Event;
import com.google.devtools.build.lib.events.EventCollector;
import com.google.devtools.build.lib.events.EventKind;
import com.google.devtools.build.lib.events.Location;
import com.google.devtools.build.lib.events.Location.LineAndColumn;
import com.google.devtools.build.lib.packages.Attribute.SkylarkComputedDefaultTemplate.CannotPrecomputeDefaultsException;
import com.google.devtools.build.lib.packages.Attribute.ValidityPredicate;
import com.google.devtools.build.lib.packages.ConfigurationFragmentPolicy.MissingFragmentPolicy;
import com.google.devtools.build.lib.packages.RuleClass.Builder.RuleClassType;
import com.google.devtools.build.lib.packages.RuleClass.Configurator;
import com.google.devtools.build.lib.packages.RuleClass.ConfiguredTargetFactory;
import com.google.devtools.build.lib.packages.RuleFactory.BuildLangTypedAttributeValuesMap;
import com.google.devtools.build.lib.packages.util.PackageLoadingTestCase;
import com.google.devtools.build.lib.syntax.BaseFunction;
import com.google.devtools.build.lib.syntax.Environment;
import com.google.devtools.build.lib.syntax.Type;
import com.google.devtools.build.lib.vfs.Path;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import javax.annotation.Nullable;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/**
* Tests for {@link RuleClass}.
*/
@RunWith(JUnit4.class)
public class RuleClassTest extends PackageLoadingTestCase {
private static final RuleClass.ConfiguredTargetFactory<Object, Object>
DUMMY_CONFIGURED_TARGET_FACTORY = new RuleClass.ConfiguredTargetFactory<Object, Object>() {
@Override
public Object create(Object ruleContext) throws InterruptedException {
throw new IllegalStateException();
}
};
private static final class DummyFragment extends BuildConfiguration.Fragment {
}
private static final Predicate<String> PREFERRED_DEPENDENCY_PREDICATE = Predicates.alwaysFalse();
private static RuleClass createRuleClassA() throws LabelSyntaxException {
return newRuleClass(
"ruleA",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.NONE,
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("my-string-attr", STRING).mandatory().build(),
attr("my-label-attr", LABEL)
.mandatory()
.legacyAllowAnyFileType()
.value(Label.parseAbsolute("//default:label"))
.build(),
attr("my-labellist-attr", LABEL_LIST).mandatory().legacyAllowAnyFileType().build(),
attr("my-integer-attr", INTEGER).value(42).build(),
attr("my-string-attr2", STRING).mandatory().value((String) null).build(),
attr("my-stringlist-attr", STRING_LIST).build(),
attr("my-sorted-stringlist-attr", STRING_LIST).orderIndependent().build());
}
private static RuleClass createRuleClassB(RuleClass ruleClassA) {
// emulates attribute inheritance
List<Attribute> attributes = new ArrayList<>(ruleClassA.getAttributes());
attributes.add(attr("another-string-attr", STRING).mandatory().build());
return newRuleClass(
"ruleB",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.NONE,
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attributes.toArray(new Attribute[0]));
}
@Test
public void testRuleClassBasics() throws Exception {
RuleClass ruleClassA = createRuleClassA();
assertEquals("ruleA", ruleClassA.getName());
assertEquals(7, ruleClassA.getAttributeCount());
assertEquals(0, (int) ruleClassA.getAttributeIndex("my-string-attr"));
assertEquals(1, (int) ruleClassA.getAttributeIndex("my-label-attr"));
assertEquals(2, (int) ruleClassA.getAttributeIndex("my-labellist-attr"));
assertEquals(3, (int) ruleClassA.getAttributeIndex("my-integer-attr"));
assertEquals(4, (int) ruleClassA.getAttributeIndex("my-string-attr2"));
assertEquals(5, (int) ruleClassA.getAttributeIndex("my-stringlist-attr"));
assertEquals(6, (int) ruleClassA.getAttributeIndex("my-sorted-stringlist-attr"));
assertEquals(ruleClassA.getAttribute(0),
ruleClassA.getAttributeByName("my-string-attr"));
assertEquals(ruleClassA.getAttribute(1),
ruleClassA.getAttributeByName("my-label-attr"));
assertEquals(ruleClassA.getAttribute(2),
ruleClassA.getAttributeByName("my-labellist-attr"));
assertEquals(ruleClassA.getAttribute(3),
ruleClassA.getAttributeByName("my-integer-attr"));
assertEquals(ruleClassA.getAttribute(4),
ruleClassA.getAttributeByName("my-string-attr2"));
assertEquals(ruleClassA.getAttribute(5),
ruleClassA.getAttributeByName("my-stringlist-attr"));
assertEquals(ruleClassA.getAttribute(6),
ruleClassA.getAttributeByName("my-sorted-stringlist-attr"));
assertEquals("", // default based on type
ruleClassA.getAttribute(0).getDefaultValue(null));
assertEquals(Label.parseAbsolute("//default:label"),
ruleClassA.getAttribute(1).getDefaultValue(null));
assertEquals(Collections.emptyList(),
ruleClassA.getAttribute(2).getDefaultValue(null));
assertEquals(42,
ruleClassA.getAttribute(3).getDefaultValue(null));
assertEquals(null, // default explicitly specified
ruleClassA.getAttribute(4).getDefaultValue(null));
assertEquals(Collections.emptyList(),
ruleClassA.getAttribute(5).getDefaultValue(null));
assertEquals(Collections.emptyList(),
ruleClassA.getAttribute(6).getDefaultValue(null));
}
@Test
public void testRuleClassInheritance() throws Exception {
RuleClass ruleClassA = createRuleClassA();
RuleClass ruleClassB = createRuleClassB(ruleClassA);
assertEquals("ruleB", ruleClassB.getName());
assertEquals(8, ruleClassB.getAttributeCount());
assertEquals(0, (int) ruleClassB.getAttributeIndex("my-string-attr"));
assertEquals(1, (int) ruleClassB.getAttributeIndex("my-label-attr"));
assertEquals(2, (int) ruleClassB.getAttributeIndex("my-labellist-attr"));
assertEquals(3, (int) ruleClassB.getAttributeIndex("my-integer-attr"));
assertEquals(4, (int) ruleClassB.getAttributeIndex("my-string-attr2"));
assertEquals(5, (int) ruleClassB.getAttributeIndex("my-stringlist-attr"));
assertEquals(6, (int) ruleClassB.getAttributeIndex("my-sorted-stringlist-attr"));
assertEquals(7, (int) ruleClassB.getAttributeIndex("another-string-attr"));
assertEquals(ruleClassB.getAttribute(0),
ruleClassB.getAttributeByName("my-string-attr"));
assertEquals(ruleClassB.getAttribute(1),
ruleClassB.getAttributeByName("my-label-attr"));
assertEquals(ruleClassB.getAttribute(2),
ruleClassB.getAttributeByName("my-labellist-attr"));
assertEquals(ruleClassB.getAttribute(3),
ruleClassB.getAttributeByName("my-integer-attr"));
assertEquals(ruleClassB.getAttribute(4),
ruleClassB.getAttributeByName("my-string-attr2"));
assertEquals(ruleClassB.getAttribute(5),
ruleClassB.getAttributeByName("my-stringlist-attr"));
assertEquals(ruleClassB.getAttribute(6),
ruleClassB.getAttributeByName("my-sorted-stringlist-attr"));
assertEquals(ruleClassB.getAttribute(7),
ruleClassB.getAttributeByName("another-string-attr"));
}
private static final String TEST_PACKAGE_NAME = "testpackage";
private static final String TEST_RULE_NAME = "my-rule-A";
private static final int TEST_RULE_DEFINED_AT_LINE = 42;
private static final String TEST_RULE_LABEL = "@//" + TEST_PACKAGE_NAME + ":" + TEST_RULE_NAME;
private Path testBuildfilePath;
private Location testRuleLocation;
@Before
public final void setRuleLocation() throws Exception {
testBuildfilePath = scratch.resolve("/home/user/workspace/testpackage/BUILD");
testRuleLocation = Location.fromPathAndStartColumn(
testBuildfilePath.asFragment(), 0, 0, new LineAndColumn(TEST_RULE_DEFINED_AT_LINE, 0));
}
private Package.Builder createDummyPackageBuilder() {
return packageFactory.newPackageBuilder(
PackageIdentifier.createInMainRepo(TEST_PACKAGE_NAME), "TESTING")
.setFilename(testBuildfilePath)
.setMakeEnv(new MakeEnvironment.Builder());
}
@Test
public void testDuplicatedDeps() throws Exception {
RuleClass depsRuleClass =
newRuleClass(
"ruleDeps",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.NONE,
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("list1", LABEL_LIST).mandatory().legacyAllowAnyFileType().build(),
attr("list2", LABEL_LIST).mandatory().legacyAllowAnyFileType().build(),
attr("list3", LABEL_LIST).mandatory().legacyAllowAnyFileType().build());
// LinkedHashMap -> predictable iteration order for testing
Map<String, Object> attributeValues = new LinkedHashMap<>();
attributeValues.put("list1", Lists.newArrayList("//testpackage:dup1", ":dup1", ":nodup"));
attributeValues.put("list2", Lists.newArrayList(":nodup1", ":nodup2"));
attributeValues.put("list3", Lists.newArrayList(":dup1", ":dup1", ":dup2", ":dup2"));
reporter.removeHandler(failFastHandler);
createRule(depsRuleClass, "depsRule", attributeValues, testRuleLocation);
assertSame(3, eventCollector.count());
assertDupError("//testpackage:dup1", "list1", "depsRule");
assertDupError("//testpackage:dup1", "list3", "depsRule");
assertDupError("//testpackage:dup2", "list3", "depsRule");
}
private void assertDupError(String label, String attrName, String ruleName) {
assertContainsEvent(String.format("Label '%s' is duplicated in the '%s' attribute of rule '%s'",
label, attrName, ruleName));
}
@Test
public void testCreateRuleWithLegacyPublicVisibility() throws Exception {
RuleClass ruleClass =
newRuleClass(
"ruleVis",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.NONE,
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("visibility", LABEL_LIST).legacyAllowAnyFileType().build());
Map<String, Object> attributeValues = new HashMap<>();
attributeValues.put("visibility", Arrays.asList("//visibility:legacy_public"));
reporter.removeHandler(failFastHandler);
EventCollector collector = new EventCollector(EventKind.ERRORS);
reporter.addHandler(collector);
createRule(ruleClass, TEST_RULE_NAME, attributeValues, testRuleLocation);
assertContainsEvent("//visibility:legacy_public only allowed in package declaration");
}
@Test
public void testCreateRule() throws Exception {
RuleClass ruleClassA = createRuleClassA();
// LinkedHashMap -> predictable iteration order for testing
Map<String, Object> attributeValues = new LinkedHashMap<>();
attributeValues.put("my-labellist-attr", "foobar"); // wrong type
attributeValues.put("bogus-attr", "foobar"); // no such attr
attributeValues.put("my-stringlist-attr", Arrays.asList("foo", "bar"));
reporter.removeHandler(failFastHandler);
EventCollector collector = new EventCollector(EventKind.ERRORS);
reporter.addHandler(collector);
Rule rule = createRule(ruleClassA, TEST_RULE_NAME, attributeValues, testRuleLocation);
// TODO(blaze-team): (2009) refactor to use assertContainsEvent
Iterator<String> expectedMessages = Arrays.asList(
"expected value of type 'list(label)' for attribute 'my-labellist-attr' "
+ "in 'ruleA' rule, but got \"foobar\" (string)",
"no such attribute 'bogus-attr' in 'ruleA' rule",
"missing value for mandatory "
+ "attribute 'my-string-attr' in 'ruleA' rule",
"missing value for mandatory attribute 'my-label-attr' in 'ruleA' rule",
"missing value for mandatory "
+ "attribute 'my-labellist-attr' in 'ruleA' rule",
"missing value for mandatory "
+ "attribute 'my-string-attr2' in 'ruleA' rule"
).iterator();
for (Event event : collector) {
assertEquals(TEST_RULE_DEFINED_AT_LINE,
event.getLocation().getStartLineAndColumn().getLine());
assertEquals(testBuildfilePath.asFragment(), event.getLocation().getPath());
assertEquals(TEST_RULE_LABEL.toString().substring(1)
+ ": " + expectedMessages.next(), event.getMessage());
}
// Test basic rule properties:
assertEquals("ruleA", rule.getRuleClass());
assertEquals(TEST_RULE_NAME, rule.getName());
assertEquals(TEST_RULE_LABEL.substring(1), rule.getLabel().toString());
// Test attribute access:
AttributeMap attributes = RawAttributeMapper.of(rule);
assertEquals("//default:label",
attributes.get("my-label-attr", BuildType.LABEL).toString());
assertEquals(42,
attributes.get("my-integer-attr", Type.INTEGER).intValue());
assertEquals("", // missing attribute -> default chosen based on type
attributes.get("my-string-attr", Type.STRING));
assertThat(attributes.get("my-labellist-attr", BuildType.LABEL_LIST)).isEmpty();
assertEquals(Arrays.asList("foo", "bar"),
attributes.get("my-stringlist-attr", Type.STRING_LIST));
try {
attributes.get("my-labellist-attr", Type.STRING); // wrong type
fail();
} catch (IllegalArgumentException e) {
assertThat(e).hasMessage("Attribute my-labellist-attr is of type list(label) "
+ "and not of type string in rule //testpackage:my-rule-A");
}
}
@Test
public void testImplicitOutputs() throws Exception {
RuleClass ruleClassC =
newRuleClass(
"ruleC",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.fromTemplates(
"foo-%{name}.bar", "lib%{name}-wazoo-%{name}.mumble", "stuff-%{outs}-bar"),
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("outs", OUTPUT_LIST).build());
Map<String, Object> attributeValues = new HashMap<>();
attributeValues.put("outs", Collections.singletonList("explicit_out"));
Rule rule = createRule(ruleClassC, "myrule", attributeValues, testRuleLocation);
Set<String> set = new HashSet<>();
for (OutputFile outputFile : rule.getOutputFiles()) {
set.add(outputFile.getName());
assertSame(rule, outputFile.getGeneratingRule());
}
assertThat(set).containsExactly("foo-myrule.bar", "libmyrule-wazoo-myrule.mumble",
"stuff-explicit_out-bar", "explicit_out");
}
@Test
public void testImplicitOutsWithBasenameDirname() throws Exception {
RuleClass ruleClass =
newRuleClass(
"ruleClass",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.fromTemplates("%{dirname}lib%{basename}.bar"),
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true);
Rule rule = createRule(ruleClass, "myRule", Collections.<String, Object>emptyMap(),
testRuleLocation);
assertEquals("libmyRule.bar", Iterables.getOnlyElement(rule.getOutputFiles()).getName());
Rule ruleWithSlash = createRule(ruleClass, "myRule/with/slash",
Collections.<String, Object>emptyMap(), testRuleLocation);
assertEquals("myRule/with/libslash.bar",
Iterables.getOnlyElement(ruleWithSlash.getOutputFiles()).getName());
}
/**
* Helper routine that instantiates a rule class with the given computed default and supporting
* attributes for the default to reference.
*/
private static RuleClass getRuleClassWithComputedDefault(Attribute computedDefault) {
return newRuleClass(
"ruleClass",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.fromTemplates("empty"),
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("condition", BOOLEAN).value(false).build(),
attr("declared1", BOOLEAN).value(false).build(),
attr("declared2", BOOLEAN).value(false).build(),
attr("nonconfigurable", BOOLEAN).nonconfigurable("test").value(false).build(),
computedDefault);
}
/**
* Helper routine that checks that a computed default is valid and bound to the expected value.
*/
private void checkValidComputedDefault(Object expectedValue, Attribute computedDefault,
ImmutableMap<String, Object> attrValueMap) throws Exception {
assertTrue(computedDefault.getDefaultValueForTesting() instanceof Attribute.ComputedDefault);
Rule rule = createRule(getRuleClassWithComputedDefault(computedDefault), "myRule",
attrValueMap, testRuleLocation);
AttributeMap attributes = RawAttributeMapper.of(rule);
assertEquals(
expectedValue,
attributes.get(computedDefault.getName(), computedDefault.getType()));
}
/**
* Helper routine that checks that a computed default is invalid due to declared dependency
* issues and fails with the expected message.
*/
private void checkInvalidComputedDefault(Attribute computedDefault, String expectedMessage)
throws Exception {
try {
createRule(getRuleClassWithComputedDefault(computedDefault), "myRule",
ImmutableMap.<String, Object>of(), testRuleLocation);
fail("Expected computed default \"" + computedDefault.getName() + "\" to fail with "
+ "declaration errors");
} catch (IllegalArgumentException e) {
// Expected outcome.
assertThat(e).hasMessage(expectedMessage);
}
}
/**
* Tests computed default values are computed as expected.
*/
@Test
public void testComputedDefault() throws Exception {
Attribute computedDefault =
attr("$result", BOOLEAN).value(new Attribute.ComputedDefault("condition") {
@Override
public Object getDefault(AttributeMap rule) {
return rule.get("condition", Type.BOOLEAN);
}
}).build();
checkValidComputedDefault(Boolean.FALSE, computedDefault,
ImmutableMap.<String, Object>of("condition", Boolean.FALSE));
checkValidComputedDefault(Boolean.TRUE, computedDefault,
ImmutableMap.<String, Object>of("condition", Boolean.TRUE));
}
/**
* Tests that computed defaults can only read attribute values for configurable attributes that
* have been explicitly declared.
*/
@Test
public void testComputedDefaultDeclarations() throws Exception {
checkValidComputedDefault(
Boolean.FALSE,
attr("$good_default_no_declares", BOOLEAN).value(
new Attribute.ComputedDefault() {
@Override public Object getDefault(AttributeMap rule) {
// OK: not a value check:
return rule.isAttributeValueExplicitlySpecified("undeclared");
}
}).build(),
ImmutableMap.<String, Object>of());
checkValidComputedDefault(
Boolean.FALSE,
attr("$good_default_one_declare", BOOLEAN).value(
new Attribute.ComputedDefault("declared1") {
@Override public Object getDefault(AttributeMap rule) {
return rule.get("declared1", Type.BOOLEAN);
}
}).build(),
ImmutableMap.<String, Object>of());
checkValidComputedDefault(
Boolean.FALSE,
attr("$good_default_two_declares", BOOLEAN).value(
new Attribute.ComputedDefault("declared1", "declared2") {
@Override public Object getDefault(AttributeMap rule) {
return rule.get("declared1", Type.BOOLEAN) && rule.get("declared2", Type.BOOLEAN);
}
}).build(),
ImmutableMap.<String, Object>of());
checkInvalidComputedDefault(
attr("$bad_default_no_declares", BOOLEAN).value(
new Attribute.ComputedDefault() {
@Override public Object getDefault(AttributeMap rule) {
return rule.get("declared1", Type.BOOLEAN);
}
}).build(),
"attribute \"declared1\" isn't available in this computed default context");
checkInvalidComputedDefault(
attr("$bad_default_one_declare", BOOLEAN).value(
new Attribute.ComputedDefault("declared1") {
@Override public Object getDefault(AttributeMap rule) {
return rule.get("declared1", Type.BOOLEAN) || rule.get("declared2", Type.BOOLEAN);
}
}).build(),
"attribute \"declared2\" isn't available in this computed default context");
checkInvalidComputedDefault(
attr("$bad_default_two_declares", BOOLEAN).value(
new Attribute.ComputedDefault("declared1", "declared2") {
@Override public Object getDefault(AttributeMap rule) {
return rule.get("condition", Type.BOOLEAN);
}
}).build(),
"attribute \"condition\" isn't available in this computed default context");
}
/**
* Tests that computed defaults *can* read attribute values for non-configurable attributes
* without needing to explicitly declare them.
*/
@Test
public void testComputedDefaultWithNonConfigurableAttributes() throws Exception {
checkValidComputedDefault(
Boolean.FALSE,
attr("$good_default_reading_undeclared_nonconfigurable_attribute", BOOLEAN).value(
new Attribute.ComputedDefault() {
@Override public Object getDefault(AttributeMap rule) {
return rule.get("nonconfigurable", Type.BOOLEAN);
}
}).build(),
ImmutableMap.<String, Object>of());
}
@Test
public void testOutputsAreOrdered() throws Exception {
RuleClass ruleClassC =
newRuleClass(
"ruleC",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.fromTemplates("first-%{name}", "second-%{name}", "out-%{outs}"),
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("outs", OUTPUT_LIST).build());
Map<String, Object> attributeValues = new HashMap<>();
attributeValues.put("outs", ImmutableList.of("third", "fourth"));
Rule rule = createRule(ruleClassC, "myrule", attributeValues, testRuleLocation);
List<String> actual = new ArrayList<>();
for (OutputFile outputFile : rule.getOutputFiles()) {
actual.add(outputFile.getName());
assertSame(rule, outputFile.getGeneratingRule());
}
assertWithMessage("unexpected output set").that(actual).containsExactly("first-myrule",
"second-myrule", "out-third", "out-fourth", "third", "fourth");
assertWithMessage("invalid output ordering").that(actual).containsExactly("first-myrule",
"second-myrule", "out-third", "out-fourth", "third", "fourth").inOrder();
}
@Test
public void testSubstitutePlaceholderIntoTemplate() throws Exception {
RuleClass ruleClass =
newRuleClass(
"ruleA",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.NONE,
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("a", STRING_LIST).mandatory().build(),
attr("b", STRING_LIST).mandatory().build(),
attr("c", STRING_LIST).mandatory().build(),
attr("baz", STRING_LIST).mandatory().build(),
attr("empty", STRING_LIST).build());
Map<String, Object> attributeValues = new LinkedHashMap<>();
attributeValues.put("a", ImmutableList.of("a", "A"));
attributeValues.put("b", ImmutableList.of("b", "B"));
attributeValues.put("c", ImmutableList.of("c", "C"));
attributeValues.put("baz", ImmutableList.of("baz", "BAZ"));
attributeValues.put("empty", ImmutableList.<String>of());
AttributeMap rule = RawAttributeMapper.of(
createRule(ruleClass, "testrule", attributeValues, testRuleLocation));
assertThat(substitutePlaceholderIntoTemplate("foo", rule)).containsExactly("foo");
assertThat(substitutePlaceholderIntoTemplate("foo-%{baz}-bar", rule)).containsExactly(
"foo-baz-bar", "foo-BAZ-bar").inOrder();
assertThat(substitutePlaceholderIntoTemplate("%{a}-%{b}-%{c}", rule)).containsExactly("a-b-c",
"a-b-C", "a-B-c", "a-B-C", "A-b-c", "A-b-C", "A-B-c", "A-B-C").inOrder();
assertThat(substitutePlaceholderIntoTemplate("%{a", rule)).containsExactly("%{a");
assertThat(substitutePlaceholderIntoTemplate("%{a}}", rule)).containsExactly("a}", "A}")
.inOrder();
assertThat(substitutePlaceholderIntoTemplate("x%{a}y%{empty}", rule)).isEmpty();
}
@Test
public void testOrderIndependentAttribute() throws Exception {
RuleClass ruleClassA = createRuleClassA();
List<String> list = Arrays.asList("foo", "bar", "baz");
Map<String, Object> attributeValues = new LinkedHashMap<>();
// mandatory values
attributeValues.put("my-string-attr", "");
attributeValues.put("my-label-attr", "//project");
attributeValues.put("my-string-attr2", "");
attributeValues.put("my-labellist-attr", Collections.emptyList());
// to compare the effect of .orderIndependent()
attributeValues.put("my-stringlist-attr", list);
attributeValues.put("my-sorted-stringlist-attr", list);
Rule rule = createRule(ruleClassA, "testrule", attributeValues, testRuleLocation);
AttributeMap attributes = RawAttributeMapper.of(rule);
assertEquals(list,
attributes.get("my-stringlist-attr", Type.STRING_LIST));
assertEquals(Arrays.asList("bar", "baz", "foo"),
attributes.get("my-sorted-stringlist-attr", Type.STRING_LIST));
}
private Rule createRule(
RuleClass ruleClass, String name, Map<String, Object> attributeValues, Location location)
throws LabelSyntaxException, InterruptedException, CannotPrecomputeDefaultsException {
Package.Builder pkgBuilder = createDummyPackageBuilder();
Label ruleLabel;
try {
ruleLabel = pkgBuilder.createLabel(name);
} catch (LabelSyntaxException e) {
throw new IllegalArgumentException("Rule has illegal label");
}
return ruleClass.createRule(
pkgBuilder,
ruleLabel,
new BuildLangTypedAttributeValuesMap(attributeValues),
reporter,
/*ast=*/ null,
location,
new AttributeContainer(ruleClass));
}
@Test
public void testOverrideWithWrongType() {
try {
RuleClass parentRuleClass = createParentRuleClass();
RuleClass.Builder childRuleClassBuilder = new RuleClass.Builder(
"child_rule", RuleClassType.NORMAL, false, parentRuleClass);
childRuleClassBuilder.override(attr("attr", INTEGER));
fail();
} catch (IllegalStateException e) {
assertThat(e).hasMessage("The type of the new attribute 'int' is different from "
+ "the original one 'string'.");
}
}
@Test
public void testOverrideWithRightType() {
RuleClass parentRuleClass = createParentRuleClass();
RuleClass.Builder childRuleClassBuilder = new RuleClass.Builder(
"child_rule", RuleClassType.NORMAL, false, parentRuleClass);
childRuleClassBuilder.override(attr("attr", STRING));
}
@Test
public void testCopyAndOverrideAttribute() throws Exception {
RuleClass parentRuleClass = createParentRuleClass();
RuleClass childRuleClass = createChildRuleClass(parentRuleClass);
Map<String, Object> parentValues = new LinkedHashMap<>();
Map<String, Object> childValues = new LinkedHashMap<>();
childValues.put("attr", "somevalue");
createRule(parentRuleClass, "parent_rule", parentValues, testRuleLocation);
createRule(childRuleClass, "child_rule", childValues, testRuleLocation);
}
@Test
public void testCopyAndOverrideAttributeMandatoryMissing() throws Exception {
RuleClass parentRuleClass = createParentRuleClass();
RuleClass childRuleClass = createChildRuleClass(parentRuleClass);
Map<String, Object> childValues = new LinkedHashMap<>();
reporter.removeHandler(failFastHandler);
createRule(childRuleClass, "child_rule", childValues, testRuleLocation);
assertSame(1, eventCollector.count());
assertContainsEvent("//testpackage:child_rule: missing value for mandatory "
+ "attribute 'attr' in 'child_rule' rule");
}
@Test
public void testRequiredFragmentInheritance() throws Exception {
RuleClass parentRuleClass = createParentRuleClass();
RuleClass childRuleClass = createChildRuleClass(parentRuleClass);
assertThat(parentRuleClass.getConfigurationFragmentPolicy().getRequiredConfigurationFragments())
.containsExactly(DummyFragment.class);
assertThat(childRuleClass.getConfigurationFragmentPolicy().getRequiredConfigurationFragments())
.containsExactly(DummyFragment.class);
}
private static RuleClass newRuleClass(
String name,
boolean skylarkExecutable,
boolean documented,
boolean publicByDefault,
boolean binaryOutput,
boolean workspaceOnly,
boolean outputsDefaultExecutable,
ImplicitOutputsFunction implicitOutputsFunction,
Configurator<?, ?> configurator,
ConfiguredTargetFactory<?, ?> configuredTargetFactory,
PredicateWithMessage<Rule> validityPredicate,
Predicate<String> preferredDependencyPredicate,
ImmutableSet<Class<?>> advertisedProviders,
boolean canHaveAnyProvider,
@Nullable BaseFunction configuredTargetFunction,
Function<? super Rule, Map<String, Label>> externalBindingsFunction,
@Nullable Environment ruleDefinitionEnvironment,
Set<Class<?>> allowedConfigurationFragments,
MissingFragmentPolicy missingFragmentPolicy,
boolean supportsConstraintChecking,
Attribute... attributes) {
String ruleDefinitionEnvironmentHashCode =
ruleDefinitionEnvironment == null
? null
: ruleDefinitionEnvironment.getTransitiveContentHashCode();
return new RuleClass(
name,
/*isSkylark=*/ skylarkExecutable,
skylarkExecutable,
/*skylarkTestable=*/ false,
documented,
publicByDefault,
binaryOutput,
workspaceOnly,
outputsDefaultExecutable,
implicitOutputsFunction,
configurator,
configuredTargetFactory,
validityPredicate,
preferredDependencyPredicate,
advertisedProviders,
canHaveAnyProvider,
configuredTargetFunction,
externalBindingsFunction,
ruleDefinitionEnvironment,
ruleDefinitionEnvironmentHashCode,
new ConfigurationFragmentPolicy.Builder()
.requiresConfigurationFragments(allowedConfigurationFragments)
.setMissingFragmentPolicy(missingFragmentPolicy)
.build(),
supportsConstraintChecking,
attributes);
}
private static RuleClass createParentRuleClass() {
return newRuleClass(
"parent_rule",
false,
false,
false,
false,
false,
false,
ImplicitOutputsFunction.NONE,
RuleClass.NO_CHANGE,
DUMMY_CONFIGURED_TARGET_FACTORY,
PredicatesWithMessage.<Rule>alwaysTrue(),
PREFERRED_DEPENDENCY_PREDICATE,
ImmutableSet.<Class<?>>of(),
false,
null,
NO_EXTERNAL_BINDINGS,
null,
ImmutableSet.<Class<?>>of(DummyFragment.class),
MissingFragmentPolicy.FAIL_ANALYSIS,
true,
attr("attr", STRING).build());
}
private static RuleClass createChildRuleClass(RuleClass parentRuleClass) {
RuleClass.Builder childRuleClassBuilder = new RuleClass.Builder(
"child_rule", RuleClassType.NORMAL, false, parentRuleClass);
return childRuleClassBuilder.override(
childRuleClassBuilder
.factory(DUMMY_CONFIGURED_TARGET_FACTORY)
.copy("attr").mandatory())
.add(attr("tags", STRING_LIST))
.build();
}
@Test
public void testValidityChecker() throws Exception {
RuleClass depClass = new RuleClass.Builder("dep", RuleClassType.NORMAL, false)
.factory(DUMMY_CONFIGURED_TARGET_FACTORY)
.add(attr("tags", STRING_LIST))
.build();
final Rule dep1 = createRule(depClass, "dep1", Collections.<String, Object>emptyMap(),
testRuleLocation);
final Rule dep2 = createRule(depClass, "dep2", Collections.<String, Object>emptyMap(),
testRuleLocation);
ValidityPredicate checker = new ValidityPredicate() {
@Override
public String checkValid(Rule from, Rule to) {
assertEquals("top", from.getName());
if (to.getName().equals("dep1")) {
return "pear";
} else if (to.getName().equals("dep2")) {
return null;
} else {
fail("invalid dependency");
return null;
}
}
};
RuleClass topClass = new RuleClass.Builder("top", RuleClassType.NORMAL, false)
.factory(DUMMY_CONFIGURED_TARGET_FACTORY)
.add(attr("tags", STRING_LIST))
.add(attr("deps", LABEL_LIST).legacyAllowAnyFileType()
.validityPredicate(checker))
.build();
Rule topRule = createRule(topClass, "top", Collections.<String, Object>emptyMap(),
testRuleLocation);
assertEquals("pear", topClass.getAttributeByName("deps").getValidityPredicate().checkValid(
topRule, dep1));
assertEquals(null, topClass.getAttributeByName("deps").getValidityPredicate().checkValid(
topRule, dep2));
}
/**
* Tests structure for making certain rules "preferential choices" for certain files
* under --compile_one_dependency.
*/
@Test
public void testPreferredDependencyChecker() throws Exception {
final String cppFile = "file.cc";
final String textFile = "file.txt";
// Default: not preferred for anything.
RuleClass defaultClass = new RuleClass.Builder("defaultClass", RuleClassType.NORMAL, false)
.factory(DUMMY_CONFIGURED_TARGET_FACTORY)
.add(attr("tags", STRING_LIST))
.build();
final Rule defaultRule = createRule(defaultClass, "defaultRule",
Collections.<String, Object>emptyMap(), testRuleLocation);
assertFalse(defaultRule.getRuleClassObject().isPreferredDependency(cppFile));
assertFalse(defaultRule.getRuleClassObject().isPreferredDependency(textFile));
// Make a rule that's preferred for C++ sources.
RuleClass cppClass = new RuleClass.Builder("cppClass", RuleClassType.NORMAL, false)
.factory(DUMMY_CONFIGURED_TARGET_FACTORY)
.add(attr("tags", STRING_LIST))
.setPreferredDependencyPredicate(new Predicate<String>() {
@Override
public boolean apply(String filename) {
return filename.endsWith(".cc");
}
})
.build();
final Rule cppRule = createRule(cppClass, "cppRule",
Collections.<String, Object>emptyMap(), testRuleLocation);
assertTrue(cppRule.getRuleClassObject().isPreferredDependency(cppFile));
assertFalse(cppRule.getRuleClassObject().isPreferredDependency(textFile));
}
@Test
public void testBadRuleClassNames() {
expectError(RuleClassType.NORMAL, "8abc");
expectError(RuleClassType.NORMAL, "!abc");
expectError(RuleClassType.NORMAL, "a b");
}
private void expectError(RuleClassType type, String name) {
try {
type.checkName(name);
fail();
} catch (IllegalArgumentException expected) {
// expected
}
}
}