// 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.packages.BuildType.SelectorList;
import com.google.devtools.build.lib.syntax.Type;
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;
}
@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);
}
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;
}
}
@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 void visitLabels(final AcceptsLabelAttribute observer) throws InterruptedException {
Type.LabelVisitor visitor = new Type.LabelVisitor() {
@Override
public void visit(@Nullable Label label, Attribute attribute) throws InterruptedException {
if (label != null) {
Label absoluteLabel = ruleLabel.resolveRepositoryRelative(label);
observer.acceptLabelAttribute(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);
}
}
}
/** 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
protected 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;
}
}