// 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.build.lib.runtime; import static com.google.common.truth.Truth.assertThat; import static org.junit.Assert.fail; import com.google.common.collect.ArrayListMultimap; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import com.google.common.collect.ListMultimap; import com.google.devtools.build.lib.analysis.BlazeDirectories; import com.google.devtools.build.lib.analysis.ServerDirectories; import com.google.devtools.build.lib.bazel.rules.BazelRulesModule; import com.google.devtools.build.lib.events.Event; import com.google.devtools.build.lib.events.StoredEventHandler; import com.google.devtools.build.lib.runtime.BlazeOptionHandler.RcChunkOfArgs; import com.google.devtools.build.lib.runtime.proto.InvocationPolicyOuterClass.InvocationPolicy; import com.google.devtools.build.lib.testutil.Scratch; import com.google.devtools.build.lib.testutil.TestConstants; import com.google.devtools.common.options.OptionsParser; import com.google.devtools.common.options.OptionsParsingException; import com.google.devtools.common.options.OptionsProvider; import com.google.devtools.common.options.TestOptions; import java.util.Arrays; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.JUnit4; /** * Tests {@link BlazeOptionHandler}. * *

Avoids testing anything that is controlled by the {@link BlazeCommandDispatcher}, for * isolation. As a part of this, this test intentionally avoids testing how errors and informational * messages are logged, to minimize dependence on the ui, and only checks for the existence of these * messages. */ @RunWith(JUnit4.class) public class BlazeOptionHandlerTest { private final Scratch scratch = new Scratch(); private final StoredEventHandler eventHandler = new StoredEventHandler(); private OptionsParser parser; private BlazeRuntime runtime; private BlazeOptionHandler optionHandler; @Before public void initStuff() throws Exception { parser = OptionsParser.newOptionsParser( ImmutableList.of(TestOptions.class, CommonCommandOptions.class, ClientOptions.class)); parser.setAllowResidue(true); String productName = TestConstants.PRODUCT_NAME; ServerDirectories serverDirectories = new ServerDirectories( scratch.dir("install_base"), scratch.dir("output_base"), scratch.dir("user_root")); this.runtime = new BlazeRuntime.Builder() .setFileSystem(scratch.getFileSystem()) .setServerDirectories(serverDirectories) .setProductName(productName) .setStartupOptionsProvider( OptionsParser.newOptionsParser(BlazeServerStartupOptions.class)) .addBlazeModule(new BazelRulesModule()) .build(); this.runtime.overrideCommands(ImmutableList.of(new C0Command())); BlazeDirectories directories = new BlazeDirectories( serverDirectories, scratch.dir("workspace"), /* defaultSystemJavabase= */ null, productName); runtime.initWorkspace(directories, /*binTools=*/ null); optionHandler = new BlazeOptionHandler( runtime, runtime.getWorkspace(), new C0Command(), C0Command.class.getAnnotation(Command.class), parser, InvocationPolicy.getDefaultInstance()); } @Command( name = "c0", shortDescription = "c0 desc", help = "c0 help", options = {TestOptions.class} ) private static class C0Command implements BlazeCommand { @Override public BlazeCommandResult exec(CommandEnvironment env, OptionsProvider options) { throw new UnsupportedOperationException(); } @Override public void editOptions(OptionsParser optionsParser) {} } private ListMultimap structuredArgsFrom2SimpleRcsWithOnlyResidue() { ListMultimap structuredArgs = ArrayListMultimap.create(); // first add all lines of rc1, then rc2, to simulate a simple, import free, 2 rc file setup. structuredArgs.put("c0", new RcChunkOfArgs("rc1", ImmutableList.of("a"))); structuredArgs.put("c0:config", new RcChunkOfArgs("rc1", ImmutableList.of("b"))); structuredArgs.put("common", new RcChunkOfArgs("rc2", ImmutableList.of("c"))); structuredArgs.put("c0", new RcChunkOfArgs("rc2", ImmutableList.of("d", "e"))); structuredArgs.put("c1:other", new RcChunkOfArgs("rc2", ImmutableList.of("f", "g"))); return structuredArgs; } private ListMultimap structuredArgsFrom2SimpleRcsWithFlags() { ListMultimap structuredArgs = ArrayListMultimap.create(); structuredArgs.put( "c0", new RcChunkOfArgs("rc1", ImmutableList.of("--test_multiple_string=foo"))); structuredArgs.put( "c0:config", new RcChunkOfArgs("rc1", ImmutableList.of("--test_multiple_string=config"))); structuredArgs.put( "common", new RcChunkOfArgs("rc2", ImmutableList.of("--test_multiple_string=common"))); structuredArgs.put( "c0", new RcChunkOfArgs("rc2", ImmutableList.of("--test_multiple_string=bar"))); structuredArgs.put( "c1:other", new RcChunkOfArgs("rc2", ImmutableList.of("--test_multiple_string=other"))); return structuredArgs; } private ListMultimap structuredArgsFromImportedRcsWithOnlyResidue() { ListMultimap structuredArgs = ArrayListMultimap.create(); // first add all lines of rc1, then rc2, but then jump back to 1 as if rc2 was loaded in an // import statement halfway through rc1. structuredArgs.put("c0", new RcChunkOfArgs("rc1", ImmutableList.of("a"))); structuredArgs.put("c0:config", new RcChunkOfArgs("rc1", ImmutableList.of("b"))); structuredArgs.put("common", new RcChunkOfArgs("rc2", ImmutableList.of("c"))); structuredArgs.put("c0", new RcChunkOfArgs("rc2", ImmutableList.of("d", "e"))); structuredArgs.put("c1:other", new RcChunkOfArgs("rc2", ImmutableList.of("f", "g"))); structuredArgs.put("c0", new RcChunkOfArgs("rc1", ImmutableList.of("h"))); return structuredArgs; } @Test public void testStructureRcOptionsAndConfigs_argumentless() throws Exception { ListMultimap structuredRc = BlazeOptionHandler.structureRcOptionsAndConfigs( eventHandler, Arrays.asList("rc1", "rc2"), Arrays.asList(), ImmutableSet.of("c0", "c1")); assertThat(structuredRc).isEmpty(); assertThat(eventHandler.isEmpty()).isTrue(); } @Test public void testStructureRcOptionsAndConfigs_configOnly() throws Exception { BlazeOptionHandler.structureRcOptionsAndConfigs( eventHandler, Arrays.asList("rc1", "rc2"), Arrays.asList(new ClientOptions.OptionOverride(0, "c0:none", "a")), ImmutableSet.of("c0")); assertThat(eventHandler.isEmpty()).isTrue(); } @Test public void testStructureRcOptionsAndConfigs_invalidCommand() throws Exception { BlazeOptionHandler.structureRcOptionsAndConfigs( eventHandler, Arrays.asList("rc1", "rc2"), Arrays.asList(new ClientOptions.OptionOverride(0, "c1", "a")), ImmutableSet.of("c0")); assertThat(eventHandler.getEvents()) .contains( Event.warn("while reading option defaults file 'rc1':\n invalid command name 'c1'.")); } @Test public void testStructureRcOptionsAndConfigs_twoRcs() throws Exception { ListMultimap structuredRc = BlazeOptionHandler.structureRcOptionsAndConfigs( eventHandler, Arrays.asList("rc1", "rc2"), Arrays.asList( new ClientOptions.OptionOverride(0, "c0", "a"), new ClientOptions.OptionOverride(0, "c0:config", "b"), new ClientOptions.OptionOverride(1, "common", "c"), new ClientOptions.OptionOverride(1, "c0", "d"), new ClientOptions.OptionOverride(1, "c0", "e"), new ClientOptions.OptionOverride(1, "c1:other", "f"), new ClientOptions.OptionOverride(1, "c1:other", "g")), ImmutableSet.of("c0", "c1")); assertThat(structuredRc).isEqualTo(structuredArgsFrom2SimpleRcsWithOnlyResidue()); assertThat(eventHandler.isEmpty()).isTrue(); } @Test public void testStructureRcOptionsAndConfigs_importedRcs() throws Exception { ListMultimap structuredRc = BlazeOptionHandler.structureRcOptionsAndConfigs( eventHandler, Arrays.asList("rc1", "rc2"), Arrays.asList( new ClientOptions.OptionOverride(0, "c0", "a"), new ClientOptions.OptionOverride(0, "c0:config", "b"), new ClientOptions.OptionOverride(1, "common", "c"), new ClientOptions.OptionOverride(1, "c0", "d"), new ClientOptions.OptionOverride(1, "c0", "e"), new ClientOptions.OptionOverride(1, "c1:other", "f"), new ClientOptions.OptionOverride(1, "c1:other", "g"), new ClientOptions.OptionOverride(0, "c0", "h")), ImmutableSet.of("c0", "c1")); assertThat(structuredRc).isEqualTo(structuredArgsFromImportedRcsWithOnlyResidue()); assertThat(eventHandler.isEmpty()).isTrue(); } @Test public void testStructureRcOptionsAndConfigs_badOverrideIndex() throws Exception { ListMultimap structuredRc = BlazeOptionHandler.structureRcOptionsAndConfigs( eventHandler, Arrays.asList("rc1", "rc2"), Arrays.asList( new ClientOptions.OptionOverride(0, "c0", "a"), new ClientOptions.OptionOverride(0, "c0:config", "b"), new ClientOptions.OptionOverride(2, "c4:other", "z"), new ClientOptions.OptionOverride(-1, "c3:other", "q"), new ClientOptions.OptionOverride(1, "common", "c"), new ClientOptions.OptionOverride(1, "c0", "d"), new ClientOptions.OptionOverride(1, "c0", "e"), new ClientOptions.OptionOverride(1, "c1:other", "f"), new ClientOptions.OptionOverride(1, "c1:other", "g")), ImmutableSet.of("c0", "c1")); assertThat(structuredRc).isEqualTo(structuredArgsFrom2SimpleRcsWithOnlyResidue()); assertThat(eventHandler.getEvents()) .containsAllOf( Event.warn("inconsistency in generated command line args. Ignoring bogus argument\n"), Event.warn("inconsistency in generated command line args. Ignoring bogus argument\n")); } @Test public void testParseRcOptions_empty() throws Exception { optionHandler.parseRcOptions(eventHandler, ArrayListMultimap.create()); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); } @Test public void testParseRcOptions_flatRcs_residue() throws Exception { optionHandler.parseRcOptions(eventHandler, structuredArgsFrom2SimpleRcsWithOnlyResidue()); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).containsExactly("c", "a", "d", "e").inOrder(); } @Test public void testParseRcOptions_flatRcs_flags() throws Exception { optionHandler.parseRcOptions(eventHandler, structuredArgsFrom2SimpleRcsWithFlags()); assertThat(eventHandler.getEvents()).isEmpty(); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString).containsExactly("common", "foo", "bar").inOrder(); } @Test public void testParseRcOptions_importedRcs_residue() throws Exception { optionHandler.parseRcOptions(eventHandler, structuredArgsFromImportedRcsWithOnlyResidue()); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).containsExactly("c", "a", "d", "e", "h").inOrder(); } @Test public void testExpandConfigOptions_configless() throws Exception { optionHandler.expandConfigOptions(eventHandler, structuredArgsFrom2SimpleRcsWithOnlyResidue()); assertThat(parser.getResidue()).isEmpty(); } @Test public void testExpandConfigOptions_withConfig() throws Exception { parser.parse("--config=config"); optionHandler.expandConfigOptions(eventHandler, structuredArgsFrom2SimpleRcsWithOnlyResidue()); assertThat(parser.getResidue()).containsExactly("b"); assertThat(optionHandler.getRcfileNotes()) .containsExactly("Found applicable config definition c0:config in file rc1: b"); } @Test public void testExpandConfigOptions_withConfigForUnapplicableCommand() throws Exception { try { parser.parse("--config=other"); optionHandler.expandConfigOptions( eventHandler, structuredArgsFrom2SimpleRcsWithOnlyResidue()); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()).isEmpty(); fail(); } catch (OptionsParsingException e) { assertThat(e).hasMessageThat().contains("Config value other is not defined in any .rc file"); } } @Test public void testUndefinedConfig() { try { parser.parse("--config=invalid"); optionHandler.expandConfigOptions(eventHandler, ArrayListMultimap.create()); fail(); } catch (OptionsParsingException e) { assertThat(e) .hasMessageThat() .contains("Config value invalid is not defined in any .rc file"); } } @Test public void testParseOptions_argless() { optionHandler.parseOptions(ImmutableList.of("c0"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()).isEmpty(); } @Test public void testParseOptions_residue() { optionHandler.parseOptions(ImmutableList.of("c0", "res"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).contains("res"); assertThat(optionHandler.getRcfileNotes()).isEmpty(); } @Test public void testParseOptions_explicitOption() { optionHandler.parseOptions( ImmutableList.of("c0", "--test_multiple_string=explicit"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()).isEmpty(); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString).containsExactly("explicit"); } @Test public void testParseOptions_rcOption() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc_a", "--default_override=0:c0=--test_multiple_string=rc_b", "--rc_source=/somewhere/.blazerc"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); // Check that multiple options in the same rc chunk are collapsed into 1 announce_rc entry. assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc_a --test_multiple_string=rc_b"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString).containsExactly("rc_a", "rc_b"); } @Test public void testParseOptions_multipleRcs() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc1_a", "--default_override=1:c0=--test_multiple_string=rc2", "--default_override=0:c0=--test_multiple_string=rc1_b", "--rc_source=/somewhere/.blazerc", "--rc_source=/some/other/.blazerc"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc1_a", "Reading rc options for 'c0' from /some/other/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc2", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc1_b"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString).containsExactly("rc1_a", "rc2", "rc1_b").inOrder(); } @Test public void testParseOptions_multipleRcsWithMultipleCommands() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc1_a", "--default_override=1:c0=--test_multiple_string=rc2", "--default_override=1:common=--test_multiple_string=rc2_common", "--default_override=0:c0=--test_multiple_string=rc1_b", "--default_override=0:common=--test_multiple_string=rc1_common", "--rc_source=/somewhere/.blazerc", "--rc_source=/some/other/.blazerc"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /some/other/.blazerc:\n" + " Inherited 'common' options: --test_multiple_string=rc2_common", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " Inherited 'common' options: --test_multiple_string=rc1_common", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc1_a", "Reading rc options for 'c0' from /some/other/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc2", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc1_b"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly("rc2_common", "rc1_common", "rc1_a", "rc2", "rc1_b") .inOrder(); } @Test public void testParseOptions_rcOptionAndExplicit() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString).containsExactly("rc", "explicit").inOrder(); } @Test public void testParseOptions_multiCommandRcOptionAndExplicit() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc_c0_1", "--default_override=0:common=--test_multiple_string=rc_common", "--default_override=0:c0=--test_multiple_string=rc_c0_2", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " Inherited 'common' options: --test_multiple_string=rc_common", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc_c0_1 --test_multiple_string=rc_c0_2"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly("rc_common", "rc_c0_1", "rc_c0_2", "explicit") .inOrder(); } @Test public void testParseOptions_multipleRcsWithMultipleCommandsPlusExplicitOption() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc1_a", "--default_override=1:c0=--test_multiple_string=rc2", "--test_multiple_string=explicit", "--default_override=1:common=--test_multiple_string=rc2_common", "--default_override=0:c0=--test_multiple_string=rc1_b", "--default_override=0:common=--test_multiple_string=rc1_common", "--rc_source=/somewhere/.blazerc", "--rc_source=/some/other/.blazerc"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /some/other/.blazerc:\n" + " Inherited 'common' options: --test_multiple_string=rc2_common", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " Inherited 'common' options: --test_multiple_string=rc1_common", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc1_a", "Reading rc options for 'c0' from /some/other/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc2", "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc1_b"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly("rc2_common", "rc1_common", "rc1_a", "rc2", "rc1_b", "explicit") .inOrder(); } @Test public void testParseOptions_explicitConfig() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:conf=--test_multiple_string=config", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit", "--config=conf"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --test_multiple_string=rc", "Found applicable config definition c0:conf in file /somewhere/.blazerc: " + "--test_multiple_string=config"); // "config" is expanded from --config=conf, which occurs last. TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString).containsExactly("rc", "explicit", "config").inOrder(); } @Test public void testParseOptions_rcSpecifiedConfig() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--config=conf", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:conf=--test_multiple_string=config", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --config=conf --test_multiple_string=rc", "Found applicable config definition c0:conf in file /somewhere/.blazerc: " + "--test_multiple_string=config"); // "config" is expanded from --config=conf, which occurs before the explicit mention of "rc". TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString).containsExactly("config", "rc", "explicit").inOrder(); } @Test public void testParseOptions_recursiveConfig() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--config=conf", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:other=--test_multiple_string=other", "--default_override=0:c0:conf=--test_multiple_string=config1", "--default_override=0:c0:conf=--config=other", "--default_override=0:common:other=--test_multiple_string=othercommon", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --config=conf --test_multiple_string=rc", "Found applicable config definition c0:conf in file /somewhere/.blazerc: " + "--test_multiple_string=config1 --config=other", "Found applicable config definition common:other in file /somewhere/.blazerc: " + "--test_multiple_string=othercommon", "Found applicable config definition c0:other in file /somewhere/.blazerc: " + "--test_multiple_string=other"); // The 2nd config, --config=other, is added by --config=conf after conf adds its own value. TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly("config1", "othercommon", "other", "rc", "explicit") .inOrder(); } @Test public void testParseOptions_recursiveConfigWithDifferentTokens() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:other=--test_multiple_string=other", "--default_override=0:c0:conf=--test_multiple_string=config1", "--default_override=0:c0:conf=--config", "--default_override=0:c0:conf=other", "--rc_source=/somewhere/.blazerc", "--config=conf"), eventHandler); assertThat(eventHandler.getEvents()) .containsExactly( Event.error( "In file /somewhere/.blazerc, the definition of config conf expands to another " + "config that either has no value or is not in the form --config=value. For " + "recursive config definitions, please do not provide the value in a " + "separate token, such as in the form '--config value'.")); } @Test public void testParseOptions_complexConfigOrder() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--test_multiple_string=rc1", "--default_override=0:c0=--config=foo", "--default_override=0:c0=--test_multiple_string=rc2", "--default_override=0:common:baz=--test_multiple_string=baz1", "--default_override=0:c0:baz=--test_multiple_string=baz2", "--default_override=0:common:foo=--test_multiple_string=foo1", "--default_override=0:common:foo=--config=bar", "--default_override=0:c0:foo=--test_multiple_string=foo3", "--default_override=0:common:foo=--test_multiple_string=foo2", "--default_override=0:c0:foo=--test_multiple_string=foo4", "--default_override=0:common:bar=--test_multiple_string=bar1", "--default_override=0:c0:bar=--test_multiple_string=bar2", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit1", "--config=baz", "--test_multiple_string=explicit2"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n 'c0' options: " + "--test_multiple_string=rc1 --config=foo --test_multiple_string=rc2", "Found applicable config definition common:foo in file /somewhere/.blazerc: " + "--test_multiple_string=foo1 --config=bar --test_multiple_string=foo2", "Found applicable config definition common:bar in file /somewhere/.blazerc: " + "--test_multiple_string=bar1", "Found applicable config definition c0:bar in file /somewhere/.blazerc: " + "--test_multiple_string=bar2", "Found applicable config definition c0:foo in file /somewhere/.blazerc: " + "--test_multiple_string=foo3 --test_multiple_string=foo4", "Found applicable config definition common:baz in file /somewhere/.blazerc: " + "--test_multiple_string=baz1", "Found applicable config definition c0:baz in file /somewhere/.blazerc: " + "--test_multiple_string=baz2"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly( "rc1", "foo1", "bar1", "bar2", "foo2", "foo3", "foo4", "rc2", "explicit1", "baz1", "baz2", "explicit2") .inOrder(); } @Test public void testParseOptions_repeatSubConfig() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--config=foo", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:foo=--test_multiple_string=foo", "--default_override=0:c0:foo=--config=bar", "--default_override=0:c0:foo=--config=bar", "--default_override=0:c0:bar=--test_multiple_string=bar", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(parser.getResidue()).isEmpty(); assertThat(eventHandler.getEvents()) .containsExactly( Event.warn( "The following configs were expanded more than once: [bar]. For repeatable flags, " + "repeats are counted twice and may lead to unexpected behavior.")); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --config=foo --test_multiple_string=rc", "Found applicable config definition c0:foo in file /somewhere/.blazerc: " + "--test_multiple_string=foo --config=bar --config=bar", "Found applicable config definition c0:bar in file /somewhere/.blazerc: " + "--test_multiple_string=bar", "Found applicable config definition c0:bar in file /somewhere/.blazerc: " + "--test_multiple_string=bar"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); // Bar is repeated, since it was included twice. assertThat(options.testMultipleString) .containsExactly("foo", "bar", "bar", "rc", "explicit") .inOrder(); } @Test public void testParseOptions_repeatConfig() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0:foo=--test_multiple_string=foo", "--default_override=0:c0:foo=--config=bar", "--default_override=0:c0:bar=--test_multiple_string=bar", "--default_override=0:c0:baz=--test_multiple_string=baz", "--rc_source=/somewhere/.blazerc", "--config=foo", "--config=baz", "--config=foo", "--config=bar"), eventHandler); assertThat(parser.getResidue()).isEmpty(); assertThat(eventHandler.getEvents()) .containsExactly( Event.warn( "The following configs were expanded more than once: [foo, bar]. For repeatable " + "flags, repeats are counted twice and may lead to unexpected behavior.")); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Found applicable config definition c0:foo in file /somewhere/.blazerc: " + "--test_multiple_string=foo --config=bar", "Found applicable config definition c0:bar in file /somewhere/.blazerc: " + "--test_multiple_string=bar", "Found applicable config definition c0:baz in file /somewhere/.blazerc: " + "--test_multiple_string=baz", "Found applicable config definition c0:foo in file /somewhere/.blazerc: " + "--test_multiple_string=foo --config=bar", "Found applicable config definition c0:bar in file /somewhere/.blazerc: " + "--test_multiple_string=bar", "Found applicable config definition c0:bar in file /somewhere/.blazerc: " + "--test_multiple_string=bar"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); // Bar is repeated, since it was included twice. assertThat(options.testMultipleString) .containsExactly("foo", "bar", "baz", "foo", "bar", "bar") .inOrder(); } @Test public void testParseOptions_configCycleLength1() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--config=foo", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:foo=--test_multiple_string=foo", "--default_override=0:c0:foo=--config=foo", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(eventHandler.getEvents()) .contains( Event.error( "Config expansion has a cycle: config value foo expands to itself, see " + "inheritance chain [foo]")); } @Test public void testParseOptions_configCycleLength2() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--config=foo", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:foo=--test_multiple_string=foo", "--default_override=0:c0:foo=--config=bar", "--default_override=0:c0:bar=--test_multiple_string=bar", "--default_override=0:c0:bar=--config=foo", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(eventHandler.getEvents()) .contains( Event.error( "Config expansion has a cycle: config value foo expands to itself, see " + "inheritance chain [foo, bar]")); } @Test public void testParseOptions_recursiveConfigWasAlreadyPresent() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0=--config=other", "--default_override=0:c0=--config=conf", "--default_override=0:c0=--test_multiple_string=rc", "--default_override=0:c0:other=--test_multiple_string=other", "--default_override=0:c0:conf=--test_multiple_string=config1", "--default_override=0:c0:conf=--config=other", "--default_override=0:common:other=--test_multiple_string=othercommon", "--rc_source=/somewhere/.blazerc", "--test_multiple_string=explicit"), eventHandler); assertThat(parser.getResidue()).isEmpty(); assertThat(eventHandler.getEvents()) .containsExactly( Event.warn( "The following configs were expanded more than once: [other]. For repeatable " + "flags, repeats are counted twice and may lead to unexpected behavior.")); // The 2nd config, --config=other, is expanded twice at the same time as --config=conf, // both initially present. The "common" definition is therefore first. other is expanded twice. assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Reading rc options for 'c0' from /somewhere/.blazerc:\n" + " 'c0' options: --config=other --config=conf --test_multiple_string=rc", "Found applicable config definition common:other in file /somewhere/.blazerc: " + "--test_multiple_string=othercommon", "Found applicable config definition c0:other in file /somewhere/.blazerc: " + "--test_multiple_string=other", "Found applicable config definition c0:conf in file /somewhere/.blazerc: " + "--test_multiple_string=config1 --config=other", "Found applicable config definition common:other in file /somewhere/.blazerc: " + "--test_multiple_string=othercommon", "Found applicable config definition c0:other in file /somewhere/.blazerc: " + "--test_multiple_string=other"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly( "othercommon", "other", "config1", "othercommon", "other", "rc", "explicit") .inOrder(); } private static final ImmutableList GREEK_ALPHABET_CHAIN = ImmutableList.of( "--default_override=0:c0:alpha=--test_multiple_string=alpha", "--default_override=0:c0:alpha=--config=beta", "--default_override=0:c0:beta=--test_multiple_string=beta", "--default_override=0:c0:beta=--config=gamma", "--default_override=0:c0:gamma=--test_multiple_string=gamma", "--default_override=0:c0:gamma=--config=delta", "--default_override=0:c0:delta=--test_multiple_string=delta", "--default_override=0:c0:delta=--config=epsilon", "--default_override=0:c0:epsilon=--test_multiple_string=epsilon", "--default_override=0:c0:epsilon=--config=zeta", "--default_override=0:c0:zeta=--test_multiple_string=zeta", "--default_override=0:c0:zeta=--config=eta", "--default_override=0:c0:eta=--test_multiple_string=eta", "--default_override=0:c0:eta=--config=theta", "--default_override=0:c0:theta=--test_multiple_string=theta", "--default_override=0:c0:theta=--config=iota", "--default_override=0:c0:iota=--test_multiple_string=iota", "--default_override=0:c0:iota=--config=kappa", "--default_override=0:c0:kappa=--test_multiple_string=kappa", "--default_override=0:c0:kappa=--config=lambda", "--default_override=0:c0:lambda=--test_multiple_string=lambda", "--default_override=0:c0:lambda=--config=mu", "--default_override=0:c0:mu=--test_multiple_string=mu"); @Test public void testParseOptions_longChain() { ImmutableList args = ImmutableList.builder() .add("c0") .addAll(GREEK_ALPHABET_CHAIN) .add("--rc_source=/somewhere/.blazerc") .add("--config=alpha") .build(); optionHandler.parseOptions(args, eventHandler); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Found applicable config definition c0:alpha in file /somewhere/.blazerc: " + "--test_multiple_string=alpha --config=beta", "Found applicable config definition c0:beta in file /somewhere/.blazerc: " + "--test_multiple_string=beta --config=gamma", "Found applicable config definition c0:gamma in file /somewhere/.blazerc: " + "--test_multiple_string=gamma --config=delta", "Found applicable config definition c0:delta in file /somewhere/.blazerc: " + "--test_multiple_string=delta --config=epsilon", "Found applicable config definition c0:epsilon in file /somewhere/.blazerc: " + "--test_multiple_string=epsilon --config=zeta", "Found applicable config definition c0:zeta in file /somewhere/.blazerc: " + "--test_multiple_string=zeta --config=eta", "Found applicable config definition c0:eta in file /somewhere/.blazerc: " + "--test_multiple_string=eta --config=theta", "Found applicable config definition c0:theta in file /somewhere/.blazerc: " + "--test_multiple_string=theta --config=iota", "Found applicable config definition c0:iota in file /somewhere/.blazerc: " + "--test_multiple_string=iota --config=kappa", "Found applicable config definition c0:kappa in file /somewhere/.blazerc: " + "--test_multiple_string=kappa --config=lambda", "Found applicable config definition c0:lambda in file /somewhere/.blazerc: " + "--test_multiple_string=lambda --config=mu", "Found applicable config definition c0:mu in file /somewhere/.blazerc: " + "--test_multiple_string=mu"); TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly( "alpha", "beta", "gamma", "delta", "epsilon", "zeta", "eta", "theta", "iota", "kappa", "lambda", "mu") .inOrder(); // Expect only one warning, we don't want multiple warnings for the same chain. assertThat(eventHandler.getEvents()) .containsExactly( Event.warn( "There is a recursive chain of configs 12 configs long: [alpha, beta, gamma, " + "delta, epsilon, zeta, eta, theta, iota, kappa, lambda, mu]. This seems " + "excessive, and might be hiding errors.")); } @Test public void testParseOptions_2LongChains() { ImmutableList args = ImmutableList.builder() .add("c0") .addAll(GREEK_ALPHABET_CHAIN) .add("--rc_source=/somewhere/.blazerc") .add("--config=alpha") .add("--config=gamma") .build(); optionHandler.parseOptions(args, eventHandler); assertThat(parser.getResidue()).isEmpty(); // Expect the second --config=gamma to have started a second chain, and get warnings about both. TestOptions options = parser.getOptions(TestOptions.class); assertThat(options).isNotNull(); assertThat(options.testMultipleString) .containsExactly( "alpha", "beta", "gamma", "delta", "epsilon", "zeta", "eta", "theta", "iota", "kappa", "lambda", "mu", "gamma", "delta", "epsilon", "zeta", "eta", "theta", "iota", "kappa", "lambda", "mu") .inOrder(); assertThat(eventHandler.getEvents()) .containsExactly( Event.warn( "There is a recursive chain of configs 12 configs long: [alpha, beta, gamma, " + "delta, epsilon, zeta, eta, theta, iota, kappa, lambda, mu]. This seems " + "excessive, and might be hiding errors."), Event.warn( "There is a recursive chain of configs 10 configs long: [gamma, delta, epsilon, " + "zeta, eta, theta, iota, kappa, lambda, mu]. This seems excessive, " + "and might be hiding errors."), Event.warn( "The following configs were expanded more than once: [gamma, delta, epsilon, zeta, " + "eta, theta, iota, kappa, lambda, mu]. For repeatable flags, repeats are " + "counted twice and may lead to unexpected behavior.")); } @Test public void testWarningFlag() { optionHandler.parseOptions( ImmutableList.of( "c0", "--unconditional_warning", "You are forcing this warning to print for no apparent reason"), eventHandler); assertThat(eventHandler.getEvents()) .containsExactly( Event.warn("You are forcing this warning to print for no apparent reason")); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()).isEmpty(); } @Test public void testWarningFlag_byConfig_notTriggered() { optionHandler.parseOptions( ImmutableList.of( "c0", "--default_override=0:c0:conf=--unconditional_warning=" + "config \"conf\" is deprecated, please stop using!", "--rc_source=/somewhere/.blazerc"), eventHandler); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()).isEmpty(); } @Test public void testWarningFlag_byConfig_triggered() { optionHandler.parseOptions( ImmutableList.of( "c0", "--config=conf", "--default_override=0:c0:conf=--unconditional_warning=" + "config \"conf\" is deprecated, please stop using!", "--rc_source=/somewhere/.blazerc"), eventHandler); assertThat(eventHandler.getEvents()) .containsExactly(Event.warn("config \"conf\" is deprecated, please stop using!")); assertThat(parser.getResidue()).isEmpty(); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Found applicable config definition c0:conf in file /somewhere/.blazerc: " + "--unconditional_warning=config \"conf\" is deprecated, please stop using!"); } @Test public void testConfigAfterExplicit() { optionHandler.parseOptions( ImmutableList.of( "c0", "--test_string=explicitValue", "--config=conf", "--default_override=0:c0:conf=--test_string=fromConf", "--rc_source=/somewhere/.blazerc"), eventHandler); TestOptions parseResult = parser.getOptions(TestOptions.class); // In the in-place expansion, the config's expansion has precedence, but issues a warning since // users might not know that their explicit value was overridden. assertThat(eventHandler.getEvents()) .containsExactly( Event.warn( "option '--config=conf' (source command line options) was expanded and now " + "overrides the explicit option --test_string=explicitValue with " + "--test_string=fromConf")); assertThat(parseResult.testString).isEqualTo("fromConf"); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Found applicable config definition c0:conf in file /somewhere/.blazerc: " + "--test_string=fromConf"); } @Test public void testExplicitOverridesConfig() { optionHandler.parseOptions( ImmutableList.of( "c0", "--config=conf", "--test_string=explicitValue", "--default_override=0:c0:conf=--test_string=fromConf", "--rc_source=/somewhere/.blazerc"), eventHandler); TestOptions parseResult = parser.getOptions(TestOptions.class); assertThat(eventHandler.getEvents()).isEmpty(); assertThat(parseResult.testString).isEqualTo("explicitValue"); assertThat(optionHandler.getRcfileNotes()) .containsExactly( "Found applicable config definition c0:conf in file /somewhere/.blazerc: " + "--test_string=fromConf"); } }