132 lines
4.2 KiB
C++
132 lines
4.2 KiB
C++
|
|
// 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
|