// Copyright 2014 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 com.google.common.annotations.VisibleForTesting; import com.google.common.collect.ImmutableList; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.events.Location; import com.google.devtools.build.lib.packages.BuildType.SelectorList; import com.google.devtools.build.lib.syntax.Type; import java.util.ArrayList; import java.util.Collection; import java.util.List; import javax.annotation.Nullable; /** * Base {@link AttributeMap} implementation providing direct, unmanipulated access to * underlying attribute data as stored within the Rule. * *

Any instantiable subclass should define a clear policy of what it does with this * data before exposing it to consumers. */ public abstract class AbstractAttributeMapper implements AttributeMap { private final Package pkg; private final RuleClass ruleClass; private final Label ruleLabel; private final AttributeContainer attributes; public AbstractAttributeMapper(Package pkg, RuleClass ruleClass, Label ruleLabel, AttributeContainer attributes) { this.pkg = pkg; this.ruleClass = ruleClass; this.ruleLabel = ruleLabel; this.attributes = attributes; } @Override public String getName() { return ruleLabel.getName(); } @Override public Label getLabel() { return ruleLabel; } @Override public String getRuleClassName() { return ruleClass.getName(); } @Nullable @Override public T get(String attributeName, Type type) { int index = getIndexWithTypeCheck(attributeName, type); Object value = attributes.getAttributeValue(index); if (value instanceof Attribute.ComputedDefault) { value = ((Attribute.ComputedDefault) value).getDefault(this); } else if (value instanceof Attribute.LateBoundDefault) { value = ((Attribute.LateBoundDefault) value).getDefault(); } try { return type.cast(value); } catch (ClassCastException e) { // getIndexWithTypeCheck checks the type is right, but unexpected configurable attributes // can still trigger cast exceptions. throw new IllegalArgumentException("wrong type for attribute \"" + attributeName + "\" in " + ruleClass + " rule " + ruleLabel, e); } } /** * Returns the given attribute if it's a computed default, null otherwise. * * @throws IllegalArgumentException if the given attribute doesn't exist with the specified * type. This happens whether or not it's a computed default. */ @VisibleForTesting // Should be protected @Nullable public Attribute.ComputedDefault getComputedDefault(String attributeName, Type type) { int index = getIndexWithTypeCheck(attributeName, type); Object value = attributes.getAttributeValue(index); if (value instanceof Attribute.ComputedDefault) { return (Attribute.ComputedDefault) value; } else { return null; } } /** * Returns the given attribute if it's a {@link Attribute.LateBoundDefault}, null otherwise. * * @throws IllegalArgumentException if the given attribute doesn't exist with the specified * type. This happens whether or not it's a late bound default. */ @Nullable @SuppressWarnings("unchecked") public Attribute.LateBoundDefault getLateBoundDefault( String attributeName, Type type) { int index = getIndexWithTypeCheck(attributeName, type); Object value = attributes.getAttributeValue(index); if (value instanceof Attribute.LateBoundDefault) { return (Attribute.LateBoundDefault) value; } else { return null; } } @Override public Iterable getAttributeNames() { ImmutableList.Builder names = ImmutableList.builder(); for (Attribute a : ruleClass.getAttributes()) { names.add(a.getName()); } return names.build(); } @Nullable @Override public Type getAttributeType(String attrName) { Attribute attr = getAttributeDefinition(attrName); return attr == null ? null : attr.getType(); } @Nullable @Override public Attribute getAttributeDefinition(String attrName) { return ruleClass.getAttributeByNameMaybe(attrName); } @Override public boolean isAttributeValueExplicitlySpecified(String attributeName) { return attributes.isAttributeValueExplicitlySpecified(attributeName); } @Override public String getPackageDefaultHdrsCheck() { return pkg.getDefaultHdrsCheck(); } @Override public Boolean getPackageDefaultTestOnly() { return pkg.getDefaultTestOnly(); } @Override public String getPackageDefaultDeprecation() { return pkg.getDefaultDeprecation(); } @Override public ImmutableList getPackageDefaultCopts() { return pkg.getDefaultCopts(); } @Override public Collection visitLabels() throws InterruptedException { List edges = new ArrayList<>(); Type.LabelVisitor visitor = (label, attribute) -> { if (label != null) { Label absoluteLabel = ruleLabel.resolveRepositoryRelative(label); edges.add(AttributeMap.DepEdge.create(absoluteLabel, attribute)); } }; for (Attribute attribute : ruleClass.getAttributes()) { Type type = attribute.getType(); // TODO(bazel-team): clean up the typing / visitation interface so we don't have to // special-case these types. if (type != BuildType.OUTPUT && type != BuildType.OUTPUT_LIST && type != BuildType.NODEP_LABEL && type != BuildType.NODEP_LABEL_LIST) { visitLabels(attribute, visitor); } } return edges; } /** Visits all labels reachable from the given attribute. */ protected void visitLabels(Attribute attribute, Type.LabelVisitor visitor) throws InterruptedException { Type type = attribute.getType(); Object value = get(attribute.getName(), type); if (value != null) { // null values are particularly possible for computed defaults. type.visitLabels(visitor, value, attribute); } } @Override public final boolean isConfigurable(String attributeName) { Attribute attrDef = getAttributeDefinition(attributeName); return attrDef == null ? false : getSelectorList(attributeName, attrDef.getType()) != null; } public static boolean isConfigurable(Rule rule, String attributeName, Type type) { SelectorList selectorMaybe = getSelectorList( rule.getRuleClassObject(), rule.getLabel(), rule.getAttributeContainer(), attributeName, type); return selectorMaybe != null; } /** * Returns a {@link SelectorList} for the given attribute if the attribute is configurable * for this rule, null otherwise. * * @return a {@link SelectorList} if the attribute takes the form * "attrName = { 'a': value1_of_type_T, 'b': value2_of_type_T }") for this rule, null * if it takes the form "attrName = value_of_type_T", null if it doesn't exist * @throws IllegalArgumentException if the attribute is configurable but of the wrong type */ @Nullable public final SelectorList getSelectorList(String attributeName, Type type) { return getSelectorList(ruleClass, ruleLabel, attributes, attributeName, type); } @Nullable @SuppressWarnings("unchecked") protected static SelectorList getSelectorList( RuleClass ruleClass, Label ruleLabel, AttributeContainer attributes, String attributeName, Type type) { Integer index = ruleClass.getAttributeIndex(attributeName); if (index == null) { return null; } Object attrValue = attributes.getAttributeValue(index); if (!(attrValue instanceof SelectorList)) { return null; } if (((SelectorList) attrValue).getOriginalType() != type) { throw new IllegalArgumentException("Attribute " + attributeName + " is not of type " + type + " in " + ruleClass + " rule " + ruleLabel); } return (SelectorList) attrValue; } /** * Returns the index of the specified attribute, if its type is 'type'. Throws * an exception otherwise. */ private int getIndexWithTypeCheck(String attrName, Type type) { Integer index = ruleClass.getAttributeIndex(attrName); if (index == null) { throw new IllegalArgumentException( "No such attribute " + attrName + " in " + ruleClass + " rule " + ruleLabel); } Attribute attr = ruleClass.getAttribute(index); if (attr.getType() != type) { throw new IllegalArgumentException( "Attribute " + attrName + " is of type " + attr.getType() + " and not of type " + type + " in " + ruleClass + " rule " + ruleLabel); } return index; } /** * Helper routine that just checks the given attribute has the given type for this rule and * throws an IllegalException if not. */ protected void checkType(String attrName, Type type) { getIndexWithTypeCheck(attrName, type); } @Override public boolean has(String attrName) { Attribute attribute = ruleClass.getAttributeByNameMaybe(attrName); return attribute != null; } @Override public boolean has(String attrName, Type type) { return getAttributeType(attrName) == type; } @Override public Location getAttributeLocation(String attrName) { return attributes.getAttributeLocation(attrName); } }