file_test_base_test.cpp 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "testing/file_test/file_test_base.h"
  5. #include <gmock/gmock.h>
  6. #include <gtest/gtest.h>
  7. #include "common/ostream.h"
  8. #include "llvm/ADT/StringExtras.h"
  9. #include "llvm/Support/FormatVariadic.h"
  10. namespace Carbon::Testing {
  11. namespace {
  12. class FileTestBaseTest : public FileTestBase {
  13. public:
  14. using FileTestBase::FileTestBase;
  15. auto Run(const llvm::SmallVector<llvm::StringRef>& test_args,
  16. llvm::vfs::InMemoryFileSystem& fs, llvm::raw_pwrite_stream& stdout,
  17. llvm::raw_pwrite_stream& stderr) -> ErrorOr<RunResult> override {
  18. llvm::ArrayRef<llvm::StringRef> args = test_args;
  19. llvm::ListSeparator sep;
  20. stdout << args.size() << " args: ";
  21. for (auto arg : args) {
  22. stdout << sep << "`" << arg << "`";
  23. }
  24. stdout << "\n";
  25. auto filename = std::filesystem::path(test_name().str()).filename();
  26. if (filename == "args.carbon") {
  27. // 'args.carbon' has custom arguments, so don't do regular argument
  28. // validation for it.
  29. return {{.success = true}};
  30. }
  31. if (args.empty() || args.front() != "default_args") {
  32. return ErrorBuilder() << "missing `default_args` argument";
  33. }
  34. args = args.drop_front();
  35. for (auto arg : args) {
  36. if (!fs.exists(arg)) {
  37. return ErrorBuilder() << "Missing file: " << arg;
  38. }
  39. }
  40. if (filename == "example.carbon") {
  41. int delta_line = 10;
  42. stdout << "something\n"
  43. << "\n"
  44. << "example.carbon:" << delta_line + 1 << ": Line delta\n"
  45. << "example.carbon:" << delta_line << ": Negative line delta\n"
  46. << "+*[]{}\n"
  47. << "Foo baz\n";
  48. return {{.success = true}};
  49. } else if (filename == "fail_example.carbon") {
  50. stderr << "Oops\n";
  51. return {{.success = false}};
  52. } else if (filename == "two_files.carbon" ||
  53. filename == "not_split.carbon") {
  54. for (auto arg : args) {
  55. // Describe file contents to stdout to validate splitting.
  56. auto file = fs.getBufferForFile(arg, /*FileSize=*/-1,
  57. /*RequiresNullTerminator=*/false);
  58. if (file.getError()) {
  59. return Error(file.getError().message());
  60. }
  61. llvm::StringRef content = file.get()->getBuffer();
  62. stdout << arg << ":1: starts with \"";
  63. stdout.write_escaped(content.take_front(40));
  64. stdout << "\", length " << content.count('\n') << " lines\n";
  65. }
  66. return {{.success = true}};
  67. } else if (filename == "alternating_files.carbon") {
  68. stdout << "unattached message 1\n"
  69. << "a.carbon:2: message 2\n"
  70. << "b.carbon:5: message 3\n"
  71. << "a.carbon:2: message 4\n"
  72. << "b.carbon:5: message 5\n"
  73. << "unattached message 6\n";
  74. stderr << "unattached message 1\n"
  75. << "a.carbon:2: message 2\n"
  76. << "b.carbon:5: message 3\n"
  77. << "a.carbon:2: message 4\n"
  78. << "b.carbon:5: message 5\n"
  79. << "unattached message 6\n";
  80. return {{.success = true}};
  81. } else if (filename == "unattached_multi_file.carbon") {
  82. stdout << "unattached message 1\n"
  83. << "unattached message 2\n";
  84. stderr << "unattached message 3\n"
  85. << "unattached message 4\n";
  86. return {{.success = true}};
  87. } else if (filename == "file_only_re_one_file.carbon") {
  88. stdout << "unattached message 1\n"
  89. << "file: file_only_re_one_file.carbon\n"
  90. << "line: 1\n"
  91. << "unattached message 2\n";
  92. return {{.success = true}};
  93. } else if (filename == "file_only_re_multi_file.carbon") {
  94. int msg_count = 0;
  95. stdout << "unattached message " << ++msg_count << "\n"
  96. << "file: a.carbon\n"
  97. << "unattached message " << ++msg_count << "\n"
  98. << "line: 3: attached message " << ++msg_count << "\n"
  99. << "unattached message " << ++msg_count << "\n"
  100. << "line: 8: late message " << ++msg_count << "\n"
  101. << "unattached message " << ++msg_count << "\n"
  102. << "file: b.carbon\n"
  103. << "line: 2: attached message " << ++msg_count << "\n"
  104. << "unattached message " << ++msg_count << "\n"
  105. << "line: 7: late message " << ++msg_count << "\n"
  106. << "unattached message " << ++msg_count << "\n";
  107. return {{.success = true}};
  108. } else if (filename == "fail_multi_success_overall_fail.carbon") {
  109. RunResult result = {.success = false};
  110. result.per_file_success.push_back({"a.carbon", true});
  111. result.per_file_success.push_back({"b.carbon", true});
  112. return result;
  113. } else if (filename == "multi_success.carbon") {
  114. RunResult result = {.success = true};
  115. result.per_file_success.push_back({"a.carbon", true});
  116. result.per_file_success.push_back({"b.carbon", true});
  117. return result;
  118. } else if (filename == "multi_success_and_fail.carbon") {
  119. RunResult result = {.success = false};
  120. result.per_file_success.push_back({"a.carbon", true});
  121. result.per_file_success.push_back({"fail_b.carbon", false});
  122. return result;
  123. } else {
  124. return ErrorBuilder() << "Unexpected file: " << filename;
  125. }
  126. }
  127. auto GetDefaultArgs() -> llvm::SmallVector<std::string> override {
  128. return {"default_args", "%s"};
  129. }
  130. auto GetDefaultFileRE(llvm::ArrayRef<llvm::StringRef> filenames)
  131. -> std::optional<RE2> override {
  132. return std::make_optional<RE2>(
  133. llvm::formatv(R"(file: ({0}))", llvm::join(filenames, "|")));
  134. }
  135. auto GetLineNumberReplacements(llvm::ArrayRef<llvm::StringRef> filenames)
  136. -> llvm::SmallVector<LineNumberReplacement> override {
  137. auto replacements = FileTestBase::GetLineNumberReplacements(filenames);
  138. auto filename = std::filesystem::path(test_name().str()).filename();
  139. if (llvm::StringRef(filename).starts_with("file_only_re_")) {
  140. replacements.push_back({.has_file = false,
  141. .re = std::make_shared<RE2>(R"(line: (\d+))"),
  142. .line_formatv = "{0}"});
  143. }
  144. return replacements;
  145. }
  146. };
  147. } // namespace
  148. CARBON_FILE_TEST_FACTORY(FileTestBaseTest);
  149. } // namespace Carbon::Testing