// 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.base.Preconditions; import com.google.common.collect.ImmutableList; import java.util.function.Function; import javax.annotation.Nullable; /** * The representation of a parsed option instance. * *

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; @Nullable private final String commandLineForm; @Nullable private final String unconvertedValue; private final OptionInstanceOrigin origin; private ParsedOptionDescription( OptionDefinition optionDefinition, @Nullable String commandLineForm, @Nullable String unconvertedValue, OptionInstanceOrigin origin) { this.optionDefinition = Preconditions.checkNotNull(optionDefinition); this.commandLineForm = commandLineForm; this.unconvertedValue = unconvertedValue; this.origin = Preconditions.checkNotNull(origin); } static ParsedOptionDescription newParsedOptionDescription( OptionDefinition optionDefinition, String commandLineForm, @Nullable String unconvertedValue, OptionInstanceOrigin origin) { // An actual ParsedOptionDescription should always have a form in which it was parsed, but some // options, such as expansion options, legitimately have no value. return new ParsedOptionDescription( optionDefinition, Preconditions.checkNotNull(commandLineForm), unconvertedValue, origin); } /** * This factory should be used when there is no actual parsed option, since in those cases we do * not have an original value or form that the option took. */ static ParsedOptionDescription newDummyInstance( OptionDefinition optionDefinition, OptionInstanceOrigin origin) { return new ParsedOptionDescription(optionDefinition, null, null, origin); } public OptionDefinition getOptionDefinition() { return optionDefinition; } @Nullable public String getCommandLineForm() { return commandLineForm; } public String getCanonicalForm() { return getCanonicalFormWithValueEscaper(s -> s); } public String getCanonicalFormWithValueEscaper(Function 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 metadataTags() { return ImmutableList.copyOf(optionDefinition.getOptionMetadataTags()); } public boolean isDocumented() { return documentationCategory() != OptionDocumentationCategory.UNDOCUMENTED && !isHidden(); } public boolean isHidden() { ImmutableList 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(); } ParsedOptionDescription getImplicitDependent() { return origin.getImplicitDependent(); } ParsedOptionDescription 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() { // Check that a dummy value-less option instance does not output all the default information. if (commandLineForm == null) { return optionDefinition.toString(); } String source = origin.getSource(); return String.format( "option '%s'%s", commandLineForm, source == null ? "" : String.format(" (source %s)", source)); } }