diff options
Diffstat (limited to 'content/public/test')
-rw-r--r-- | content/public/test/test_launcher.cc | 230 |
1 files changed, 135 insertions, 95 deletions
diff --git a/content/public/test/test_launcher.cc b/content/public/test/test_launcher.cc index 0e75b52..31dbef2 100644 --- a/content/public/test/test_launcher.cc +++ b/content/public/test/test_launcher.cc @@ -103,33 +103,26 @@ class WrapperTestLauncherDelegate : public base::TestLauncherDelegate { const testing::TestInfo* test_info) OVERRIDE; virtual bool ShouldRunTest(const testing::TestCase* test_case, const testing::TestInfo* test_info) OVERRIDE; - virtual void RunTest( - const testing::TestCase* test_case, - const testing::TestInfo* test_info, - const base::TestLauncherDelegate::TestResultCallback& callback) OVERRIDE; - virtual void RunRemainingTests() OVERRIDE; + virtual void RunTests(base::TestLauncher* test_launcher, + const std::vector<std::string>& test_names) OVERRIDE; + virtual size_t RetryTests( + base::TestLauncher* test_launcher, + const std::vector<std::string>& test_names) OVERRIDE; private: - struct TestInfo { - std::string GetFullName() const { return test_case_name + "." + test_name; } - - std::string test_case_name; - std::string test_name; - std::vector<base::TestLauncherDelegate::TestResultCallback> callbacks; - }; - - // Launches test from |test_info| using |command_line| and parallel launcher. - void DoRunTest(const TestInfo& test_info, const CommandLine& command_line); + void DoRunTest(base::TestLauncher* test_launcher, + const std::string& test_name); - // Launches test named |test_name| using |command_line| and parallel launcher, + // Launches test named |test_name| using parallel launcher, // given result of PRE_ test |pre_test_result|. - void RunDependentTest(const std::string test_name, - const CommandLine& command_line, + void RunDependentTest(base::TestLauncher* test_launcher, + const std::string test_name, const base::TestResult& pre_test_result); // Callback to receive result of a test. void GTestCallback( - const TestInfo& test_info, + base::TestLauncher* test_launcher, + const std::string& test_name, int exit_code, const base::TimeDelta& elapsed_time, bool was_timeout, @@ -146,10 +139,19 @@ class WrapperTestLauncherDelegate : public base::TestLauncherDelegate { base::ParallelTestLauncher parallel_launcher_; - // Store all tests to run before running any of them to properly - // handle PRE_ tests. The map is indexed by test full name (e.g. "A.B"). - typedef std::map<std::string, TestInfo> TestInfoMap; - TestInfoMap tests_to_run_; + // Store dependent test name (map is indexed by full test name). + typedef std::map<std::string, std::string> DependentTestMap; + DependentTestMap dependent_test_map_; + DependentTestMap reverse_dependent_test_map_; + + // Store unique data directory prefix for test names (without PRE_ prefixes). + // PRE_ tests and tests that depend on them must share the same + // data directory. Using test name as directory name leads to too long + // names (exceeding UNIX_PATH_MAX, which creates a problem with + // process_singleton_linux). Create a randomly-named temporary directory + // and keep track of the names so that PRE_ tests can still re-use them. + typedef std::map<std::string, base::FilePath> UserDataDirMap; + UserDataDirMap user_data_dir_map_; // Temporary directory for user data directories. base::ScopedTempDir temp_dir_; @@ -158,7 +160,8 @@ class WrapperTestLauncherDelegate : public base::TestLauncherDelegate { }; void WrapperTestLauncherDelegate::OnTestIterationStarting() { - tests_to_run_.clear(); + dependent_test_map_.clear(); + user_data_dir_map_.clear(); } std::string WrapperTestLauncherDelegate::GetTestNameForFiltering( @@ -188,74 +191,112 @@ bool WrapperTestLauncherDelegate::ShouldRunTest( return true; } -void WrapperTestLauncherDelegate::RunTest( - const testing::TestCase* test_case, - const testing::TestInfo* test_info, - const base::TestLauncherDelegate::TestResultCallback& callback) { - TestInfo run_test_info; - run_test_info.test_case_name = test_case->name(); - run_test_info.test_name = test_info->name(); - run_test_info.callbacks.push_back(callback); - - DCHECK(!ContainsKey(tests_to_run_, run_test_info.GetFullName())); - tests_to_run_[run_test_info.GetFullName()] = run_test_info; -} - -void WrapperTestLauncherDelegate::RunRemainingTests() { - // PRE_ tests and tests that depend on them must share the same - // data directory. Using test name as directory name leads to too long - // names (exceeding UNIX_PATH_MAX, which creates a problem with - // process_singleton_linux). Create a randomly-named temporary directory - // and keep track of the names so that PRE_ tests can still re-use them. - std::map<std::string, base::FilePath> temp_directories; - +void WrapperTestLauncherDelegate::RunTests( + base::TestLauncher* test_launcher, + const std::vector<std::string>& test_names) { // List of tests we can kick off right now, depending on no other tests. - std::vector<std::pair<std::string, CommandLine> > tests_to_run_now; + std::vector<std::string> tests_to_run_now; - for (TestInfoMap::iterator i = tests_to_run_.begin(); - i != tests_to_run_.end(); - ++i) { - const TestInfo& test_info = i->second; + std::set<std::string> test_names_set(test_names.begin(), test_names.end()); + + for (size_t i = 0; i < test_names.size(); i++) { + std::string full_name(test_names[i]); // Make sure PRE_ tests and tests that depend on them share the same // data directory - based it on the test name without prefixes. - std::string test_name_no_pre(RemoveAnyPrePrefixes(test_info.GetFullName())); - if (!ContainsKey(temp_directories, test_name_no_pre)) { + std::string test_name_no_pre(RemoveAnyPrePrefixes(full_name)); + if (!ContainsKey(user_data_dir_map_, test_name_no_pre)) { base::FilePath temp_dir; CHECK(file_util::CreateTemporaryDirInDir( temp_dir_.path(), FILE_PATH_LITERAL("d"), &temp_dir)); - temp_directories[test_name_no_pre] = temp_dir; + user_data_dir_map_[test_name_no_pre] = temp_dir; } - CommandLine new_cmd_line(*CommandLine::ForCurrentProcess()); - CHECK(launcher_delegate_->AdjustChildProcessCommandLine( - &new_cmd_line, temp_directories[test_name_no_pre])); - + size_t dot_pos = full_name.find('.'); + CHECK_NE(dot_pos, std::string::npos); + std::string test_case_name = full_name.substr(0, dot_pos); + std::string test_name = full_name.substr(dot_pos + 1); std::string pre_test_name( - test_info.test_case_name + "." + kPreTestPrefix + test_info.test_name); - if (ContainsKey(tests_to_run_, pre_test_name)) { - tests_to_run_[pre_test_name].callbacks.push_back( - base::Bind(&WrapperTestLauncherDelegate::RunDependentTest, - base::Unretained(this), - test_info.GetFullName(), - new_cmd_line)); + test_case_name + "." + kPreTestPrefix + test_name); + if (ContainsKey(test_names_set, pre_test_name)) { + DCHECK(!ContainsKey(dependent_test_map_, pre_test_name)); + dependent_test_map_[pre_test_name] = full_name; + + DCHECK(!ContainsKey(reverse_dependent_test_map_, full_name)); + reverse_dependent_test_map_[full_name] = pre_test_name; } else { - tests_to_run_now.push_back( - std::make_pair(test_info.GetFullName(), new_cmd_line)); + tests_to_run_now.push_back(full_name); } } - for (size_t i = 0; i < tests_to_run_now.size(); i++) { - const TestInfo& test_info = tests_to_run_[tests_to_run_now[i].first]; - const CommandLine& cmd_line = tests_to_run_now[i].second; - DoRunTest(test_info, cmd_line); + for (size_t i = 0; i < tests_to_run_now.size(); i++) + DoRunTest(test_launcher, tests_to_run_now[i]); +} + +size_t WrapperTestLauncherDelegate::RetryTests( + base::TestLauncher* test_launcher, + const std::vector<std::string>& test_names) { + // List of tests we can kick off right now, depending on no other tests. + std::vector<std::string> tests_to_run_now; + + // We retry at least the tests requested to retry. + std::set<std::string> test_names_set(test_names.begin(), test_names.end()); + + // In the face of PRE_ tests, we need to retry the entire chain of tests, + // from the very first one. + for (size_t i = 0; i < test_names.size(); i++) { + std::string test_name(test_names[i]); + while (ContainsKey(reverse_dependent_test_map_, test_name)) { + test_name = reverse_dependent_test_map_[test_name]; + test_names_set.insert(test_name); + } } + + // Discard user data directories from any previous runs. Start with + // fresh state. + user_data_dir_map_.clear(); + + for (std::set<std::string>::const_iterator i = test_names_set.begin(); + i != test_names_set.end(); + ++i) { + std::string full_name(*i); + + // Make sure PRE_ tests and tests that depend on them share the same + // data directory - based it on the test name without prefixes. + std::string test_name_no_pre(RemoveAnyPrePrefixes(full_name)); + if (!ContainsKey(user_data_dir_map_, test_name_no_pre)) { + base::FilePath temp_dir; + CHECK(file_util::CreateTemporaryDirInDir( + temp_dir_.path(), FILE_PATH_LITERAL("d"), &temp_dir)); + user_data_dir_map_[test_name_no_pre] = temp_dir; + } + + size_t dot_pos = full_name.find('.'); + CHECK_NE(dot_pos, std::string::npos); + std::string test_case_name = full_name.substr(0, dot_pos); + std::string test_name = full_name.substr(dot_pos + 1); + std::string pre_test_name( + test_case_name + "." + kPreTestPrefix + test_name); + if (!ContainsKey(test_names_set, pre_test_name)) + tests_to_run_now.push_back(full_name); + } + + for (size_t i = 0; i < tests_to_run_now.size(); i++) + DoRunTest(test_launcher, tests_to_run_now[i]); + + return test_names_set.size(); } -void WrapperTestLauncherDelegate::DoRunTest(const TestInfo& test_info, - const CommandLine& command_line) { - CommandLine new_cmd_line(command_line.GetProgram()); - CommandLine::SwitchMap switches = command_line.GetSwitches(); +void WrapperTestLauncherDelegate::DoRunTest(base::TestLauncher* test_launcher, + const std::string& test_name) { + std::string test_name_no_pre(RemoveAnyPrePrefixes(test_name)); + + CommandLine cmd_line(*CommandLine::ForCurrentProcess()); + CHECK(launcher_delegate_->AdjustChildProcessCommandLine( + &cmd_line, user_data_dir_map_[test_name_no_pre])); + + CommandLine new_cmd_line(cmd_line.GetProgram()); + CommandLine::SwitchMap switches = cmd_line.GetSwitches(); // Strip out gtest_output flag because otherwise we would overwrite results // of the other tests. @@ -269,55 +310,52 @@ void WrapperTestLauncherDelegate::DoRunTest(const TestInfo& test_info, // Always enable disabled tests. This method is not called with disabled // tests unless this flag was specified to the browser test executable. new_cmd_line.AppendSwitch("gtest_also_run_disabled_tests"); - new_cmd_line.AppendSwitchASCII( - "gtest_filter", - test_info.test_case_name + "." + test_info.test_name); + new_cmd_line.AppendSwitchASCII("gtest_filter", test_name); new_cmd_line.AppendSwitch(kSingleProcessTestsFlag); char* browser_wrapper = getenv("BROWSER_WRAPPER"); - // PRE_ tests and tests that depend on them should share the sequence token - // name, so that they are run serially. - std::string test_name_no_pre = RemoveAnyPrePrefixes( - test_info.test_case_name + "." + test_info.test_name); - parallel_launcher_.LaunchChildGTestProcess( new_cmd_line, browser_wrapper ? browser_wrapper : std::string(), TestTimeouts::action_max_timeout(), base::Bind(&WrapperTestLauncherDelegate::GTestCallback, base::Unretained(this), - test_info)); + test_launcher, + test_name)); } void WrapperTestLauncherDelegate::RunDependentTest( + base::TestLauncher* test_launcher, const std::string test_name, - const CommandLine& command_line, const base::TestResult& pre_test_result) { - const TestInfo& test_info = tests_to_run_[test_name]; if (pre_test_result.status == base::TestResult::TEST_SUCCESS) { // Only run the dependent test if PRE_ test succeeded. - DoRunTest(test_info, command_line); + DoRunTest(test_launcher, test_name); } else { // Otherwise skip the test. base::TestResult test_result; - test_result.test_case_name = test_info.test_case_name; - test_result.test_name = test_info.test_name; + test_result.full_name = test_name; test_result.status = base::TestResult::TEST_SKIPPED; - for (size_t i = 0; i < test_info.callbacks.size(); i++) - test_info.callbacks[i].Run(test_result); + test_launcher->OnTestFinished(test_result); + + if (ContainsKey(dependent_test_map_, test_name)) { + RunDependentTest(test_launcher, + dependent_test_map_[test_name], + test_result); + } } } void WrapperTestLauncherDelegate::GTestCallback( - const TestInfo& test_info, + base::TestLauncher* test_launcher, + const std::string& test_name, int exit_code, const base::TimeDelta& elapsed_time, bool was_timeout, const std::string& output) { base::TestResult result; - result.test_case_name = test_info.test_case_name; - result.test_name = test_info.test_name; + result.full_name = test_name; // TODO(phajdan.jr): Recognize crashes. if (exit_code == 0) @@ -334,8 +372,10 @@ void WrapperTestLauncherDelegate::GTestCallback( fprintf(stdout, "%s", output.c_str()); fflush(stdout); - for (size_t i = 0; i < test_info.callbacks.size(); i++) - test_info.callbacks[i].Run(result); + if (ContainsKey(dependent_test_map_, test_name)) + RunDependentTest(test_launcher, dependent_test_map_[test_name], result); + + test_launcher->OnTestFinished(result); parallel_launcher_.ResetOutputWatchdog(); } |