|  | // Copyright 2016 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.rules; | 
|  |  | 
|  | import com.google.common.base.Joiner; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import com.google.devtools.build.lib.analysis.ConfiguredTarget; | 
|  | import com.google.devtools.build.lib.analysis.TransitiveInfoCollection; | 
|  | import com.google.devtools.build.lib.analysis.TransitiveInfoProvider; | 
|  | import com.google.devtools.build.lib.cmdline.Label; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable; | 
|  | import com.google.devtools.build.lib.util.Preconditions; | 
|  |  | 
|  | /** | 
|  | * A provider that gives information about the aliases a rule was resolved through. | 
|  | */ | 
|  | @Immutable | 
|  | public final class AliasProvider implements TransitiveInfoProvider { | 
|  | // We don't expect long alias chains, so it's better to have a list instead of a nested set | 
|  | private final ImmutableList<Label> aliasChain; | 
|  |  | 
|  | public AliasProvider(ImmutableList<Label> aliasChain) { | 
|  | Preconditions.checkState(!aliasChain.isEmpty()); | 
|  | this.aliasChain = aliasChain; | 
|  | } | 
|  |  | 
|  | public static AliasProvider fromAliasRule(Label label, ConfiguredTarget actual) { | 
|  | ImmutableList.Builder<Label> chain = ImmutableList.builder(); | 
|  | chain.add(label); | 
|  | AliasProvider dep = actual.getProvider(AliasProvider.class); | 
|  | if (dep != null) { | 
|  | chain.addAll(dep.getAliasChain()); | 
|  | } | 
|  |  | 
|  | return new AliasProvider(chain.build()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the label by which it was referred to in the BUILD file. | 
|  | * | 
|  | * <p>For non-alias rules, it's the label of the rule itself, for alias rules, it's the label of | 
|  | * the alias rule. | 
|  | */ | 
|  | public static Label getDependencyLabel(TransitiveInfoCollection dep) { | 
|  | AliasProvider aliasProvider = dep.getProvider(AliasProvider.class); | 
|  | return aliasProvider != null | 
|  | ? aliasProvider.getAliasChain().get(0) | 
|  | : dep.getLabel(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the list of aliases from top to bottom (i.e. the last alias depends on the actual | 
|  | * resolved target and the first alias is the one that was in the attribute of the rule currently | 
|  | * being analyzed) | 
|  | */ | 
|  | public ImmutableList<Label> getAliasChain() { | 
|  | return aliasChain; | 
|  | } | 
|  |  | 
|  | public static String printLabelWithAliasChain(ConfiguredTarget target) { | 
|  | AliasProvider aliasProvider = target.getProvider(AliasProvider.class); | 
|  | String suffix = aliasProvider == null | 
|  | ? "" | 
|  | : " (aliased through '" + Joiner.on("' -> '").join(aliasProvider.getAliasChain()) + "')"; | 
|  |  | 
|  | return "'" + target.getLabel() + "'" + suffix; | 
|  | } | 
|  | } |