aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/main/java/com/google/devtools/build/lib/packages/AbstractAttributeMapper.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/main/java/com/google/devtools/build/lib/packages/AbstractAttributeMapper.java')
-rw-r--r--src/main/java/com/google/devtools/build/lib/packages/AbstractAttributeMapper.java212
1 files changed, 212 insertions, 0 deletions
diff --git a/src/main/java/com/google/devtools/build/lib/packages/AbstractAttributeMapper.java b/src/main/java/com/google/devtools/build/lib/packages/AbstractAttributeMapper.java
new file mode 100644
index 0000000000..20b9304fc9
--- /dev/null
+++ b/src/main/java/com/google/devtools/build/lib/packages/AbstractAttributeMapper.java
@@ -0,0 +1,212 @@
+// Copyright 2014 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 com.google.devtools.build.lib.packages;
+
+import com.google.common.collect.ImmutableList;
+import com.google.devtools.build.lib.syntax.Label;
+
+import javax.annotation.Nullable;
+
+/**
+ * Base {@link AttributeMap} implementation providing direct, unmanipulated access to
+ * underlying attribute data as stored within the Rule.
+ *
+ * <p>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> T get(String attributeName, Type<T> type) {
+ int index = getIndexWithTypeCheck(attributeName, type);
+ Object value = attributes.getAttributeValue(index);
+ if (value instanceof Attribute.ComputedDefault) {
+ value = ((Attribute.ComputedDefault) value).getDefault(this);
+ }
+ return type.cast(value);
+ }
+
+ /**
+ * 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.
+ */
+ protected <T> Attribute.ComputedDefault getComputedDefault(String attributeName, Type<T> 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<String> getAttributeNames() {
+ ImmutableList.Builder<String> 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 getPackageDefaultObsolete() {
+ return pkg.getDefaultObsolete();
+ }
+
+ @Override
+ public Boolean getPackageDefaultTestOnly() {
+ return pkg.getDefaultTestOnly();
+ }
+
+ @Override
+ public String getPackageDefaultDeprecation() {
+ return pkg.getDefaultDeprecation();
+ }
+
+ @Override
+ public ImmutableList<String> getPackageDefaultCopts() {
+ return pkg.getDefaultCopts();
+ }
+
+ @Override
+ public void visitLabels(AcceptsLabelAttribute observer) {
+ for (Attribute attribute : ruleClass.getAttributes()) {
+ Type<?> type = attribute.getType();
+ // TODO(bazel-team): This is incoherent: we shouldn't have to special-case these types
+ // for our visitation policy (e.g., why is Type.NODEP_LABEL_LIST excluded but not
+ // Type.NODEP_LABEL?). But this is the semantics the calling code requires. Audit
+ // exactly which calling code expects what and clean up this interface.
+ if (type == Type.OUTPUT || type == Type.OUTPUT_LIST || type == Type.NODEP_LABEL_LIST) {
+ continue;
+ }
+ for (Object value : visitAttribute(attribute.getName(), type)) {
+ if (value == null) {
+ // This is particularly possible for computed defaults.
+ continue;
+ }
+ for (Label label : type.getLabels(value)) {
+ observer.acceptLabelAttribute(label, attribute);
+ }
+ }
+ }
+ }
+
+ /**
+ * Implementations should provide policy-appropriate mappings when an attribute is requested in
+ * the context of a rule visitation.
+ */
+ protected abstract <T> Iterable<T> visitAttribute(String attributeName, Type<T> type);
+
+ /**
+ * Returns a {@link Type.Selector} for the given attribute if the attribute is configurable
+ * for this rule, null otherwise.
+ *
+ * @return a {@link Type.Selector} 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 <T> Type.Selector<T> getSelector(String attributeName, Type<T> type) {
+ Integer index = ruleClass.getAttributeIndex(attributeName);
+ if (index == null) {
+ return null;
+ }
+ Object attrValue = attributes.getAttributeValue(index);
+ if (!(attrValue instanceof Type.Selector<?>)) {
+ return null;
+ }
+ if (((Type.Selector<?>) attrValue).getOriginalType() != type) {
+ throw new IllegalArgumentException("Attribute " + attributeName
+ + " is not of type " + type + " in rule " + ruleLabel.getName());
+ }
+ return (Type.Selector<T>) 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 rule " + ruleLabel.getName());
+ }
+ Attribute attr = ruleClass.getAttribute(index);
+ if (attr.getType() != type) {
+ throw new IllegalArgumentException("Attribute " + attrName
+ + " is not of type " + type + " in rule " + ruleLabel.getName());
+ }
+ 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);
+ }
+}