1 //===-- ConfigTesting.h - Helpers for configuration tests -------*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #ifndef LLVM_CLANG_TOOLS_EXTRA_CLANGD_UNITTESTS_CONFIGTESTING_H
10 #define LLVM_CLANG_TOOLS_EXTRA_CLANGD_UNITTESTS_CONFIGTESTING_H
11 
12 #include "Protocol.h"
13 #include "llvm/Support/ScopedPrinter.h"
14 #include "llvm/Support/SourceMgr.h"
15 #include "gmock/gmock.h"
16 #include <functional>
17 
18 namespace clang {
19 namespace clangd {
20 namespace config {
21 
22 // Provides a DiagnosticsCallback that records diganostics.
23 // Unlike just pushing them into a vector, underlying storage need not survive.
24 struct CapturedDiags {
callbackCapturedDiags25   std::function<void(const llvm::SMDiagnostic &)> callback() {
26     return [this](const llvm::SMDiagnostic &D) {
27       if (Files.empty() || Files.back() != D.getFilename())
28         Files.push_back(D.getFilename().str());
29 
30       if (D.getKind() > llvm::SourceMgr::DK_Warning)
31         return;
32 
33       Diagnostics.emplace_back();
34       Diag &Out = Diagnostics.back();
35       Out.Message = D.getMessage().str();
36       Out.Kind = D.getKind();
37       Out.Pos.line = D.getLineNo() - 1;
38       Out.Pos.character = D.getColumnNo(); // Zero-based - bug in SourceMgr?
39       if (!D.getRanges().empty()) {
40         const auto &R = D.getRanges().front();
41         Out.Rng.emplace();
42         Out.Rng->start.line = Out.Rng->end.line = Out.Pos.line;
43         Out.Rng->start.character = R.first;
44         Out.Rng->end.character = R.second;
45       }
46     };
47   }
48   struct Diag {
49     std::string Message;
50     llvm::SourceMgr::DiagKind Kind;
51     Position Pos;
52     llvm::Optional<Range> Rng;
53 
PrintToCapturedDiags::Diag54     friend void PrintTo(const Diag &D, std::ostream *OS) {
55       *OS << (D.Kind == llvm::SourceMgr::DK_Error ? "error: " : "warning: ")
56           << D.Message << "@" << llvm::to_string(D.Pos);
57     }
58   };
59   std::vector<Diag> Diagnostics;  // Warning or higher.
60   std::vector<std::string> Files; // Filename from diagnostics including notes.
61 
clearCapturedDiags62   void clear() {
63     Diagnostics.clear();
64     Files.clear();
65   }
66 };
67 
68 MATCHER_P(DiagMessage, M, "") { return arg.Message == M; }
69 MATCHER_P(DiagKind, K, "") { return arg.Kind == K; }
70 MATCHER_P(DiagPos, P, "") { return arg.Pos == P; }
71 MATCHER_P(DiagRange, R, "") { return arg.Rng == R; }
72 
toPosition(llvm::SMLoc L,const llvm::SourceMgr & SM)73 inline Position toPosition(llvm::SMLoc L, const llvm::SourceMgr &SM) {
74   auto LineCol = SM.getLineAndColumn(L);
75   Position P;
76   P.line = LineCol.first - 1;
77   P.character = LineCol.second - 1;
78   return P;
79 }
80 
toRange(llvm::SMRange R,const llvm::SourceMgr & SM)81 inline Range toRange(llvm::SMRange R, const llvm::SourceMgr &SM) {
82   return Range{toPosition(R.Start, SM), toPosition(R.End, SM)};
83 }
84 
85 } // namespace config
86 } // namespace clangd
87 } // namespace clang
88 
89 #endif
90