diagnostic_emitter_test.cpp 3.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  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 "toolchain/diagnostics/diagnostic_emitter.h"
  5. #include <gmock/gmock.h>
  6. #include <gtest/gtest.h>
  7. #include "llvm/ADT/StringRef.h"
  8. #include "llvm/Support/FormatVariadic.h"
  9. #include "toolchain/diagnostics/mocks.h"
  10. namespace Carbon {
  11. namespace {
  12. using Testing::DiagnosticAt;
  13. using Testing::DiagnosticLevel;
  14. using Testing::DiagnosticMessage;
  15. using Testing::DiagnosticShortName;
  16. struct FakeDiagnostic {
  17. static constexpr llvm::StringLiteral ShortName = "fake-diagnostic";
  18. // TODO: consider ways to put the Message into `format` to allow dynamic
  19. // selection of the message.
  20. static constexpr llvm::StringLiteral Message = "{0}";
  21. std::string message;
  22. auto Format() -> std::string {
  23. // Work around a bug in Clang's unused const variable warning by marking it
  24. // used here with a no-op.
  25. static_cast<void>(ShortName);
  26. return llvm::formatv(Message.data(), message).str();
  27. }
  28. };
  29. struct FakeDiagnosticLocationTranslator : DiagnosticLocationTranslator<int> {
  30. auto GetLocation(int n) -> Diagnostic::Location override {
  31. return {.file_name = "test", .line_number = 1, .column_number = n};
  32. }
  33. };
  34. TEST(DiagTest, EmitErrors) {
  35. FakeDiagnosticLocationTranslator translator;
  36. Testing::MockDiagnosticConsumer consumer;
  37. DiagnosticEmitter<int> emitter(translator, consumer);
  38. EXPECT_CALL(consumer, HandleDiagnostic(
  39. AllOf(DiagnosticLevel(Diagnostic::Error),
  40. DiagnosticAt(1, 1), DiagnosticMessage("M1"),
  41. DiagnosticShortName("fake-diagnostic"))));
  42. EXPECT_CALL(consumer, HandleDiagnostic(
  43. AllOf(DiagnosticLevel(Diagnostic::Error),
  44. DiagnosticAt(1, 2), DiagnosticMessage("M2"),
  45. DiagnosticShortName("fake-diagnostic"))));
  46. emitter.EmitError<FakeDiagnostic>(1, {.message = "M1"});
  47. emitter.EmitError<FakeDiagnostic>(2, {.message = "M2"});
  48. }
  49. TEST(DiagTest, EmitWarnings) {
  50. std::vector<std::string> reported;
  51. FakeDiagnosticLocationTranslator translator;
  52. Testing::MockDiagnosticConsumer consumer;
  53. DiagnosticEmitter<int> emitter(translator, consumer);
  54. EXPECT_CALL(consumer, HandleDiagnostic(
  55. AllOf(DiagnosticLevel(Diagnostic::Warning),
  56. DiagnosticAt(1, 3), DiagnosticMessage("M1"),
  57. DiagnosticShortName("fake-diagnostic"))));
  58. EXPECT_CALL(consumer, HandleDiagnostic(
  59. AllOf(DiagnosticLevel(Diagnostic::Warning),
  60. DiagnosticAt(1, 5), DiagnosticMessage("M3"),
  61. DiagnosticShortName("fake-diagnostic"))));
  62. emitter.EmitWarningIf<FakeDiagnostic>(3, [](FakeDiagnostic& diagnostic) {
  63. diagnostic.message = "M1";
  64. return true;
  65. });
  66. emitter.EmitWarningIf<FakeDiagnostic>(4, [](FakeDiagnostic& diagnostic) {
  67. diagnostic.message = "M2";
  68. return false;
  69. });
  70. emitter.EmitWarningIf<FakeDiagnostic>(5, [](FakeDiagnostic& diagnostic) {
  71. diagnostic.message = "M3";
  72. return true;
  73. });
  74. }
  75. } // namespace
  76. } // namespace Carbon