// Copyright 2017 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.common.options; import com.google.common.collect.ImmutableList; import java.util.function.Function; import javax.annotation.Nullable; /** * The representation of a parsed option instance. * * <p>An option instance is distinct from the final value of an option, as multiple instances * provide values may be overridden or combined in some way. */ public final class ParsedOptionDescription { private final OptionDefinition optionDefinition; private final String commandLineForm; @Nullable private final String unconvertedValue; private final OptionInstanceOrigin origin; public ParsedOptionDescription( OptionDefinition optionDefinition, String commandLineForm, @Nullable String unconvertedValue, OptionInstanceOrigin origin) { this.optionDefinition = optionDefinition; this.commandLineForm = commandLineForm; this.unconvertedValue = unconvertedValue; this.origin = origin; } public OptionDefinition getOptionDefinition() { return optionDefinition; } public String getCommandLineForm() { return commandLineForm; } public String getCanonicalForm() { return getCanonicalFormWithValueEscaper(s -> s); } public String getCanonicalFormWithValueEscaper(Function<String, String> escapingFunction) { // For boolean flags (note that here we do not check for TriState flags, only flags with actual // boolean values, so that we know the return type of getConvertedValue), use the --[no]flag // form for the canonical value. if (optionDefinition.getType().equals(boolean.class)) { try { return ((boolean) getConvertedValue() ? "--" : "--no") + optionDefinition.getOptionName(); } catch (OptionsParsingException e) { throw new RuntimeException("Unexpected parsing exception", e); } } else { String optionString = "--" + optionDefinition.getOptionName(); if (unconvertedValue != null) { // Can be null for Void options. optionString += "=" + escapingFunction.apply(unconvertedValue); } return optionString; } } @Deprecated // TODO(b/65646296) Once external dependencies are cleaned up, use getCanonicalForm() String getDeprecatedCanonicalForm() { String value = unconvertedValue; // For boolean flags (note that here we do not check for TriState flags, only flags with actual // boolean values, so that we know the return type of getConvertedValue), set them all to 1 or // 0, instead of keeping the wide variety of values we accept in their original form. if (optionDefinition.getType().equals(boolean.class)) { try { value = (boolean) getConvertedValue() ? "1" : "0"; } catch (OptionsParsingException e) { throw new RuntimeException("Unexpected parsing exception", e); } } return String.format("--%s=%s", optionDefinition.getOptionName(), value); } public boolean isBooleanOption() { return optionDefinition.getType().equals(boolean.class); } private OptionDocumentationCategory documentationCategory() { return optionDefinition.getDocumentationCategory(); } private ImmutableList<OptionMetadataTag> metadataTags() { return ImmutableList.copyOf(optionDefinition.getOptionMetadataTags()); } public boolean isDocumented() { return documentationCategory() != OptionDocumentationCategory.UNDOCUMENTED && !isHidden(); } public boolean isHidden() { ImmutableList<OptionMetadataTag> tags = metadataTags(); return tags.contains(OptionMetadataTag.HIDDEN) || tags.contains(OptionMetadataTag.INTERNAL); } public String getUnconvertedValue() { return unconvertedValue; } public OptionInstanceOrigin getOrigin() { return origin; } public OptionPriority getPriority() { return origin.getPriority(); } public String getSource() { return origin.getSource(); } OptionDefinition getImplicitDependent() { return origin.getImplicitDependent(); } OptionDefinition getExpandedFrom() { return origin.getExpandedFrom(); } public boolean isExplicit() { return origin.getExpandedFrom() == null && origin.getImplicitDependent() == null; } public Object getConvertedValue() throws OptionsParsingException { Converter<?> converter = optionDefinition.getConverter(); try { return converter.convert(unconvertedValue); } catch (OptionsParsingException e) { // The converter doesn't know the option name, so we supply it here by re-throwing: throw new OptionsParsingException( String.format("While parsing option %s: %s", commandLineForm, e.getMessage()), e); } } @Override public String toString() { StringBuilder result = new StringBuilder(); result.append(optionDefinition); result.append("set to '").append(unconvertedValue).append("' "); result.append("with priority ").append(origin.getPriority()); if (origin.getSource() != null) { result.append(" and source '").append(origin.getSource()).append("'"); } return result.toString(); } }