|  | // 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.annotations.VisibleForTesting; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import com.google.common.collect.Iterables; | 
|  | 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 com.google.devtools.build.lib.vfs.PathFragment; | 
|  |  | 
|  | 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 static final PathFragment VISIBILITY = new PathFragment("visibility"); | 
|  | private static final PathFragment EXTERNAL = new PathFragment("external"); | 
|  |  | 
|  | 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); | 
|  | } | 
|  | 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 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 | 
|  | public <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 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): 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, observer); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Visits all labels reachable from the given attribute. | 
|  | */ | 
|  | protected void visitLabels(Attribute attribute, AcceptsLabelAttribute observer) { | 
|  | Type<?> type = attribute.getType(); | 
|  | Object value = get(attribute.getName(), type); | 
|  | if (value != null) { // null values are particularly possible for computed defaults. | 
|  | for (Label label : extractLabels(type, value)) { | 
|  | Label absoluteLabel; | 
|  | if (attribute.isImplicit() || attribute.isLateBound() | 
|  | || !attributes.isAttributeValueExplicitlySpecified(attribute)) { | 
|  | // Implicit dependencies are not usually present in remote repositories. They are | 
|  | // generally tools, which go to the main repository. | 
|  | absoluteLabel = label; | 
|  | } else if (label.getPackageIdentifier().getRepository().isDefault() | 
|  | && (VISIBILITY.equals(label.getPackageIdentifier().getPackageFragment()) | 
|  | || EXTERNAL.equals(label.getPackageIdentifier().getPackageFragment()))) { | 
|  | // //visibility: and //external: labels must also be special-cased :( | 
|  | absoluteLabel = label; | 
|  | } else { | 
|  | absoluteLabel = ruleLabel.resolveRepositoryRelative(label); | 
|  | } | 
|  | observer.acceptLabelAttribute(absoluteLabel, attribute); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public <T> boolean isConfigurable(String attributeName, Type<T> type) { | 
|  | return getSelectorList(attributeName, type) != 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 | 
|  | @SuppressWarnings("unchecked") | 
|  | protected <T> SelectorList<T> getSelectorList(String attributeName, Type<T> 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 rule " + ruleLabel); | 
|  | } | 
|  | return (SelectorList<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); | 
|  | } | 
|  | 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, Type<?> type) { | 
|  | Attribute attribute = ruleClass.getAttributeByNameMaybe(attrName); | 
|  | return attribute != null && attribute.getType() == type; | 
|  | } | 
|  |  | 
|  | protected static Iterable<Label> extractLabels(Type type, Object value) { | 
|  | return Iterables.filter(type.flatten(value), Label.class); | 
|  | } | 
|  | } |