// 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. #include "src/main/cpp/option_processor.h" #include "src/main/cpp/bazel_startup_options.h" #include "src/main/cpp/blaze_util.h" #include "src/main/cpp/blaze_util_platform.h" #include "src/main/cpp/option_processor-internal.h" #include "src/main/cpp/rc_file.h" #include "src/main/cpp/util/file.h" #include "src/main/cpp/util/file_platform.h" #include "src/main/cpp/util/path.h" #include "src/main/cpp/util/path_platform.h" #include "src/main/cpp/workspace_layout.h" #include "googlemock/include/gmock/gmock.h" #include "googletest/include/gtest/gtest.h" namespace blaze { using ::testing::HasSubstr; using ::testing::MatchesRegex; #if defined(_WIN32) || defined(__CYGWIN__) constexpr const char* kNullDevice = "nul"; #else // Assume POSIX if not Windows. constexpr const char* kNullDevice = "/dev/null"; #endif extern const char* system_bazelrc_path; class RcFileTest : public ::testing::Test { protected: RcFileTest() : workspace_( blaze_util::JoinPath(blaze::GetEnv("TEST_TMPDIR"), "workspace")), cwd_(blaze_util::JoinPath(blaze::GetEnv("TEST_TMPDIR"), "cwd")), binary_dir_( blaze_util::JoinPath(blaze::GetEnv("TEST_TMPDIR"), "bazeldir")), binary_path_(blaze_util::JoinPath(binary_dir_, "bazel")), workspace_layout_(new WorkspaceLayout()), old_system_bazelrc_path_(system_bazelrc_path) {} void SetUp() override { // We modify the global system_bazelrc_path to be a relative path. // This test allows us to verify that the global bazelrc is read correctly, // in the right order relative to the other files. // // However, this does not test the default path of this file, nor does it // test that absolute paths are accepted properly. This is an unfortunate // limitation of our testing - within the sandboxed environment of a test, // we cannot place a file in arbitrary locations. system_bazelrc_path = "bazel.bazelrc"; ASSERT_TRUE(blaze_util::MakeDirectories(workspace_, 0755)); ASSERT_TRUE(blaze_util::MakeDirectories(cwd_, 0755)); ASSERT_TRUE(blaze_util::ChangeDirectory(cwd_)); #if defined(_WIN32) || defined(__CYGWIN__) // GetCwd returns a short path on Windows, so we store this expectation now // to keep assertions sane in the tests. std::string short_cwd; std::string error; ASSERT_TRUE(blaze_util::AsShortWindowsPath(cwd_, &short_cwd, &error)) << error; cwd_ = short_cwd; #endif ASSERT_TRUE(blaze_util::MakeDirectories(binary_dir_, 0755)); option_processor_.reset(new OptionProcessor( workspace_layout_.get(), std::unique_ptr( new BazelStartupOptions(workspace_layout_.get())))); } void TearDown() override { // TODO(bazel-team): The code below deletes all the files in the workspace // and other rc-related directories, but it intentionally skips directories. // As a consequence, there may be empty directories from test to test. // Remove this once blaze_util::DeleteDirectories(path) exists. std::vector files; blaze_util::GetAllFilesUnder(workspace_, &files); for (const std::string& file : files) { blaze_util::UnlinkPath(file); } blaze_util::GetAllFilesUnder(cwd_, &files); for (const std::string& file : files) { blaze_util::UnlinkPath(file); } blaze_util::GetAllFilesUnder(binary_dir_, &files); for (const std::string& file : files) { blaze_util::UnlinkPath(file); } system_bazelrc_path = old_system_bazelrc_path_.c_str(); } bool SetUpSystemRcFile(const std::string& contents, std::string* rcfile_path) const { const std::string system_rc_path = blaze_util::ConvertPath(blaze_util::JoinPath(cwd_, "bazel.bazelrc")); if (blaze_util::WriteFile(contents, system_rc_path, 0755)) { *rcfile_path = system_rc_path; return true; } return false; } bool SetUpWorkspaceRcFile(const std::string& contents, std::string* rcfile_path) const { const std::string workspace_user_rc_path = blaze_util::JoinPath(workspace_, ".bazelrc"); if (blaze_util::WriteFile(contents, workspace_user_rc_path, 0755)) { *rcfile_path = workspace_user_rc_path; return true; } return false; } // TODO(b/36168162): Make it possible to configure the home directory so we // can test --home_rc as well. bool SetUpLegacyMasterRcFileInWorkspace(const std::string& contents, std::string* rcfile_path) const { const std::string tools_dir = blaze_util::JoinPath(workspace_, "tools"); const std::string workspace_rc_path = blaze_util::JoinPath(tools_dir, "bazel.rc"); if (blaze_util::MakeDirectories(tools_dir, 0755) && blaze_util::WriteFile(contents, workspace_rc_path, 0755)) { *rcfile_path = workspace_rc_path; return true; } return false; } bool SetUpLegacyMasterRcFileAlongsideBinary(const std::string& contents, std::string* rcfile_path) const { const std::string binary_rc_path = blaze_util::JoinPath(binary_dir_, "bazel.bazelrc"); if (blaze_util::WriteFile(contents, binary_rc_path, 0755)) { *rcfile_path = binary_rc_path; return true; } return false; } const std::string workspace_; std::string cwd_; const std::string binary_dir_; const std::string binary_path_; const std::unique_ptr workspace_layout_; const std::string old_system_bazelrc_path_; std::unique_ptr option_processor_; }; using GetRcFileTest = RcFileTest; TEST_F(GetRcFileTest, GetRcFilesLoadsAllDefaultBazelrcs) { std::string system_rc; ASSERT_TRUE(SetUpSystemRcFile("", &system_rc)); std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("", &workspace_rc)); const CommandLine cmd_line = CommandLine(binary_path_, {}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); // There should be 2 rc files: the system one and the workspace one. --bazelrc // is not passed and therefore is not relevant. ASSERT_EQ(2, parsed_rcs.size()); const std::deque expected_system_rc_que = {system_rc}; const std::deque expected_workspace_rc_que = {workspace_rc}; EXPECT_EQ(expected_system_rc_que, parsed_rcs[0].get()->sources()); EXPECT_EQ(expected_workspace_rc_que, parsed_rcs[1].get()->sources()); } TEST_F(GetRcFileTest, GetRcFilesRespectsNoSystemRc) { std::string system_rc; ASSERT_TRUE(SetUpSystemRcFile("", &system_rc)); std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("", &workspace_rc)); const CommandLine cmd_line = CommandLine(binary_path_, {"--nosystem_rc"}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); ASSERT_EQ(1, parsed_rcs.size()); const std::deque expected_workspace_rc_que = {workspace_rc}; EXPECT_EQ(expected_workspace_rc_que, parsed_rcs[0].get()->sources()); } TEST_F(GetRcFileTest, GetRcFilesRespectsNoWorkspaceRc) { std::string system_rc; ASSERT_TRUE(SetUpSystemRcFile("", &system_rc)); std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("", &workspace_rc)); const CommandLine cmd_line = CommandLine(binary_path_, {"--noworkspace_rc"}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); ASSERT_EQ(1, parsed_rcs.size()); const std::deque expected_system_rc_que = {system_rc}; EXPECT_EQ(expected_system_rc_que, parsed_rcs[0].get()->sources()); } TEST_F(GetRcFileTest, GetRcFilesRespectsNoWorkspaceRcAndNoSystemCombined) { std::string system_rc; ASSERT_TRUE(SetUpSystemRcFile("", &system_rc)); std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("", &workspace_rc)); const CommandLine cmd_line = CommandLine( binary_path_, {"--noworkspace_rc", "--nosystem_rc"}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); ASSERT_EQ(0, parsed_rcs.size()); } TEST_F(GetRcFileTest, GetRcFilesWarnsAboutIgnoredMasterRcFiles) { std::string workspace_rc; ASSERT_TRUE(SetUpLegacyMasterRcFileInWorkspace("", &workspace_rc)); std::string binary_rc; ASSERT_TRUE(SetUpLegacyMasterRcFileAlongsideBinary("", &binary_rc)); const CommandLine cmd_line = CommandLine(binary_path_, {}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; testing::internal::CaptureStderr(); const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); // Expect that GetRcFiles outputs a warning about these files that are not // read as expected. EXPECT_THAT(output, HasSubstr("The following rc files are no longer being read")); EXPECT_THAT(output, HasSubstr(workspace_rc)); EXPECT_THAT(output, HasSubstr(binary_rc)); } TEST_F( GetRcFileTest, GetRcFilesDoesNotWarnAboutIgnoredMasterRcFilesWhenNoMasterBazelrcIsPassed) { std::string workspace_rc; ASSERT_TRUE(SetUpLegacyMasterRcFileInWorkspace("", &workspace_rc)); std::string binary_rc; ASSERT_TRUE(SetUpLegacyMasterRcFileAlongsideBinary("", &binary_rc)); const CommandLine cmd_line = CommandLine(binary_path_, {"--nomaster_bazelrc"}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; testing::internal::CaptureStderr(); const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); // Expect that nothing is logged to stderr about ignored rc files when these // files are disabled. EXPECT_THAT( output, Not(HasSubstr("The following rc files are no longer being read"))); EXPECT_THAT(output, Not(HasSubstr(workspace_rc))); EXPECT_THAT(output, Not(HasSubstr(binary_rc))); } TEST_F(GetRcFileTest, GetRcFilesReadsCommandLineRc) { const std::string cmdline_rc_path = blaze_util::JoinPath(workspace_, "mybazelrc"); ASSERT_TRUE( blaze_util::MakeDirectories(blaze_util::Dirname(cmdline_rc_path), 0755)); ASSERT_TRUE(blaze_util::WriteFile("", cmdline_rc_path, 0755)); const CommandLine cmd_line = CommandLine( binary_path_, {"--nomaster_bazelrc", "--bazelrc=" + cmdline_rc_path}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); // Because of the variety of path representations in windows, this // equality test does not attempt to check the entire path. ASSERT_EQ(1, parsed_rcs.size()); ASSERT_EQ(1, parsed_rcs[0].get()->sources().size()); EXPECT_THAT(parsed_rcs[0].get()->sources().front(), HasSubstr("mybazelrc")); } TEST_F(GetRcFileTest, GetRcFilesAcceptsNullCommandLineRc) { const CommandLine cmd_line = CommandLine(binary_path_, {"--nosystem_rc", "--noworkspace_rc", "--nohome_rc", "--bazelrc=/dev/null"}, "build", {}); std::string error = "check that this string is not modified"; std::vector> parsed_rcs; const blaze_exit_code::ExitCode exit_code = option_processor_->GetRcFiles(workspace_layout_.get(), workspace_, cwd_, &cmd_line, &parsed_rcs, &error); // /dev/null is not an error EXPECT_EQ(blaze_exit_code::SUCCESS, exit_code); EXPECT_EQ("check that this string is not modified", error); // but it does technically count as a file ASSERT_EQ(1, parsed_rcs.size()); const std::deque expected_rc_que = {kNullDevice}; EXPECT_EQ(expected_rc_que, parsed_rcs[0].get()->sources()); } using ParseOptionsTest = RcFileTest; TEST_F(ParseOptionsTest, IgnoreAllRcFilesIgnoresAllMasterAndUserRcFiles) { // Put fake options in different expected rc files, to check that none of them // are read. std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("startup --workspacefoo", &workspace_rc)); std::string system_rc; ASSERT_TRUE(SetUpSystemRcFile("startup --systemfoo", &system_rc)); const std::string cmdline_rc_path = blaze_util::JoinPath(workspace_, "mybazelrc"); ASSERT_TRUE( blaze_util::MakeDirectories(blaze_util::Dirname(cmdline_rc_path), 0755)); ASSERT_TRUE(blaze_util::WriteFile("startup --myfoo", cmdline_rc_path, 0755)); const std::vector args = {binary_path_, "--ignore_all_rc_files", "build"}; // Expect no error due to the incorrect options, as non of them should have // been loaded. std::string error; EXPECT_EQ(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)); ASSERT_EQ("", error); // Check that the startup options' provenance message contains nothing testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_EQ(output, ""); } TEST_F(ParseOptionsTest, LaterIgnoreAllRcFilesValueWins) { std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("startup --workspacefoo", &workspace_rc)); const std::vector args = {binary_path_, "--ignore_all_rc_files", "--noignore_all_rc_files", "build"}; std::string error; EXPECT_EQ(blaze_exit_code::BAD_ARGV, option_processor_->ParseOptions(args, workspace_, cwd_, &error)); ASSERT_EQ( "Unknown startup option: '--workspacefoo'.\n For more info, run " "'bazel help startup_options'.", error); // Check that the startup options' provenance message contains the provenance // of the incorrect option. testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_THAT( output, MatchesRegex("INFO: Reading 'startup' options from .*workspace.*bazelrc: " "--workspacefoo\n")); } TEST_F(ParseOptionsTest, IgnoreAllRcFilesIgnoresCommandLineRcFileToo) { // Put fake options in different expected rc files, to check that none of them // are read. std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("startup --workspacefoo", &workspace_rc)); std::string system_rc; ASSERT_TRUE(SetUpSystemRcFile("startup --systemfoo", &system_rc)); const std::string cmdline_rc_path = blaze_util::JoinPath(workspace_, "mybazelrc"); ASSERT_TRUE( blaze_util::MakeDirectories(blaze_util::Dirname(cmdline_rc_path), 0755)); ASSERT_TRUE( blaze_util::WriteFile("startup --cmdlinefoo", cmdline_rc_path, 0755)); const std::vector args = {binary_path_, "--ignore_all_rc_files", "--bazelrc=" + cmdline_rc_path, "build"}; // Expect no error due to the incorrect options, as non of them should have // been loaded. std::string error; EXPECT_EQ(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)); ASSERT_EQ("", error); // Check that the startup options' provenance message contains nothing testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_EQ(output, ""); } TEST_F(ParseOptionsTest, CommandLineBazelrcHasUnknownOption) { const std::string cmdline_rc_path = blaze_util::JoinPath(workspace_, "mybazelrc"); ASSERT_TRUE( blaze_util::MakeDirectories(blaze_util::Dirname(cmdline_rc_path), 0755)); ASSERT_TRUE(blaze_util::WriteFile("startup --foo", cmdline_rc_path, 0755)); const std::vector args = {binary_path_, "--nomaster_bazelrc", "--bazelrc=" + cmdline_rc_path, "build"}; const std::string expected_error = "Unknown startup option: '--foo'.\n" " For more info, run 'bazel help startup_options'."; std::string error; ASSERT_NE(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)) << error; ASSERT_EQ(expected_error, error); // Check that the startup options' provenance message contains the correct // information for the incorrect flag, and does not print the command-line // provided startup flags. testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_THAT(output, MatchesRegex( "INFO: Reading 'startup' options from .*mybazelrc: --foo\n")); } TEST_F(ParseOptionsTest, BazelrcHasUnknownOption) { std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("startup --foo", &workspace_rc)); const std::vector args = {binary_path_, "build"}; // Expect no error due to the incorrect --foo. std::string error; EXPECT_EQ(blaze_exit_code::BAD_ARGV, option_processor_->ParseOptions(args, workspace_, cwd_, &error)); ASSERT_EQ( "Unknown startup option: '--foo'.\n" " For more info, run 'bazel help startup_options'.", error); // Check that the startup options' provenance message contains nothing for the // master bazelrc. testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_THAT(output, MatchesRegex("INFO: Reading 'startup' options from " ".*workspace.*bazelrc: --foo\n")); } TEST_F(ParseOptionsTest, IncorrectWorkspaceBazelrcIgnoredWhenNoWorkspaceRcIsPresent) { std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("startup --foo", &workspace_rc)); const std::vector args = {binary_path_, "--noworkspace_rc", "build"}; // Expect no error due to the incorrect --foo. std::string error; EXPECT_EQ(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)); ASSERT_EQ("", error); // Check that the startup options' provenance message contains nothing for the // master bazelrc. testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_EQ(output, ""); } TEST_F(ParseOptionsTest, PositiveOptionOverridesNegativeOption) { std::string workspace_rc; ASSERT_TRUE( SetUpWorkspaceRcFile("startup --max_idle_secs=123", &workspace_rc)); const std::vector args = {"bazel", "--noworkspace_rc", "--workspace_rc", "build"}; std::string error; ASSERT_EQ(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)) << error; EXPECT_EQ(123, option_processor_->GetParsedStartupOptions()->max_idle_secs); // Check that the startup options' provenance message contains the correct // information for the master bazelrc. testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_THAT(output, MatchesRegex("INFO: Reading 'startup' options from " ".*workspace.*bazelrc: --max_idle_secs=123\n")); } TEST_F(ParseOptionsTest, MultipleStartupArgsInMasterBazelrcWorksCorrectly) { // Add startup flags to the master bazelrc. std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile( "startup --max_idle_secs=42\nstartup --io_nice_level=6", &workspace_rc)); const std::vector args = {binary_path_, "build"}; std::string error; ASSERT_EQ(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)) << error; EXPECT_EQ(42, option_processor_->GetParsedStartupOptions()->max_idle_secs); EXPECT_EQ(6, option_processor_->GetParsedStartupOptions()->io_nice_level); // Check that the startup options get grouped together properly in the output // message. testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_THAT( output, MatchesRegex("INFO: Reading 'startup' options from .*workspace.*bazelrc: " "--max_idle_secs=42 --io_nice_level=6\n")); } TEST_F(ParseOptionsTest, CommandLineBazelrcHasPriorityOverDefaultBazelrc) { // Add startup flags to the workspace bazelrc. std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile( "startup --max_idle_secs=42\nstartup --io_nice_level=6", &workspace_rc)); // Override one of the master bazelrc's flags in the commandline rc. const std::string cmdline_rc_path = blaze_util::JoinPath(workspace_, "mybazelrc"); ASSERT_TRUE( blaze_util::MakeDirectories(blaze_util::Dirname(cmdline_rc_path), 0755)); ASSERT_TRUE(blaze_util::WriteFile("startup --max_idle_secs=123", cmdline_rc_path, 0755)); const std::vector args = { "bazel", "--bazelrc=" + cmdline_rc_path, "build"}; std::string error; ASSERT_EQ(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)) << error; EXPECT_EQ(123, option_processor_->GetParsedStartupOptions()->max_idle_secs); EXPECT_EQ(6, option_processor_->GetParsedStartupOptions()->io_nice_level); // Check that the options are reported in the correct order in the provenance // message. testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_THAT( output, MatchesRegex("INFO: Reading 'startup' options from .*workspace.*bazelrc: " "--max_idle_secs=42 --io_nice_level=6\n" "INFO: Reading 'startup' options from .*mybazelrc: " "--max_idle_secs=123\n")); } TEST_F(ParseOptionsTest, BazelRcImportsMaintainsFlagOrdering) { // Override one of the master bazelrc's flags in the custom bazelrc. const std::string imported_rc_path = blaze_util::JoinPath(workspace_, "myimportedbazelrc"); ASSERT_TRUE( blaze_util::MakeDirectories(blaze_util::Dirname(imported_rc_path), 0755)); ASSERT_TRUE(blaze_util::WriteFile( "startup --max_idle_secs=123\nstartup --io_nice_level=4", imported_rc_path, 0755)); // Add startup flags the imported bazelrc. std::string workspace_rc; ASSERT_TRUE(SetUpWorkspaceRcFile("startup --max_idle_secs=42\nimport " + imported_rc_path + "\nstartup --io_nice_level=6", &workspace_rc)); const std::vector args = {"bazel", "build"}; std::string error; ASSERT_EQ(blaze_exit_code::SUCCESS, option_processor_->ParseOptions(args, workspace_, cwd_, &error)) << error; EXPECT_EQ(123, option_processor_->GetParsedStartupOptions()->max_idle_secs); EXPECT_EQ(6, option_processor_->GetParsedStartupOptions()->io_nice_level); // Check that the options are reported in the correct order in the provenance // message, the imported file between the two master flags testing::internal::CaptureStderr(); option_processor_->PrintStartupOptionsProvenanceMessage(); const std::string& output = testing::internal::GetCapturedStderr(); EXPECT_THAT( output, MatchesRegex("INFO: Reading 'startup' options from .*workspace.*bazelrc: " "--max_idle_secs=42\n" "INFO: Reading 'startup' options from .*myimportedbazelrc: " "--max_idle_secs=123 --io_nice_level=4\n" "INFO: Reading 'startup' options from .*workspace.*bazelrc: " "--io_nice_level=6\n")); } } // namespace blaze