1 // Copyright 2017 The Bazel Authors. All rights reserved. 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // http://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 15 package com.google.devtools.common.options; 16 17 import com.google.common.collect.ImmutableList; 18 import com.google.common.collect.ImmutableMap; 19 import java.lang.reflect.Constructor; 20 import java.lang.reflect.Modifier; 21 import java.util.Collection; 22 import java.util.Map; 23 import javax.annotation.concurrent.Immutable; 24 25 /** 26 * This extends IsolatedOptionsData with information that can only be determined once all the {@link 27 * OptionsBase} subclasses for a parser are known. In particular, this includes expansion 28 * information. 29 */ 30 @Immutable 31 final class OptionsData extends IsolatedOptionsData { 32 33 /** Mapping from each option to the (unparsed) options it expands to, if any. */ 34 private final ImmutableMap<OptionDefinition, ImmutableList<String>> evaluatedExpansions; 35 36 /** Construct {@link OptionsData} by extending an {@link IsolatedOptionsData} with new info. */ 37 private OptionsData( 38 IsolatedOptionsData base, Map<OptionDefinition, ImmutableList<String>> evaluatedExpansions) { 39 super(base); 40 this.evaluatedExpansions = ImmutableMap.copyOf(evaluatedExpansions); 41 } 42 43 private static final ImmutableList<String> EMPTY_EXPANSION = ImmutableList.<String>of(); 44 45 /** 46 * Returns the expansion of an options field, regardless of whether it was defined using {@link 47 * Option#expansion} or {@link Option#expansionFunction}. If the field is not an expansion option, 48 * returns an empty array. 49 */ 50 public ImmutableList<String> getEvaluatedExpansion(OptionDefinition optionDefinition) { 51 ImmutableList<String> result = evaluatedExpansions.get(optionDefinition); 52 return result != null ? result : EMPTY_EXPANSION; 53 } 54 55 /** 56 * Constructs an {@link OptionsData} object for a parser that knows about the given {@link 57 * OptionsBase} classes. In addition to the work done to construct the {@link 58 * IsolatedOptionsData}, this also computes expansion information. If an option has static 59 * expansions or uses an expansion function that takes a Void object, try to precalculate the 60 * expansion here. 61 */ 62 static OptionsData from(Collection<Class<? extends OptionsBase>> classes) { 63 IsolatedOptionsData isolatedData = IsolatedOptionsData.from(classes); 64 65 // All that's left is to compute expansions. 66 ImmutableMap.Builder<OptionDefinition, ImmutableList<String>> evaluatedExpansionsBuilder = 67 ImmutableMap.builder(); 68 for (Map.Entry<String, OptionDefinition> entry : isolatedData.getAllOptionDefinitions()) { 69 OptionDefinition optionDefinition = entry.getValue(); 70 // Determine either the hard-coded expansion, or the ExpansionFunction class. The 71 // OptionProcessor checks at compile time that these aren't used together. 72 String[] constExpansion = optionDefinition.getOptionExpansion(); 73 Class<? extends ExpansionFunction> expansionFunctionClass = 74 optionDefinition.getExpansionFunction(); 75 if (constExpansion.length > 0) { 76 evaluatedExpansionsBuilder.put(optionDefinition, ImmutableList.copyOf(constExpansion)); 77 } else if (optionDefinition.usesExpansionFunction()) { 78 if (Modifier.isAbstract(expansionFunctionClass.getModifiers())) { 79 throw new AssertionError( 80 "The expansionFunction type " + expansionFunctionClass + " must be a concrete type"); 81 } 82 // Evaluate the ExpansionFunction. 83 ExpansionFunction instance; 84 try { 85 Constructor<?> constructor = expansionFunctionClass.getConstructor(); 86 instance = (ExpansionFunction) constructor.newInstance(); 87 } catch (Exception e) { 88 // This indicates an error in the ExpansionFunction, and should be discovered the first 89 // time it is used. 90 throw new AssertionError(e); 91 } 92 ImmutableList<String> expansion = instance.getExpansion(isolatedData); 93 evaluatedExpansionsBuilder.put(optionDefinition, expansion); 94 } 95 } 96 return new OptionsData(isolatedData, evaluatedExpansionsBuilder.build()); 97 } 98 } 99