1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "base/test/launcher/test_results_tracker.h"
6 
7 #include <stddef.h>
8 
9 #include <memory>
10 #include <utility>
11 
12 #include "base/base64.h"
13 #include "base/command_line.h"
14 #include "base/files/file.h"
15 #include "base/files/file_path.h"
16 #include "base/files/file_util.h"
17 #include "base/format_macros.h"
18 #include "base/json/json_writer.h"
19 #include "base/json/string_escape.h"
20 #include "base/logging.h"
21 #include "base/strings/string_util.h"
22 #include "base/strings/stringprintf.h"
23 #include "base/test/gtest_util.h"
24 #include "base/test/launcher/test_launcher.h"
25 #include "base/time/time.h"
26 #include "base/values.h"
27 
28 namespace base {
29 
30 namespace {
31 
32 // The default output file for XML output.
33 const FilePath::CharType kDefaultOutputFile[] = FILE_PATH_LITERAL(
34     "test_detail.xml");
35 
36 // Converts the given epoch time in milliseconds to a date string in the ISO
37 // 8601 format, without the timezone information.
38 // TODO(xyzzyz): Find a good place in Chromium to put it and refactor all uses
39 // to point to it.
FormatTimeAsIso8601(Time time)40 std::string FormatTimeAsIso8601(Time time) {
41   Time::Exploded exploded;
42   time.UTCExplode(&exploded);
43   return StringPrintf("%04d-%02d-%02dT%02d:%02d:%02d",
44                       exploded.year,
45                       exploded.month,
46                       exploded.day_of_month,
47                       exploded.hour,
48                       exploded.minute,
49                       exploded.second);
50 }
51 
52 struct TestSuiteResultsAggregator {
TestSuiteResultsAggregatorbase::__anon75e886bf0111::TestSuiteResultsAggregator53   TestSuiteResultsAggregator()
54       : tests(0), failures(0), disabled(0), errors(0) {}
55 
Addbase::__anon75e886bf0111::TestSuiteResultsAggregator56   void Add(const TestResult& result) {
57     tests++;
58     elapsed_time += result.elapsed_time;
59 
60     switch (result.status) {
61       case TestResult::TEST_SUCCESS:
62         break;
63       case TestResult::TEST_FAILURE:
64         failures++;
65         break;
66       case TestResult::TEST_EXCESSIVE_OUTPUT:
67       case TestResult::TEST_FAILURE_ON_EXIT:
68       case TestResult::TEST_TIMEOUT:
69       case TestResult::TEST_CRASH:
70       case TestResult::TEST_UNKNOWN:
71         errors++;
72         break;
73       case TestResult::TEST_SKIPPED:
74         disabled++;
75         break;
76     }
77   }
78 
79   int tests;
80   int failures;
81   int disabled;
82   int errors;
83 
84   TimeDelta elapsed_time;
85 };
86 
87 }  // namespace
88 
TestResultsTracker()89 TestResultsTracker::TestResultsTracker() : iteration_(-1), out_(nullptr) {}
90 
~TestResultsTracker()91 TestResultsTracker::~TestResultsTracker() {
92   DCHECK(thread_checker_.CalledOnValidThread());
93 
94   if (!out_)
95     return;
96 
97   // Maps test case names to test results.
98   typedef std::map<std::string, std::vector<TestResult> > TestCaseMap;
99   TestCaseMap test_case_map;
100 
101   TestSuiteResultsAggregator all_tests_aggregator;
102   for (const PerIterationData::ResultsMap::value_type& i
103            : per_iteration_data_[iteration_].results) {
104     // Use the last test result as the final one.
105     TestResult result = i.second.test_results.back();
106     test_case_map[result.GetTestCaseName()].push_back(result);
107     all_tests_aggregator.Add(result);
108   }
109 
110   fprintf(out_, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
111   fprintf(out_,
112           "<testsuites name=\"AllTests\" tests=\"%d\" failures=\"%d\""
113           " disabled=\"%d\" errors=\"%d\" time=\"%.3f\" timestamp=\"%s\">\n",
114           all_tests_aggregator.tests, all_tests_aggregator.failures,
115           all_tests_aggregator.disabled, all_tests_aggregator.errors,
116           all_tests_aggregator.elapsed_time.InSecondsF(),
117           FormatTimeAsIso8601(Time::Now()).c_str());
118 
119   for (const TestCaseMap::value_type& i : test_case_map) {
120     const std::string testsuite_name = i.first;
121     const std::vector<TestResult>& results = i.second;
122 
123     TestSuiteResultsAggregator aggregator;
124     for (const TestResult& result : results) {
125       aggregator.Add(result);
126     }
127     fprintf(out_,
128             "  <testsuite name=\"%s\" tests=\"%d\" "
129             "failures=\"%d\" disabled=\"%d\" errors=\"%d\" time=\"%.3f\" "
130             "timestamp=\"%s\">\n",
131             testsuite_name.c_str(), aggregator.tests, aggregator.failures,
132             aggregator.disabled, aggregator.errors,
133             aggregator.elapsed_time.InSecondsF(),
134             FormatTimeAsIso8601(Time::Now()).c_str());
135 
136     for (const TestResult& result : results) {
137       fprintf(out_, "    <testcase name=\"%s\" status=\"run\" time=\"%.3f\""
138               " classname=\"%s\">\n",
139               result.GetTestName().c_str(),
140               result.elapsed_time.InSecondsF(),
141               result.GetTestCaseName().c_str());
142       if (result.status != TestResult::TEST_SUCCESS) {
143         // The actual failure message is not propagated up to here, as it's too
144         // much work to escape it properly, and in case of failure, almost
145         // always one needs to look into full log anyway.
146         fprintf(out_, "      <failure message=\"\" type=\"\"></failure>\n");
147       }
148       fprintf(out_, "    </testcase>\n");
149     }
150     fprintf(out_, "  </testsuite>\n");
151   }
152 
153   fprintf(out_, "</testsuites>\n");
154   fclose(out_);
155 }
156 
Init(const CommandLine & command_line)157 bool TestResultsTracker::Init(const CommandLine& command_line) {
158   DCHECK(thread_checker_.CalledOnValidThread());
159 
160   // Prevent initializing twice.
161   if (out_) {
162     NOTREACHED();
163     return false;
164   }
165 
166   if (!command_line.HasSwitch(kGTestOutputFlag))
167     return true;
168 
169   std::string flag = command_line.GetSwitchValueASCII(kGTestOutputFlag);
170   size_t colon_pos = flag.find(':');
171   FilePath path;
172   if (colon_pos != std::string::npos) {
173     FilePath flag_path =
174         command_line.GetSwitchValuePath(kGTestOutputFlag);
175     FilePath::StringType path_string = flag_path.value();
176     path = FilePath(path_string.substr(colon_pos + 1));
177     // If the given path ends with '/', consider it is a directory.
178     // Note: This does NOT check that a directory (or file) actually exists
179     // (the behavior is same as what gtest does).
180     if (path.EndsWithSeparator()) {
181       FilePath executable = command_line.GetProgram().BaseName();
182       path = path.Append(executable.ReplaceExtension(
183                              FilePath::StringType(FILE_PATH_LITERAL("xml"))));
184     }
185   }
186   if (path.value().empty())
187     path = FilePath(kDefaultOutputFile);
188   FilePath dir_name = path.DirName();
189   if (!DirectoryExists(dir_name)) {
190     LOG(WARNING) << "The output directory does not exist. "
191                  << "Creating the directory: " << dir_name.value();
192     // Create the directory if necessary (because the gtest does the same).
193     if (!CreateDirectory(dir_name)) {
194       LOG(ERROR) << "Failed to created directory " << dir_name.value();
195       return false;
196     }
197   }
198   out_ = OpenFile(path, "w");
199   if (!out_) {
200     LOG(ERROR) << "Cannot open output file: "
201                << path.value() << ".";
202     return false;
203   }
204 
205   return true;
206 }
207 
OnTestIterationStarting()208 void TestResultsTracker::OnTestIterationStarting() {
209   DCHECK(thread_checker_.CalledOnValidThread());
210 
211   // Start with a fresh state for new iteration.
212   iteration_++;
213   per_iteration_data_.push_back(PerIterationData());
214 }
215 
AddTest(const std::string & test_name)216 void TestResultsTracker::AddTest(const std::string& test_name) {
217   // Record disabled test names without DISABLED_ prefix so that they are easy
218   // to compare with regular test names, e.g. before or after disabling.
219   all_tests_.insert(TestNameWithoutDisabledPrefix(test_name));
220 }
221 
AddDisabledTest(const std::string & test_name)222 void TestResultsTracker::AddDisabledTest(const std::string& test_name) {
223   // Record disabled test names without DISABLED_ prefix so that they are easy
224   // to compare with regular test names, e.g. before or after disabling.
225   disabled_tests_.insert(TestNameWithoutDisabledPrefix(test_name));
226 }
227 
AddTestLocation(const std::string & test_name,const std::string & file,int line)228 void TestResultsTracker::AddTestLocation(const std::string& test_name,
229                                          const std::string& file,
230                                          int line) {
231   test_locations_.insert(std::make_pair(test_name, CodeLocation(file, line)));
232 }
233 
AddTestResult(const TestResult & result)234 void TestResultsTracker::AddTestResult(const TestResult& result) {
235   DCHECK(thread_checker_.CalledOnValidThread());
236 
237   // Record disabled test names without DISABLED_ prefix so that they are easy
238   // to compare with regular test names, e.g. before or after disabling.
239   per_iteration_data_[iteration_].results[
240       TestNameWithoutDisabledPrefix(result.full_name)].test_results.push_back(
241           result);
242 }
243 
PrintSummaryOfCurrentIteration() const244 void TestResultsTracker::PrintSummaryOfCurrentIteration() const {
245   TestStatusMap tests_by_status(GetTestStatusMapForCurrentIteration());
246 
247   PrintTests(tests_by_status[TestResult::TEST_FAILURE].begin(),
248              tests_by_status[TestResult::TEST_FAILURE].end(),
249              "failed");
250   PrintTests(tests_by_status[TestResult::TEST_FAILURE_ON_EXIT].begin(),
251              tests_by_status[TestResult::TEST_FAILURE_ON_EXIT].end(),
252              "failed on exit");
253   PrintTests(tests_by_status[TestResult::TEST_EXCESSIVE_OUTPUT].begin(),
254              tests_by_status[TestResult::TEST_EXCESSIVE_OUTPUT].end(),
255              "produced excessive output");
256   PrintTests(tests_by_status[TestResult::TEST_TIMEOUT].begin(),
257              tests_by_status[TestResult::TEST_TIMEOUT].end(),
258              "timed out");
259   PrintTests(tests_by_status[TestResult::TEST_CRASH].begin(),
260              tests_by_status[TestResult::TEST_CRASH].end(),
261              "crashed");
262   PrintTests(tests_by_status[TestResult::TEST_SKIPPED].begin(),
263              tests_by_status[TestResult::TEST_SKIPPED].end(),
264              "skipped");
265   PrintTests(tests_by_status[TestResult::TEST_UNKNOWN].begin(),
266              tests_by_status[TestResult::TEST_UNKNOWN].end(),
267              "had unknown result");
268 }
269 
PrintSummaryOfAllIterations() const270 void TestResultsTracker::PrintSummaryOfAllIterations() const {
271   DCHECK(thread_checker_.CalledOnValidThread());
272 
273   TestStatusMap tests_by_status(GetTestStatusMapForAllIterations());
274 
275   fprintf(stdout, "Summary of all test iterations:\n");
276   fflush(stdout);
277 
278   PrintTests(tests_by_status[TestResult::TEST_FAILURE].begin(),
279              tests_by_status[TestResult::TEST_FAILURE].end(),
280              "failed");
281   PrintTests(tests_by_status[TestResult::TEST_FAILURE_ON_EXIT].begin(),
282              tests_by_status[TestResult::TEST_FAILURE_ON_EXIT].end(),
283              "failed on exit");
284   PrintTests(tests_by_status[TestResult::TEST_EXCESSIVE_OUTPUT].begin(),
285              tests_by_status[TestResult::TEST_EXCESSIVE_OUTPUT].end(),
286              "produced excessive output");
287   PrintTests(tests_by_status[TestResult::TEST_TIMEOUT].begin(),
288              tests_by_status[TestResult::TEST_TIMEOUT].end(),
289              "timed out");
290   PrintTests(tests_by_status[TestResult::TEST_CRASH].begin(),
291              tests_by_status[TestResult::TEST_CRASH].end(),
292              "crashed");
293   PrintTests(tests_by_status[TestResult::TEST_SKIPPED].begin(),
294              tests_by_status[TestResult::TEST_SKIPPED].end(),
295              "skipped");
296   PrintTests(tests_by_status[TestResult::TEST_UNKNOWN].begin(),
297              tests_by_status[TestResult::TEST_UNKNOWN].end(),
298              "had unknown result");
299 
300   fprintf(stdout, "End of the summary.\n");
301   fflush(stdout);
302 }
303 
AddGlobalTag(const std::string & tag)304 void TestResultsTracker::AddGlobalTag(const std::string& tag) {
305   global_tags_.insert(tag);
306 }
307 
SaveSummaryAsJSON(const FilePath & path,const std::vector<std::string> & additional_tags) const308 bool TestResultsTracker::SaveSummaryAsJSON(
309     const FilePath& path,
310     const std::vector<std::string>& additional_tags) const {
311   std::unique_ptr<DictionaryValue> summary_root(new DictionaryValue);
312 
313   std::unique_ptr<ListValue> global_tags(new ListValue);
314   for (const auto& global_tag : global_tags_) {
315     global_tags->AppendString(global_tag);
316   }
317   for (const auto& tag : additional_tags) {
318     global_tags->AppendString(tag);
319   }
320   summary_root->Set("global_tags", std::move(global_tags));
321 
322   std::unique_ptr<ListValue> all_tests(new ListValue);
323   for (const auto& test : all_tests_) {
324     all_tests->AppendString(test);
325   }
326   summary_root->Set("all_tests", std::move(all_tests));
327 
328   std::unique_ptr<ListValue> disabled_tests(new ListValue);
329   for (const auto& disabled_test : disabled_tests_) {
330     disabled_tests->AppendString(disabled_test);
331   }
332   summary_root->Set("disabled_tests", std::move(disabled_tests));
333 
334   std::unique_ptr<ListValue> per_iteration_data(new ListValue);
335 
336   for (int i = 0; i <= iteration_; i++) {
337     std::unique_ptr<DictionaryValue> current_iteration_data(
338         new DictionaryValue);
339 
340     for (PerIterationData::ResultsMap::const_iterator j =
341              per_iteration_data_[i].results.begin();
342          j != per_iteration_data_[i].results.end();
343          ++j) {
344       std::unique_ptr<ListValue> test_results(new ListValue);
345 
346       for (size_t k = 0; k < j->second.test_results.size(); k++) {
347         const TestResult& test_result = j->second.test_results[k];
348 
349         std::unique_ptr<DictionaryValue> test_result_value(new DictionaryValue);
350 
351         test_result_value->SetString("status", test_result.StatusAsString());
352         test_result_value->SetInteger(
353             "elapsed_time_ms",
354             static_cast<int>(test_result.elapsed_time.InMilliseconds()));
355 
356         bool lossless_snippet = false;
357         if (IsStringUTF8(test_result.output_snippet)) {
358           test_result_value->SetString(
359               "output_snippet", test_result.output_snippet);
360           lossless_snippet = true;
361         } else {
362           test_result_value->SetString(
363               "output_snippet",
364               "<non-UTF-8 snippet, see output_snippet_base64>");
365         }
366 
367         // TODO(phajdan.jr): Fix typo in JSON key (losless -> lossless)
368         // making sure not to break any consumers of this data.
369         test_result_value->SetBoolean("losless_snippet", lossless_snippet);
370 
371         // Also include the raw version (base64-encoded so that it can be safely
372         // JSON-serialized - there are no guarantees about character encoding
373         // of the snippet). This can be very useful piece of information when
374         // debugging a test failure related to character encoding.
375         std::string base64_output_snippet;
376         Base64Encode(test_result.output_snippet, &base64_output_snippet);
377         test_result_value->SetString("output_snippet_base64",
378                                      base64_output_snippet);
379 
380         std::unique_ptr<ListValue> test_result_parts(new ListValue);
381         for (const TestResultPart& result_part :
382              test_result.test_result_parts) {
383           std::unique_ptr<DictionaryValue> result_part_value(
384               new DictionaryValue);
385           result_part_value->SetString("type", result_part.TypeAsString());
386           result_part_value->SetString("file", result_part.file_name);
387           result_part_value->SetInteger("line", result_part.line_number);
388 
389           bool lossless_summary = IsStringUTF8(result_part.summary);
390           if (lossless_summary) {
391             result_part_value->SetString("summary", result_part.summary);
392           } else {
393             result_part_value->SetString(
394                 "summary", "<non-UTF-8 snippet, see summary_base64>");
395           }
396           result_part_value->SetBoolean("lossless_summary", lossless_summary);
397 
398           std::string encoded_summary;
399           Base64Encode(result_part.summary, &encoded_summary);
400           result_part_value->SetString("summary_base64", encoded_summary);
401 
402           bool lossless_message = IsStringUTF8(result_part.message);
403           if (lossless_message) {
404             result_part_value->SetString("message", result_part.message);
405           } else {
406             result_part_value->SetString(
407                 "message", "<non-UTF-8 snippet, see message_base64>");
408           }
409           result_part_value->SetBoolean("lossless_message", lossless_message);
410 
411           std::string encoded_message;
412           Base64Encode(result_part.message, &encoded_message);
413           result_part_value->SetString("message_base64", encoded_message);
414 
415           test_result_parts->Append(std::move(result_part_value));
416         }
417         test_result_value->Set("result_parts", std::move(test_result_parts));
418 
419         test_results->Append(std::move(test_result_value));
420       }
421 
422       current_iteration_data->SetWithoutPathExpansion(j->first,
423                                                       std::move(test_results));
424     }
425     per_iteration_data->Append(std::move(current_iteration_data));
426   }
427   summary_root->Set("per_iteration_data", std::move(per_iteration_data));
428 
429   std::unique_ptr<DictionaryValue> test_locations(new DictionaryValue);
430   for (const auto& item : test_locations_) {
431     std::string test_name = item.first;
432     CodeLocation location = item.second;
433     std::unique_ptr<DictionaryValue> location_value(new DictionaryValue);
434     location_value->SetString("file", location.file);
435     location_value->SetInteger("line", location.line);
436     test_locations->SetWithoutPathExpansion(test_name,
437                                             std::move(location_value));
438   }
439   summary_root->Set("test_locations", std::move(test_locations));
440 
441   std::string json;
442   if (!JSONWriter::Write(*summary_root, &json))
443     return false;
444 
445   File output(path, File::FLAG_CREATE_ALWAYS | File::FLAG_WRITE);
446   if (!output.IsValid())
447     return false;
448 
449   int json_size = static_cast<int>(json.size());
450   if (output.WriteAtCurrentPos(json.data(), json_size) != json_size) {
451     return false;
452   }
453 
454   // File::Flush() will call fsync(). This is important on Fuchsia to ensure
455   // that the file is written to the disk - the system running under qemu will
456   // shutdown shortly after the test completes. On Fuchsia fsync() times out
457   // after 15 seconds. Apparently this may not be enough in some cases,
458   // particularly when running net_unittests on buildbots, see
459   // https://crbug.com/796318. Try calling fsync() more than once to workaround
460   // this issue.
461   //
462   // TODO(sergeyu): Figure out a better solution.
463   int flush_attempts_left = 4;
464   while (flush_attempts_left-- > 0) {
465     if (output.Flush())
466       return true;
467     LOG(ERROR) << "fsync() failed when saving test output summary. "
468                << ((flush_attempts_left > 0) ? "Retrying." : " Giving up.");
469   }
470 
471   return false;
472 }
473 
474 TestResultsTracker::TestStatusMap
GetTestStatusMapForCurrentIteration() const475     TestResultsTracker::GetTestStatusMapForCurrentIteration() const {
476   TestStatusMap tests_by_status;
477   GetTestStatusForIteration(iteration_, &tests_by_status);
478   return tests_by_status;
479 }
480 
481 TestResultsTracker::TestStatusMap
GetTestStatusMapForAllIterations() const482     TestResultsTracker::GetTestStatusMapForAllIterations() const {
483   TestStatusMap tests_by_status;
484   for (int i = 0; i <= iteration_; i++)
485     GetTestStatusForIteration(i, &tests_by_status);
486   return tests_by_status;
487 }
488 
GetTestStatusForIteration(int iteration,TestStatusMap * map) const489 void TestResultsTracker::GetTestStatusForIteration(
490     int iteration, TestStatusMap* map) const {
491   for (PerIterationData::ResultsMap::const_iterator j =
492            per_iteration_data_[iteration].results.begin();
493        j != per_iteration_data_[iteration].results.end();
494        ++j) {
495     // Use the last test result as the final one.
496     const TestResult& result = j->second.test_results.back();
497     (*map)[result.status].insert(result.full_name);
498   }
499 }
500 
501 // Utility function to print a list of test names. Uses iterator to be
502 // compatible with different containers, like vector and set.
503 template<typename InputIterator>
PrintTests(InputIterator first,InputIterator last,const std::string & description) const504 void TestResultsTracker::PrintTests(InputIterator first,
505                                     InputIterator last,
506                                     const std::string& description) const {
507   size_t count = std::distance(first, last);
508   if (count == 0)
509     return;
510 
511   fprintf(stdout,
512           "%" PRIuS " test%s %s:\n",
513           count,
514           count != 1 ? "s" : "",
515           description.c_str());
516   for (InputIterator it = first; it != last; ++it) {
517     const std::string& test_name = *it;
518     const auto location_it = test_locations_.find(test_name);
519     DCHECK(location_it != test_locations_.end()) << test_name;
520     const CodeLocation& location = location_it->second;
521     fprintf(stdout, "    %s (%s:%d)\n", test_name.c_str(),
522             location.file.c_str(), location.line);
523   }
524   fflush(stdout);
525 }
526 
527 TestResultsTracker::AggregateTestResult::AggregateTestResult() = default;
528 
529 TestResultsTracker::AggregateTestResult::AggregateTestResult(
530     const AggregateTestResult& other) = default;
531 
532 TestResultsTracker::AggregateTestResult::~AggregateTestResult() = default;
533 
534 TestResultsTracker::PerIterationData::PerIterationData() = default;
535 
536 TestResultsTracker::PerIterationData::PerIterationData(
537     const PerIterationData& other) = default;
538 
539 TestResultsTracker::PerIterationData::~PerIterationData() = default;
540 
541 }  // namespace base
542