| // Copyright 2013 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "testing/data_driven_testing/data_driven_test.h" |
| |
| #include "base/files/file_enumerator.h" |
| #include "base/files/file_util.h" |
| #include "base/strings/string_util.h" |
| #include "base/threading/thread_restrictions.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "third_party/re2/src/re2/re2.h" |
| |
| namespace testing { |
| namespace { |
| |
| // Reads |file| into |content|, and converts Windows line-endings to Unix ones. |
| // Returns true on success. |
| bool ReadFile(const base::FilePath& file, std::string* content) { |
| if (!base::ReadFileToString(file, content)) |
| return false; |
| |
| base::ReplaceSubstringsAfterOffset(content, 0, "\r\n", "\n"); |
| return true; |
| } |
| |
| // Write |content| to |file|. Returns true on success. |
| bool WriteFile(const base::FilePath& file, const std::string& content) { |
| return base::WriteFile(file, content); |
| } |
| |
| // Removes lines starting with (optional) whitespace and a #. |
| void StripComments(std::string* content) { |
| RE2::GlobalReplace( |
| content, |
| // Enable multi-line mode, ^ and $ match begin/end line in addition to |
| // begin/end text. |
| "(?m)" |
| // Search for start of lines (^), ignore spaces (\\s*), and then look for |
| // '#'. |
| "^\\s*#" |
| // Consume all characters (.*) until end of line ($). |
| ".*$" |
| // Consume the line wrapping so that the entire line is gone. |
| "[\\r\\n]*", |
| // Replace entire line with empty string. |
| ""); |
| } |
| |
| } // namespace |
| |
| void DataDrivenTest::RunDataDrivenTest( |
| const base::FilePath& input_directory, |
| const base::FilePath& output_directory, |
| const base::FilePath::StringType& file_name_pattern) { |
| base::ScopedAllowBlockingForTesting allow_blocking; |
| ASSERT_TRUE(base::DirectoryExists(input_directory)); |
| ASSERT_TRUE(base::DirectoryExists(output_directory)); |
| base::FileEnumerator input_files( |
| input_directory, false, base::FileEnumerator::FILES, file_name_pattern); |
| const bool kIsExpectedToPass = true; |
| for (base::FilePath input_file = input_files.Next(); !input_file.empty(); |
| input_file = input_files.Next()) { |
| RunOneDataDrivenTest(input_file, output_directory, kIsExpectedToPass); |
| } |
| } |
| |
| void DataDrivenTest::RunOneDataDrivenTest( |
| const base::FilePath& test_file_name, |
| const base::FilePath& output_directory, |
| bool is_expected_to_pass) { |
| base::ScopedAllowBlockingForTesting allow_blocking; |
| // iOS doesn't get rid of removed test files. TODO(estade): remove this after |
| // all iOS bots are clobbered. |
| if (test_file_name.BaseName().value() == FILE_PATH_LITERAL("multimerge.in")) |
| return; |
| |
| ASSERT_TRUE(base::DirectoryExists(output_directory)); |
| SCOPED_TRACE(test_file_name.BaseName().value()); |
| |
| std::string input; |
| ReadFile(test_file_name, &input); |
| |
| std::string output; |
| { |
| base::ScopedDisallowBlocking disallow_blocking; |
| GenerateResults(input, &output); |
| } |
| |
| base::FilePath output_file = output_directory.Append( |
| test_file_name.BaseName().StripTrailingSeparators().ReplaceExtension( |
| FILE_PATH_LITERAL(".out"))); |
| |
| std::string output_file_contents; |
| if (!ReadFile(output_file, &output_file_contents)) { |
| ASSERT_TRUE(WriteFile(output_file, output)); |
| return; |
| } |
| // Remove comment lines (lead by '#' character). |
| StripComments(&output_file_contents); |
| |
| if (is_expected_to_pass) { |
| EXPECT_EQ(output_file_contents, output); |
| } else { |
| EXPECT_NE(output_file_contents, output); |
| } |
| } |
| |
| base::FilePath DataDrivenTest::GetInputDirectory() { |
| return test_data_directory_.Append(feature_directory_) |
| .Append(test_name_) |
| .AppendASCII("input"); |
| } |
| |
| base::FilePath DataDrivenTest::GetOutputDirectory() { |
| return test_data_directory_.Append(feature_directory_) |
| .Append(test_name_) |
| .AppendASCII("output"); |
| } |
| |
| DataDrivenTest::DataDrivenTest( |
| const base::FilePath& test_data_directory, |
| const base::FilePath::StringType& feature_directory, |
| const base::FilePath::StringType& test_name) |
| : test_data_directory_(test_data_directory), |
| feature_directory_(feature_directory), |
| test_name_(test_name) {} |
| |
| DataDrivenTest::~DataDrivenTest() {} |
| |
| } // namespace testing |