|  | // 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 static com.google.common.truth.Truth.assertThat; | 
|  | import static org.junit.Assert.fail; | 
|  |  | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import java.lang.reflect.Field; | 
|  | import java.util.ArrayList; | 
|  | import java.util.List; | 
|  | import java.util.Map; | 
|  | import org.junit.Test; | 
|  | import org.junit.runner.RunWith; | 
|  | import org.junit.runners.JUnit4; | 
|  |  | 
|  | /** Tests for {@link IsolatedOptionsData} and {@link OptionsData}. */ | 
|  | @RunWith(JUnit4.class) | 
|  | public class OptionsDataTest { | 
|  |  | 
|  | private static IsolatedOptionsData construct(Class<? extends OptionsBase> optionsClass) | 
|  | throws OptionsParser.ConstructionException { | 
|  | return IsolatedOptionsData.from(ImmutableList.<Class<? extends OptionsBase>>of(optionsClass)); | 
|  | } | 
|  |  | 
|  | private static IsolatedOptionsData construct( | 
|  | Class<? extends OptionsBase> optionsClass1, | 
|  | Class<? extends OptionsBase> optionsClass2) | 
|  | throws OptionsParser.ConstructionException { | 
|  | return IsolatedOptionsData.from( | 
|  | ImmutableList.<Class<? extends OptionsBase>>of(optionsClass1, optionsClass2)); | 
|  | } | 
|  |  | 
|  | private static IsolatedOptionsData construct( | 
|  | Class<? extends OptionsBase> optionsClass1, | 
|  | Class<? extends OptionsBase> optionsClass2, | 
|  | Class<? extends OptionsBase> optionsClass3) | 
|  | throws OptionsParser.ConstructionException { | 
|  | return IsolatedOptionsData.from( | 
|  | ImmutableList.<Class<? extends OptionsBase>>of( | 
|  | optionsClass1, optionsClass2, optionsClass3)); | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class ExampleNameConflictOptions extends OptionsBase { | 
|  | @Option( | 
|  | name = "foo", | 
|  | defaultValue = "1" | 
|  | ) | 
|  | public int foo; | 
|  |  | 
|  | @Option( | 
|  | name = "foo", | 
|  | defaultValue = "I should conflict with foo" | 
|  | ) | 
|  | public String anotherFoo; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testNameConflictInSingleClass() { | 
|  | try { | 
|  | construct(ExampleNameConflictOptions.class); | 
|  | fail("foo should conflict with the previous flag foo"); | 
|  | } catch (DuplicateOptionDeclarationException e) { | 
|  | assertThat(e.getMessage()).contains( | 
|  | "Duplicate option name, due to option: --foo"); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class ExampleIntegerFooOptions extends OptionsBase { | 
|  | @Option( | 
|  | name = "foo", | 
|  | defaultValue = "5" | 
|  | ) | 
|  | public int foo; | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class ExampleBooleanFooOptions extends OptionsBase { | 
|  | @Option( | 
|  | name = "foo", | 
|  | defaultValue = "false" | 
|  | ) | 
|  | public boolean foo; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testNameConflictInTwoClasses() { | 
|  | try { | 
|  | construct(ExampleIntegerFooOptions.class, ExampleBooleanFooOptions.class); | 
|  | fail("foo should conflict with the previous flag foo"); | 
|  | } catch (DuplicateOptionDeclarationException e) { | 
|  | assertThat(e.getMessage()).contains( | 
|  | "Duplicate option name, due to option: --foo"); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class ExamplePrefixFooOptions extends OptionsBase { | 
|  | @Option( | 
|  | name = "nofoo", | 
|  | defaultValue = "false" | 
|  | ) | 
|  | public boolean noFoo; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testBooleanPrefixNameConflict() { | 
|  | // Try the same test in both orders, the parser should fail if the overlapping flag is defined | 
|  | // before or after the boolean flag introduces the alias. | 
|  | try { | 
|  | construct(ExampleBooleanFooOptions.class, ExamplePrefixFooOptions.class); | 
|  | fail("nofoo should conflict with the previous flag foo, " | 
|  | + "since foo, as a boolean flag, can be written as --nofoo"); | 
|  | } catch (DuplicateOptionDeclarationException e) { | 
|  | assertThat(e.getMessage()).contains( | 
|  | "Duplicate option name, due to option --nofoo, it " | 
|  | + "conflicts with a negating alias for boolean flag --foo"); | 
|  | } | 
|  |  | 
|  | try { | 
|  | construct(ExamplePrefixFooOptions.class, ExampleBooleanFooOptions.class); | 
|  | fail("nofoo should conflict with the previous flag foo, " | 
|  | + "since foo, as a boolean flag, can be written as --nofoo"); | 
|  | } catch (DuplicateOptionDeclarationException e) { | 
|  | assertThat(e.getMessage()).contains( | 
|  | "Duplicate option name, due to boolean option alias: --nofoo"); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class ExampleBarWasNamedFooOption extends OptionsBase { | 
|  | @Option( | 
|  | name = "bar", | 
|  | oldName = "foo", | 
|  | defaultValue = "false" | 
|  | ) | 
|  | public boolean bar; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testBooleanAliasWithOldNameConflict() { | 
|  | // Try the same test in both orders, the parser should fail if the overlapping flag is defined | 
|  | // before or after the boolean flag introduces the alias. | 
|  | try { | 
|  | construct(ExamplePrefixFooOptions.class, ExampleBarWasNamedFooOption.class); | 
|  | fail("nofoo should conflict with the previous flag foo, " | 
|  | + "since foo, as a boolean flag, can be written as --nofoo"); | 
|  | } catch (DuplicateOptionDeclarationException e) { | 
|  | assertThat(e.getMessage()).contains( | 
|  | "Duplicate option name, due to boolean option alias: --nofoo"); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class ExampleBarWasNamedNoFooOption extends OptionsBase { | 
|  | @Option( | 
|  | name = "bar", | 
|  | oldName = "nofoo", | 
|  | defaultValue = "false" | 
|  | ) | 
|  | public boolean bar; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testBooleanWithOldNameAsAliasOfBooleanConflict() { | 
|  | // Try the same test in both orders, the parser should fail if the overlapping flag is defined | 
|  | // before or after the boolean flag introduces the alias. | 
|  | try { | 
|  | construct(ExampleBooleanFooOptions.class, ExampleBarWasNamedNoFooOption.class); | 
|  | fail("nofoo, the old name for bar, should conflict with the previous flag foo, " | 
|  | + "since foo, as a boolean flag, can be written as --nofoo"); | 
|  | } catch (DuplicateOptionDeclarationException e) { | 
|  | assertThat(e.getMessage()).contains( | 
|  | "Duplicate option name, due to old option name --nofoo, it conflicts with a " | 
|  | + "negating alias for boolean flag --foo"); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class OldNameConflictExample extends OptionsBase { | 
|  | @Option( | 
|  | name = "new_name", | 
|  | oldName = "old_name", | 
|  | defaultValue = "defaultValue" | 
|  | ) | 
|  | public String flag1; | 
|  |  | 
|  | @Option( | 
|  | name = "old_name", | 
|  | defaultValue = "defaultValue" | 
|  | ) | 
|  | public String flag2; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testOldNameConflict() { | 
|  | try { | 
|  | construct(OldNameConflictExample.class); | 
|  | fail("old_name should conflict with the flag already named old_name"); | 
|  | } catch (DuplicateOptionDeclarationException expected) { | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class StringConverter implements Converter<String> { | 
|  | @Override | 
|  | public String convert(String input) { | 
|  | return input; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String getTypeDescription() { | 
|  | return "a string"; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class InvalidOptionConverter extends OptionsBase { | 
|  | @Option( | 
|  | name = "foo", | 
|  | converter = StringConverter.class, | 
|  | defaultValue = "1" | 
|  | ) | 
|  | public Integer foo; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void errorForInvalidOptionConverter() throws Exception { | 
|  | try { | 
|  | construct(InvalidOptionConverter.class); | 
|  | } catch (AssertionError e) { | 
|  | // Expected exception | 
|  | return; | 
|  | } | 
|  | fail(); | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class InvalidListOptionConverter extends OptionsBase { | 
|  | @Option( | 
|  | name = "foo", | 
|  | converter = StringConverter.class, | 
|  | defaultValue = "1", | 
|  | allowMultiple = true | 
|  | ) | 
|  | public List<Integer> foo; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void errorForInvalidListOptionConverter() throws Exception { | 
|  | try { | 
|  | construct(InvalidListOptionConverter.class); | 
|  | } catch (AssertionError e) { | 
|  | // Expected exception | 
|  | return; | 
|  | } | 
|  | fail(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Dummy options class. | 
|  | * | 
|  | * <p>Option name order is different from field name order. | 
|  | * | 
|  | * <p>There are four fields to increase the likelihood of a non-deterministic order being noticed. | 
|  | */ | 
|  | public static class FieldNamesDifferOptions extends OptionsBase { | 
|  |  | 
|  | @Option( | 
|  | name = "foo", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int aFoo; | 
|  |  | 
|  | @Option( | 
|  | name = "bar", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int bBar; | 
|  |  | 
|  | @Option( | 
|  | name = "baz", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int cBaz; | 
|  |  | 
|  | @Option( | 
|  | name = "qux", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int dQux; | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class EndOfAlphabetOptions extends OptionsBase { | 
|  | @Option( | 
|  | name = "X", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int x; | 
|  |  | 
|  | @Option( | 
|  | name = "Y", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int y; | 
|  | } | 
|  |  | 
|  | /** Dummy options class. */ | 
|  | public static class ReverseOrderedOptions extends OptionsBase { | 
|  | @Option( | 
|  | name = "C", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int c; | 
|  |  | 
|  | @Option( | 
|  | name = "B", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int b; | 
|  |  | 
|  | @Option( | 
|  | name = "A", | 
|  | defaultValue = "0" | 
|  | ) | 
|  | public int a; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void optionsClassesIsOrdered() throws Exception { | 
|  | IsolatedOptionsData data = construct( | 
|  | FieldNamesDifferOptions.class, | 
|  | EndOfAlphabetOptions.class, | 
|  | ReverseOrderedOptions.class); | 
|  | assertThat(data.getOptionsClasses()).containsExactly( | 
|  | FieldNamesDifferOptions.class, | 
|  | EndOfAlphabetOptions.class, | 
|  | ReverseOrderedOptions.class).inOrder(); | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void getAllNamedFieldsIsOrdered() throws Exception { | 
|  | IsolatedOptionsData data = construct( | 
|  | FieldNamesDifferOptions.class, | 
|  | EndOfAlphabetOptions.class, | 
|  | ReverseOrderedOptions.class); | 
|  | ArrayList<String> names = new ArrayList<>(); | 
|  | for (Map.Entry<String, Field> entry : data.getAllNamedFields()) { | 
|  | names.add(entry.getKey()); | 
|  | } | 
|  | assertThat(names).containsExactly( | 
|  | "bar", "baz", "foo", "qux", "X", "Y", "A", "B", "C").inOrder(); | 
|  | } | 
|  |  | 
|  | private List<String> getOptionNames(Iterable<Field> fields) { | 
|  | ArrayList<String> result = new ArrayList<>(); | 
|  | for (Field field : fields) { | 
|  | result.add(field.getAnnotation(Option.class).name()); | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void getFieldsForClassIsOrdered() throws Exception { | 
|  | IsolatedOptionsData data = construct( | 
|  | FieldNamesDifferOptions.class, | 
|  | EndOfAlphabetOptions.class, | 
|  | ReverseOrderedOptions.class); | 
|  | assertThat(getOptionNames(data.getFieldsForClass(FieldNamesDifferOptions.class))) | 
|  | .containsExactly("bar", "baz", "foo", "qux").inOrder(); | 
|  | assertThat(getOptionNames(data.getFieldsForClass(EndOfAlphabetOptions.class))) | 
|  | .containsExactly("X", "Y").inOrder(); | 
|  | assertThat(getOptionNames(data.getFieldsForClass(ReverseOrderedOptions.class))) | 
|  | .containsExactly("A", "B", "C").inOrder(); | 
|  | } | 
|  | } |