// Copyright 2014 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.escape.CharEscaperBuilder; import com.google.common.escape.Escaper; import java.lang.reflect.Field; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.Map.Entry; /** * Base class for all options classes. Extend this class, adding public instance fields annotated * with {@link Option}. Then you can create instances either programmatically: * * <pre> * X x = Options.getDefaults(X.class); * x.host = "localhost"; * x.port = 80; * </pre> * * or from an array of command-line arguments: * * <pre> * OptionsParser parser = OptionsParser.newOptionsParser(X.class); * parser.parse("--host", "localhost", "--port", "80"); * X x = parser.getOptions(X.class); * </pre> * * <p>Subclasses of {@code OptionsBase} <i>must</i> be constructed reflectively, i.e. using not * {@code new MyOptions()}, but one of the above methods instead. (Direct construction creates an * empty instance, not containing default values. This leads to surprising behavior and often {@code * NullPointerExceptions}, etc.) */ public abstract class OptionsBase { private static final Escaper ESCAPER = new CharEscaperBuilder() .addEscape('\\', "\\\\").addEscape('"', "\\\"").toEscaper(); /** * Subclasses must provide a default (no argument) constructor. */ protected OptionsBase() { // There used to be a sanity check here that checks the stack trace of this constructor // invocation; unfortunately, that makes the options construction about 10x slower. So be // careful with how you construct options classes. } /** * Returns a mapping from option names to values, for each option on this object, including * inherited ones. The mapping is a copy, so subsequent mutations to it or to this object are * independent. Entries are sorted alphabetically. */ public final <O extends OptionsBase> Map<String, Object> asMap() { // Generic O is needed to tell the type system that the toMap() call is safe. // The casts are safe because "this" is an instance of "getClass()" // which subclasses OptionsBase. @SuppressWarnings("unchecked") O castThis = (O) this; @SuppressWarnings("unchecked") Class<O> castClass = (Class<O>) getClass(); Map<String, Object> map = new LinkedHashMap<>(); for (Map.Entry<Field, Object> entry : OptionsParser.toMap(castClass, castThis).entrySet()) { String name = entry.getKey().getAnnotation(Option.class).name(); map.put(name, entry.getValue()); } return map; } @Override public final String toString() { return getClass().getName() + asMap(); } /** * Returns a string that uniquely identifies the options. This value is * intended for analysis caching. */ public final String cacheKey() { StringBuilder result = new StringBuilder(getClass().getName()).append("{"); for (Entry<String, Object> entry : asMap().entrySet()) { result.append(entry.getKey()).append("="); Object value = entry.getValue(); // This special case is needed because List.toString() prints the same // ("[]") for an empty list and for a list with a single empty string. if (value instanceof List<?> && ((List<?>) value).isEmpty()) { result.append("EMPTY"); } else if (value == null) { result.append("NULL"); } else { result .append('"') .append(ESCAPER.escape(value.toString())) .append('"'); } result.append(", "); } return result.append("}").toString(); } @Override public final boolean equals(Object that) { return that != null && this.getClass() == that.getClass() && this.asMap().equals(((OptionsBase) that).asMap()); } @Override public final int hashCode() { return this.getClass().hashCode() + asMap().hashCode(); } }