| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #include "toolchain/diagnostics/emitter.h"
- #include <gmock/gmock.h>
- #include <gtest/gtest.h>
- #include "llvm/ADT/StringRef.h"
- #include "toolchain/diagnostics/mocks.h"
- namespace Carbon::Testing {
- namespace {
- using testing::ElementsAre;
- class FakeEmitter : public Diagnostics::Emitter<int> {
- public:
- using Emitter::Emitter;
- protected:
- auto ConvertLoc(int n, ContextFnT /*context_fn*/) const
- -> Diagnostics::ConvertedLoc override {
- return {.loc = {.line_number = 1, .column_number = n},
- .last_byte_offset = -1};
- }
- };
- class EmitterTest : public ::testing::Test {
- public:
- EmitterTest() : emitter_(&consumer_) {}
- Testing::MockDiagnosticConsumer consumer_;
- FakeEmitter emitter_;
- };
- TEST_F(EmitterTest, EmitSimpleError) {
- CARBON_DIAGNOSTIC(TestDiagnostic, Error, "simple error");
- EXPECT_CALL(consumer_, HandleDiagnostic(IsSingleDiagnostic(
- Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Error, 1, 1, "simple error")));
- EXPECT_CALL(consumer_, HandleDiagnostic(IsSingleDiagnostic(
- Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Error, 1, 2, "simple error")));
- emitter_.Emit(1, TestDiagnostic);
- emitter_.Emit(2, TestDiagnostic);
- }
- TEST_F(EmitterTest, EmitSimpleWarning) {
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- EXPECT_CALL(consumer_,
- HandleDiagnostic(IsSingleDiagnostic(
- Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1, "simple warning")));
- emitter_.Emit(1, TestDiagnostic);
- }
- TEST_F(EmitterTest, EmitOneArgDiagnostic) {
- CARBON_DIAGNOSTIC(TestDiagnostic, Error, "arg: `{0}`", std::string);
- EXPECT_CALL(consumer_, HandleDiagnostic(IsSingleDiagnostic(
- Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Error, 1, 1, "arg: `str`")));
- emitter_.Emit(1, TestDiagnostic, "str");
- }
- TEST_F(EmitterTest, EmitNote) {
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- CARBON_DIAGNOSTIC(TestDiagnosticNote, Note, "note");
- EXPECT_CALL(
- consumer_,
- HandleDiagnostic(IsDiagnostic(
- Diagnostics::Level::Warning,
- ElementsAre(
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1,
- "simple warning"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticNote,
- Diagnostics::Level::Note, 1, 2, "note")))));
- emitter_.Build(1, TestDiagnostic).Note(2, TestDiagnosticNote).Emit();
- }
- TEST_F(EmitterTest, EmitContext) {
- CARBON_DIAGNOSTIC(TestDiagnosticContext, Context, "context");
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- EXPECT_CALL(
- consumer_,
- HandleDiagnostic(IsDiagnostic(
- Diagnostics::Level::Warning,
- ElementsAre(
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticContext,
- Diagnostics::Level::Context, 1, 2, "context"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1,
- "simple warning")))));
- Diagnostics::ContextScope scope(&emitter_, [&](auto& builder) {
- builder.Context(2, TestDiagnosticContext);
- });
- emitter_.Emit(1, TestDiagnostic);
- }
- TEST_F(EmitterTest, EmitSoftContext) {
- CARBON_DIAGNOSTIC(TestDiagnosticSoftContext, SoftContext, "soft context");
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- EXPECT_CALL(
- consumer_,
- HandleDiagnostic(IsDiagnostic(
- Diagnostics::Level::Warning,
- ElementsAre(
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticSoftContext,
- Diagnostics::Level::SoftContext, 1, 2,
- "soft context"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1,
- "simple warning")))));
- Diagnostics::ContextScope soft_scope(&emitter_, [&](auto& builder) {
- builder.Context(2, TestDiagnosticSoftContext);
- });
- emitter_.Emit(1, TestDiagnostic);
- }
- TEST_F(EmitterTest, EmitSoftContextAndContext) {
- CARBON_DIAGNOSTIC(TestDiagnosticSoftContext, SoftContext, "soft context");
- CARBON_DIAGNOSTIC(TestDiagnosticContext, Context, "context");
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- EXPECT_CALL(
- consumer_,
- HandleDiagnostic(IsDiagnostic(
- Diagnostics::Level::Warning,
- ElementsAre(
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticSoftContext,
- Diagnostics::Level::SoftContext, 1, 3,
- "soft context"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticContext,
- Diagnostics::Level::Context, 1, 2, "context"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1,
- "simple warning")))));
- Diagnostics::ContextScope soft_scope(&emitter_, [&](auto& builder) {
- builder.Context(3, TestDiagnosticSoftContext);
- });
- Diagnostics::ContextScope scope(&emitter_, [&](auto& builder) {
- builder.Context(2, TestDiagnosticContext);
- });
- emitter_.Emit(1, TestDiagnostic);
- }
- TEST_F(EmitterTest, EmitContextAndSoftContext) {
- CARBON_DIAGNOSTIC(TestDiagnosticContext, Context, "context");
- CARBON_DIAGNOSTIC(TestDiagnosticSoftContext, SoftContext, "soft context");
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- EXPECT_CALL(
- consumer_,
- HandleDiagnostic(IsDiagnostic(
- Diagnostics::Level::Warning,
- ElementsAre(
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticContext,
- Diagnostics::Level::Context, 1, 3, "context"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1,
- "simple warning")))));
- Diagnostics::ContextScope scope(&emitter_, [&](auto& builder) {
- builder.Context(3, TestDiagnosticContext);
- });
- // This SoftContext does not produce a message, since the earlire Context
- // supersedes it.
- Diagnostics::ContextScope soft_scope(&emitter_, [&](auto& builder) {
- builder.Context(2, TestDiagnosticSoftContext);
- });
- emitter_.Emit(1, TestDiagnostic);
- }
- TEST_F(EmitterTest, EmitTwoContext) {
- CARBON_DIAGNOSTIC(TestDiagnosticContext, Context, "context");
- CARBON_DIAGNOSTIC(TestDiagnosticContext2, Context, "context 2");
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- EXPECT_CALL(
- consumer_,
- HandleDiagnostic(IsDiagnostic(
- Diagnostics::Level::Warning,
- ElementsAre(
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticContext,
- Diagnostics::Level::Context, 1, 3, "context"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticContext2,
- Diagnostics::Level::Context, 1, 2,
- "context 2"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1,
- "simple warning")))));
- Diagnostics::ContextScope scope(&emitter_, [&](auto& builder) {
- builder.Context(3, TestDiagnosticContext);
- });
- Diagnostics::ContextScope scope2(&emitter_, [&](auto& builder) {
- builder.Context(2, TestDiagnosticContext2);
- });
- emitter_.Emit(1, TestDiagnostic);
- }
- TEST_F(EmitterTest, EmitTwoSoftContext) {
- CARBON_DIAGNOSTIC(TestDiagnosticSoftContext, SoftContext, "soft context");
- CARBON_DIAGNOSTIC(TestDiagnosticSoftContext2, SoftContext, "soft context 2");
- CARBON_DIAGNOSTIC(TestDiagnostic, Warning, "simple warning");
- EXPECT_CALL(
- consumer_,
- HandleDiagnostic(IsDiagnostic(
- Diagnostics::Level::Warning,
- ElementsAre(
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnosticSoftContext,
- Diagnostics::Level::SoftContext, 1, 3,
- "soft context"),
- IsDiagnosticMessage(Diagnostics::Kind::TestDiagnostic,
- Diagnostics::Level::Warning, 1, 1,
- "simple warning")))));
- Diagnostics::ContextScope scope(&emitter_, [&](auto& builder) {
- builder.Context(3, TestDiagnosticSoftContext);
- });
- // This SoftContext does not produce a message, since the earlire Context
- // supersedes it.
- Diagnostics::ContextScope soft_scope(&emitter_, [&](auto& builder) {
- builder.Context(2, TestDiagnosticSoftContext2);
- });
- emitter_.Emit(1, TestDiagnostic);
- }
- TEST_F(EmitterTest, Flush) {
- bool flushed = false;
- auto flush_fn = [&]() { flushed = true; };
- {
- FakeEmitter emitter(&consumer_);
- emitter.AddFlushFn(flush_fn);
- // Registering the function does not flush.
- EXPECT_FALSE(flushed);
- // Explicit calls to `Flush` should flush.
- emitter.Flush();
- EXPECT_TRUE(flushed);
- flushed = false;
- {
- Diagnostics::AnnotationScope annot(&emitter, [](auto&) {});
- // Registering an annotation scope should flush.
- EXPECT_TRUE(flushed);
- flushed = false;
- }
- // Unregistering an annotation scope should flush.
- EXPECT_TRUE(flushed);
- flushed = false;
- }
- // Destroying the emitter should not flush, as that could call back into the
- // base class emitter after the derived-class emitter has been destroyed.
- EXPECT_FALSE(flushed);
- }
- } // namespace
- } // namespace Carbon::Testing
|