file_test_base_test.cpp 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  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<bool> 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 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 true;
  49. } else if (filename == "fail_example.carbon") {
  50. stderr << "Oops\n";
  51. return 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 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 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 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 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: 1: 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: 1: 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 true;
  108. } else {
  109. return ErrorBuilder() << "Unexpected file: " << filename;
  110. }
  111. }
  112. auto GetDefaultArgs() -> llvm::SmallVector<std::string> override {
  113. return {"default_args", "%s"};
  114. }
  115. auto GetDefaultFileRE(llvm::ArrayRef<llvm::StringRef> filenames)
  116. -> std::optional<RE2> override {
  117. return std::make_optional<RE2>(
  118. llvm::formatv(R"(file: ({0}))", llvm::join(filenames, "|")));
  119. }
  120. auto GetLineNumberReplacements(llvm::ArrayRef<llvm::StringRef> filenames)
  121. -> llvm::SmallVector<LineNumberReplacement> override {
  122. auto replacements = FileTestBase::GetLineNumberReplacements(filenames);
  123. auto filename = std::filesystem::path(test_name().str()).filename();
  124. if (llvm::StringRef(filename).startswith("file_only_re_")) {
  125. replacements.push_back({.has_file = false,
  126. .re = std::make_shared<RE2>(R"(line: (\d+))"),
  127. .line_formatv = "{0}"});
  128. }
  129. return replacements;
  130. }
  131. };
  132. } // namespace
  133. CARBON_FILE_TEST_FACTORY(FileTestBaseTest);
  134. } // namespace Carbon::Testing