|  | // 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 static com.google.devtools.build.lib.syntax.SkylarkType.castMap; | 
|  | import static java.util.Collections.singleton; | 
|  |  | 
|  | import com.google.common.base.Function; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import com.google.common.collect.ImmutableMap; | 
|  | import com.google.common.collect.Iterables; | 
|  | import com.google.common.collect.Lists; | 
|  | import com.google.common.collect.Sets; | 
|  | import com.google.common.escape.Escaper; | 
|  | import com.google.common.escape.Escapers; | 
|  | import com.google.devtools.build.lib.events.Location; | 
|  | import com.google.devtools.build.lib.syntax.ClassObject; | 
|  | import com.google.devtools.build.lib.syntax.ClassObject.SkylarkClassObject; | 
|  | import com.google.devtools.build.lib.syntax.Environment; | 
|  | import com.google.devtools.build.lib.syntax.EvalException; | 
|  | import com.google.devtools.build.lib.syntax.Label; | 
|  | import com.google.devtools.build.lib.syntax.SkylarkCallbackFunction; | 
|  | import com.google.devtools.build.lib.util.StringUtil; | 
|  | import com.google.devtools.build.lib.vfs.FileSystemUtils; | 
|  | import com.google.devtools.build.lib.vfs.PathFragment; | 
|  |  | 
|  | import java.util.Arrays; | 
|  | import java.util.Collection; | 
|  | import java.util.Collections; | 
|  | import java.util.HashMap; | 
|  | import java.util.LinkedHashSet; | 
|  | import java.util.List; | 
|  | import java.util.Map; | 
|  | import java.util.Set; | 
|  |  | 
|  | import javax.annotation.Nullable; | 
|  |  | 
|  | /** | 
|  | * A function interface allowing rules to specify their set of implicit outputs | 
|  | * in a more dynamic way than just simple template-substitution.  For example, | 
|  | * the set of implicit outputs may be a function of rule attributes. | 
|  | */ | 
|  | public abstract class ImplicitOutputsFunction { | 
|  |  | 
|  | /** | 
|  | * Implicit output functions for Skylark supporting key value access of expanded implicit outputs. | 
|  | */ | 
|  | public abstract static class SkylarkImplicitOutputsFunction extends ImplicitOutputsFunction { | 
|  |  | 
|  | public abstract ImmutableMap<String, String> calculateOutputs(AttributeMap map) | 
|  | throws EvalException; | 
|  |  | 
|  | @Override | 
|  | public Iterable<String> getImplicitOutputs(AttributeMap map) throws EvalException { | 
|  | return calculateOutputs(map).values(); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Implicit output functions executing Skylark code. | 
|  | */ | 
|  | public static final class SkylarkImplicitOutputsFunctionWithCallback | 
|  | extends SkylarkImplicitOutputsFunction { | 
|  |  | 
|  | private final SkylarkCallbackFunction callback; | 
|  | private final Location loc; | 
|  |  | 
|  | public SkylarkImplicitOutputsFunctionWithCallback( | 
|  | SkylarkCallbackFunction callback, Location loc) { | 
|  | this.callback = callback; | 
|  | this.loc = loc; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public ImmutableMap<String, String> calculateOutputs(AttributeMap map) throws EvalException { | 
|  | Map<String, Object> attrValues = new HashMap<>(); | 
|  | for (String attrName : map.getAttributeNames()) { | 
|  | Type<?> attrType = map.getAttributeType(attrName); | 
|  | // Don't include configurable attributes: we don't know which value they might take | 
|  | // since we don't yet have a build configuration. | 
|  | if (!map.isConfigurable(attrName, attrType)) { | 
|  | Object value = map.get(attrName, attrType); | 
|  | attrValues.put(attrName, value == null ? Environment.NONE : value); | 
|  | } | 
|  | } | 
|  | ClassObject attrs = new SkylarkClassObject(attrValues, "Attribute '%s' either doesn't exist " | 
|  | + "or uses a select() (i.e. could have multiple values)"); | 
|  | try { | 
|  | ImmutableMap.Builder<String, String> builder = ImmutableMap.builder(); | 
|  | for (Map.Entry<String, String> entry : castMap(callback.call(attrs), | 
|  | String.class, String.class, "implicit outputs function return value").entrySet()) { | 
|  | Iterable<String> substitutions = fromTemplates(entry.getValue()).getImplicitOutputs(map); | 
|  | if (!Iterables.isEmpty(substitutions)) { | 
|  | builder.put(entry.getKey(), Iterables.getOnlyElement(substitutions)); | 
|  | } | 
|  | } | 
|  | return builder.build(); | 
|  | } catch (IllegalArgumentException e) { | 
|  | throw new EvalException(loc, e.getMessage()); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Implicit output functions using a simple an output map. | 
|  | */ | 
|  | public static final class SkylarkImplicitOutputsFunctionWithMap | 
|  | extends SkylarkImplicitOutputsFunction { | 
|  |  | 
|  | private final ImmutableMap<String, String> outputMap; | 
|  |  | 
|  | public SkylarkImplicitOutputsFunctionWithMap(ImmutableMap<String, String> outputMap) { | 
|  | this.outputMap = outputMap; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public ImmutableMap<String, String> calculateOutputs(AttributeMap map) { | 
|  | ImmutableMap.Builder<String, String> builder = ImmutableMap.builder(); | 
|  | for (Map.Entry<String, String> entry : outputMap.entrySet()) { | 
|  | Iterable<String> substitutions = fromTemplates(entry.getValue()).getImplicitOutputs(map); | 
|  | if (!Iterables.isEmpty(substitutions)) { | 
|  | builder.put(entry.getKey(), Iterables.getOnlyElement(substitutions)); | 
|  | } | 
|  | } | 
|  | return builder.build(); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Implicit output functions which can not throw an EvalException. | 
|  | */ | 
|  | public abstract static class SafeImplicitOutputsFunction extends ImplicitOutputsFunction { | 
|  | @Override | 
|  | public abstract Iterable<String> getImplicitOutputs(AttributeMap map); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * An interface to objects that can retrieve rule attributes. | 
|  | */ | 
|  | public interface AttributeValueGetter { | 
|  | /** | 
|  | * Returns the value(s) of attribute "attr" in "rule", or empty set if attribute unknown. | 
|  | */ | 
|  | Set<String> get(AttributeMap rule, String attr); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * The default rule attribute retriever. | 
|  | * | 
|  | * <p>Custom {@link AttributeValueGetter} implementations may delegate to this object as a | 
|  | * fallback mechanism. | 
|  | */ | 
|  | public static final AttributeValueGetter DEFAULT_RULE_ATTRIBUTE_GETTER = | 
|  | new AttributeValueGetter() { | 
|  | @Override | 
|  | public Set<String> get(AttributeMap rule, String attr) { | 
|  | return attributeValues(rule, attr); | 
|  | } | 
|  | }; | 
|  |  | 
|  | private static final Escaper PERCENT_ESCAPER = Escapers.builder().addEscape('%', "%%").build(); | 
|  |  | 
|  | /** | 
|  | * Given a newly-constructed Rule instance (with attributes populated), | 
|  | * returns the list of output files that this rule produces implicitly. | 
|  | */ | 
|  | public abstract Iterable<String> getImplicitOutputs(AttributeMap rule) throws EvalException; | 
|  |  | 
|  | /** | 
|  | * The implicit output function that returns no files. | 
|  | */ | 
|  | public static final SafeImplicitOutputsFunction NONE = new SafeImplicitOutputsFunction() { | 
|  | @Override public Iterable<String> getImplicitOutputs(AttributeMap rule) { | 
|  | return Collections.emptyList(); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * A convenience wrapper for {@link #fromTemplates(Iterable)}. | 
|  | */ | 
|  | public static SafeImplicitOutputsFunction fromTemplates(String... templates) { | 
|  | return fromTemplates(Arrays.asList(templates)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * The implicit output function that generates files based on a set of | 
|  | * template substitutions using rule attribute values. | 
|  | * | 
|  | * @param templates The templates used to construct the name of the implicit | 
|  | *   output file target.  The substring "%{name}" will be replaced by the | 
|  | *   actual name of the rule, the substring "%{srcs}" will be replaced by the | 
|  | *   name of each source file without its extension.  If multiple %{} | 
|  | *   substrings exist, the cross-product of them is generated. | 
|  | */ | 
|  | public static SafeImplicitOutputsFunction fromTemplates(final Iterable<String> templates) { | 
|  | return new SafeImplicitOutputsFunction() { | 
|  | // TODO(bazel-team): parse the templates already here | 
|  | @Override | 
|  | public Iterable<String> getImplicitOutputs(AttributeMap rule) { | 
|  | Iterable<String> result = null; | 
|  | for (String template : templates) { | 
|  | List<String> substitutions = substitutePlaceholderIntoTemplate(template, rule); | 
|  | if (substitutions.isEmpty()) { | 
|  | continue; | 
|  | } | 
|  | if (result == null) { | 
|  | result = substitutions; | 
|  | } else { | 
|  | result = Iterables.concat(result, substitutions); | 
|  | } | 
|  | } | 
|  | if (result == null) { | 
|  | return ImmutableList.<String>of(); | 
|  | } else { | 
|  | return Sets.newLinkedHashSet(result); | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String toString() { | 
|  | return StringUtil.joinEnglishList(templates); | 
|  | } | 
|  | }; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * A convenience wrapper for {@link #fromFunctions(Iterable)}. | 
|  | */ | 
|  | public static SafeImplicitOutputsFunction fromFunctions( | 
|  | SafeImplicitOutputsFunction... functions) { | 
|  | return fromFunctions(Arrays.asList(functions)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * The implicit output function that generates files based on a set of | 
|  | * template substitutions using rule attribute values. | 
|  | * | 
|  | * @param functions The functions used to construct the name of the implicit | 
|  | *   output file target.  The substring "%{name}" will be replaced by the | 
|  | *   actual name of the rule, the substring "%{srcs}" will be replaced by the | 
|  | *   name of each source file without its extension.  If multiple %{} | 
|  | *   substrings exist, the cross-product of them is generated. | 
|  | */ | 
|  | public static SafeImplicitOutputsFunction fromFunctions( | 
|  | final Iterable<SafeImplicitOutputsFunction> functions) { | 
|  | return new SafeImplicitOutputsFunction() { | 
|  | @Override | 
|  | public Iterable<String> getImplicitOutputs(AttributeMap rule) { | 
|  | Collection<String> result = new LinkedHashSet<>(); | 
|  | for (SafeImplicitOutputsFunction function : functions) { | 
|  | Iterables.addAll(result, function.getImplicitOutputs(rule)); | 
|  | } | 
|  | return result; | 
|  | } | 
|  | @Override | 
|  | public String toString() { | 
|  | return StringUtil.joinEnglishList(functions); | 
|  | } | 
|  | }; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Coerces attribute "attrName" of the specified rule into a sequence of | 
|  | * strings.  Helper function for {@link #fromTemplates(Iterable)}. | 
|  | */ | 
|  | private static Set<String> attributeValues(AttributeMap rule, String attrName) { | 
|  | // Special case "name" since it's not treated as an attribute. | 
|  | if (attrName.equals("name")) { | 
|  | return singleton(rule.getName()); | 
|  | } else if (attrName.equals("dirname")) { | 
|  | PathFragment dir = new PathFragment(rule.getName()).getParentDirectory(); | 
|  | return (dir.segmentCount() == 0) ? singleton("") : singleton(dir.getPathString() + "/"); | 
|  | } else if (attrName.equals("basename")) { | 
|  | return singleton(new PathFragment(rule.getName()).getBaseName()); | 
|  | } | 
|  |  | 
|  | Type<?> attrType = rule.getAttributeType(attrName); | 
|  | if (attrType == null) { return Collections.emptySet(); } | 
|  | // String attributes and lists are easy. | 
|  | if (Type.STRING == attrType) { | 
|  | return singleton(rule.get(attrName, Type.STRING)); | 
|  | } else if (Type.STRING_LIST == attrType) { | 
|  | return Sets.newLinkedHashSet(rule.get(attrName, Type.STRING_LIST)); | 
|  | } else if (Type.LABEL == attrType) { | 
|  | // Labels are most often used to change the extension, | 
|  | // e.g. %.foo -> %.java, so we return the basename w/o extension. | 
|  | Label label = rule.get(attrName, Type.LABEL); | 
|  | return singleton(FileSystemUtils.removeExtension(label.getName())); | 
|  | } else if (Type.LABEL_LIST == attrType) { | 
|  | // Labels are most often used to change the extension, | 
|  | // e.g. %.foo -> %.java, so we return the basename w/o extension. | 
|  | return Sets.newLinkedHashSet( | 
|  | Iterables.transform(rule.get(attrName, Type.LABEL_LIST), | 
|  | new Function<Label, String>() { | 
|  | @Override | 
|  | public String apply(Label label) { | 
|  | return FileSystemUtils.removeExtension(label.getName()); | 
|  | } | 
|  | })); | 
|  | } else if (Type.OUTPUT == attrType) { | 
|  | Label out = rule.get(attrName, Type.OUTPUT); | 
|  | return singleton(out.getName()); | 
|  | } else if (Type.OUTPUT_LIST == attrType) { | 
|  | return Sets.newLinkedHashSet( | 
|  | Iterables.transform(rule.get(attrName, Type.OUTPUT_LIST), | 
|  | new Function<Label, String>() { | 
|  | @Override | 
|  | public String apply(Label label) { | 
|  | return label.getName(); | 
|  | } | 
|  | })); | 
|  | } | 
|  | throw new IllegalArgumentException( | 
|  | "Don't know how to handle " + attrName + " : " + attrType); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Collects all named placeholders from the template while replacing them with %s. | 
|  | * | 
|  | * <p>Example: for {@code template} "%{name}_%{locales}.foo", it will return "%s_%s.foo" and | 
|  | * store "name" and "locales" in {@code placeholders}. | 
|  | * | 
|  | * <p>Incomplete placeholders are treated like text: for "a-%{x}-%{y" this method returns | 
|  | * "a-%s-%%{y" and stores "x" in {@code placeholders}. | 
|  | * | 
|  | * @param template a string with placeholders of the format %{...} | 
|  | * @param placeholders a collection to collect placeholders into; may contain duplicates if not a | 
|  | *     Set | 
|  | * @return a format string for {@link String#format}, created from the template string with every | 
|  | *     placeholder replaced by %s | 
|  | */ | 
|  | public static String createPlaceholderSubstitutionFormatString(String template, | 
|  | Collection<String> placeholders) { | 
|  | return createPlaceholderSubstitutionFormatStringRecursive(template, placeholders, | 
|  | new StringBuilder()); | 
|  | } | 
|  |  | 
|  | private static String createPlaceholderSubstitutionFormatStringRecursive(String template, | 
|  | Collection<String> placeholders, StringBuilder formatBuilder) { | 
|  | int start = template.indexOf("%{"); | 
|  | if (start < 0) { | 
|  | return formatBuilder.append(PERCENT_ESCAPER.escape(template)).toString(); | 
|  | } | 
|  |  | 
|  | int end = template.indexOf('}', start + 2); | 
|  | if (end < 0) { | 
|  | return formatBuilder.append(PERCENT_ESCAPER.escape(template)).toString(); | 
|  | } | 
|  |  | 
|  | formatBuilder.append(PERCENT_ESCAPER.escape(template.substring(0, start))).append("%s"); | 
|  | placeholders.add(template.substring(start + 2, end)); | 
|  | return createPlaceholderSubstitutionFormatStringRecursive(template.substring(end + 1), | 
|  | placeholders, formatBuilder); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Given a template string, replaces all placeholders of the form %{...} with | 
|  | * the values from attributeSource.  If there are multiple placeholders, then | 
|  | * the output is the cross product of substitutions. | 
|  | */ | 
|  | public static ImmutableList<String> substitutePlaceholderIntoTemplate(String template, | 
|  | AttributeMap rule) { | 
|  | return substitutePlaceholderIntoTemplate(template, rule, DEFAULT_RULE_ATTRIBUTE_GETTER, null); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Substitutes attribute-placeholders in a template string, producing all possible combinations. | 
|  | * | 
|  | * @param template the template string, may contain named placeholders for rule attributes, like | 
|  | *     <code>%{name}</code> or <code>%{deps}</code> | 
|  | * @param rule the rule whose attributes the placeholders correspond to | 
|  | * @param placeholdersInTemplate if specified, will contain all placeholders found in the | 
|  | *     template; may contain duplicates | 
|  | * @return all possible combinations of the attributes referenced by the placeholders, | 
|  | *     substituted into the template; empty if any of the placeholders expands to no values | 
|  | */ | 
|  | public static ImmutableList<String> substitutePlaceholderIntoTemplate(String template, | 
|  | AttributeMap rule, AttributeValueGetter attributeGetter, | 
|  | @Nullable List<String> placeholdersInTemplate) { | 
|  | List<String> placeholders = (placeholdersInTemplate == null) | 
|  | ? Lists.<String>newArrayList() | 
|  | : placeholdersInTemplate; | 
|  | String formatStr = createPlaceholderSubstitutionFormatString(template, placeholders); | 
|  | if (placeholders.isEmpty()) { | 
|  | return ImmutableList.of(template); | 
|  | } | 
|  |  | 
|  | List<Set<String>> values = Lists.newArrayListWithCapacity(placeholders.size()); | 
|  | for (String placeholder : placeholders) { | 
|  | Set<String> attrValues = attributeGetter.get(rule, placeholder); | 
|  | if (attrValues.isEmpty()) { | 
|  | return ImmutableList.<String>of(); | 
|  | } | 
|  | values.add(attrValues); | 
|  | } | 
|  | ImmutableList.Builder<String> out = new ImmutableList.Builder<>(); | 
|  | for (List<String> combination : Sets.cartesianProduct(values)) { | 
|  | out.add(String.format(formatStr, combination.toArray())); | 
|  | } | 
|  | return out.build(); | 
|  | } | 
|  | } |