9#include "llvm/ADT/SmallSet.h"
10#include "llvm/ADT/SmallString.h"
11#include "llvm/ADT/StringRef.h"
12#include "llvm/Support/MemoryBuffer.h"
13#include "llvm/Support/SourceMgr.h"
14#include "llvm/Support/YAMLParser.h"
22using llvm::yaml::BlockScalarNode;
23using llvm::yaml::MappingNode;
24using llvm::yaml::Node;
25using llvm::yaml::ScalarNode;
26using llvm::yaml::SequenceNode;
28std::optional<llvm::StringRef>
29bestGuess(llvm::StringRef Search,
30 llvm::ArrayRef<llvm::StringRef> AllowedValues) {
31 unsigned MaxEdit = (Search.size() + 1) / 3;
34 std::optional<llvm::StringRef> Result;
35 for (
const auto &AllowedValue : AllowedValues) {
36 unsigned EditDistance = Search.edit_distance(AllowedValue,
true, MaxEdit);
39 if (EditDistance == 1U)
41 if (EditDistance == MaxEdit && !Result) {
42 Result = AllowedValue;
43 }
else if (EditDistance < MaxEdit) {
44 Result = AllowedValue;
45 MaxEdit = EditDistance;
53 bool HadError =
false;
56 Parser(llvm::SourceMgr &SM) : SM(SM) {}
60 bool parse(Fragment &F, Node &N) {
61 DictParser Dict(
"Config",
this);
62 Dict.handle(
"If", [&](Node &N) { parse(F.If, N); });
63 Dict.handle(
"CompileFlags", [&](Node &N) { parse(F.CompileFlags, N); });
64 Dict.handle(
"Index", [&](Node &N) { parse(F.Index, N); });
65 Dict.handle(
"Style", [&](Node &N) { parse(F.Style, N); });
66 Dict.handle(
"Diagnostics", [&](Node &N) { parse(F.Diagnostics, N); });
67 Dict.handle(
"Completion", [&](Node &N) { parse(F.Completion, N); });
68 Dict.handle(
"Hover", [&](Node &N) { parse(F.Hover, N); });
69 Dict.handle(
"InlayHints", [&](Node &N) { parse(F.InlayHints, N); });
70 Dict.handle(
"SemanticTokens", [&](Node &N) { parse(F.SemanticTokens, N); });
72 return !(N.failed() || HadError);
76 void parse(Fragment::IfBlock &F, Node &N) {
77 DictParser Dict(
"If",
this);
78 Dict.unrecognized([&](Located<std::string>, Node &) {
79 F.HasUnrecognizedCondition =
true;
82 Dict.handle(
"PathMatch", [&](Node &N) {
83 if (
auto Values = scalarValues(N))
84 F.PathMatch = std::move(*Values);
86 Dict.handle(
"PathExclude", [&](Node &N) {
87 if (
auto Values = scalarValues(N))
88 F.PathExclude = std::move(*Values);
93 void parse(Fragment::CompileFlagsBlock &F, Node &N) {
94 DictParser Dict(
"CompileFlags",
this);
95 Dict.handle(
"Compiler", [&](Node &N) {
96 if (
auto Value = scalarValue(N,
"Compiler"))
97 F.Compiler = std::move(*
Value);
99 Dict.handle(
"Add", [&](Node &N) {
100 if (
auto Values = scalarValues(N))
101 F.Add = std::move(*Values);
103 Dict.handle(
"Remove", [&](Node &N) {
104 if (
auto Values = scalarValues(N))
105 F.Remove = std::move(*Values);
107 Dict.handle(
"CompilationDatabase", [&](Node &N) {
108 F.CompilationDatabase = scalarValue(N,
"CompilationDatabase");
113 void parse(Fragment::StyleBlock &F, Node &N) {
114 DictParser Dict(
"Style",
this);
115 Dict.handle(
"FullyQualifiedNamespaces", [&](Node &N) {
116 if (
auto Values = scalarValues(N))
117 F.FullyQualifiedNamespaces = std::move(*Values);
122 void parse(Fragment::DiagnosticsBlock &F, Node &N) {
123 DictParser Dict(
"Diagnostics",
this);
124 Dict.handle(
"Suppress", [&](Node &N) {
125 if (
auto Values = scalarValues(N))
126 F.Suppress = std::move(*Values);
128 Dict.handle(
"UnusedIncludes", [&](Node &N) {
129 F.UnusedIncludes = scalarValue(N,
"UnusedIncludes");
131 Dict.handle(
"MissingIncludes", [&](Node &N) {
132 F.MissingIncludes = scalarValue(N,
"MissingIncludes");
134 Dict.handle(
"Includes", [&](Node &N) { parse(F.Includes, N); });
135 Dict.handle(
"ClangTidy", [&](Node &N) { parse(F.ClangTidy, N); });
139 void parse(Fragment::DiagnosticsBlock::ClangTidyBlock &F, Node &N) {
140 DictParser Dict(
"ClangTidy",
this);
141 Dict.handle(
"Add", [&](Node &N) {
142 if (
auto Values = scalarValues(N))
143 F.Add = std::move(*Values);
145 Dict.handle(
"Remove", [&](Node &N) {
146 if (
auto Values = scalarValues(N))
147 F.Remove = std::move(*Values);
149 Dict.handle(
"CheckOptions", [&](Node &N) {
150 DictParser CheckOptDict(
"CheckOptions",
this);
151 CheckOptDict.unrecognized([&](Located<std::string> &&Key, Node &Val) {
152 if (
auto Value = scalarValue(Val, *Key))
153 F.CheckOptions.emplace_back(std::move(Key), std::move(*
Value));
156 CheckOptDict.parse(N);
158 Dict.handle(
"FastCheckFilter", [&](Node &N) {
159 if (
auto FastCheckFilter = scalarValue(N,
"FastCheckFilter"))
160 F.FastCheckFilter = *FastCheckFilter;
165 void parse(Fragment::DiagnosticsBlock::IncludesBlock &F, Node &N) {
166 DictParser Dict(
"Includes",
this);
167 Dict.handle(
"IgnoreHeader", [&](Node &N) {
168 if (
auto Values = scalarValues(N))
169 F.IgnoreHeader = std::move(*Values);
171 Dict.handle(
"AnalyzeAngledIncludes", [&](Node &N) {
172 if (
auto Value = boolValue(N,
"AnalyzeAngledIncludes"))
173 F.AnalyzeAngledIncludes = *
Value;
178 void parse(Fragment::IndexBlock &F, Node &N) {
179 DictParser Dict(
"Index",
this);
180 Dict.handle(
"Background",
181 [&](Node &N) { F.Background = scalarValue(N,
"Background"); });
182 Dict.handle(
"External", [&](Node &N) {
183 Fragment::IndexBlock::ExternalBlock External;
186 if (N.getType() == Node::NK_Mapping) {
188 }
else if (N.getType() == Node::NK_Scalar ||
189 N.getType() == Node::NK_BlockScalar) {
190 parse(External, *scalarValue(N,
"External"));
192 error(
"External must be either a scalar or a mapping.", N);
195 F.External.emplace(std::move(External));
196 F.External->Range = N.getSourceRange();
198 Dict.handle(
"StandardLibrary", [&](Node &N) {
199 if (
auto StandardLibrary = boolValue(N,
"StandardLibrary"))
200 F.StandardLibrary = *StandardLibrary;
205 void parse(Fragment::IndexBlock::ExternalBlock &F,
206 Located<std::string> ExternalVal) {
207 if (!llvm::StringRef(*ExternalVal).equals_insensitive(
"none")) {
208 error(
"Only scalar value supported for External is 'None'",
213 F.IsNone.Range = ExternalVal.Range;
216 void parse(Fragment::IndexBlock::ExternalBlock &F, Node &N) {
217 DictParser Dict(
"External",
this);
218 Dict.handle(
"File", [&](Node &N) { F.File = scalarValue(N,
"File"); });
219 Dict.handle(
"Server",
220 [&](Node &N) { F.Server = scalarValue(N,
"Server"); });
221 Dict.handle(
"MountPoint",
222 [&](Node &N) { F.MountPoint = scalarValue(N,
"MountPoint"); });
226 void parse(Fragment::CompletionBlock &F, Node &N) {
227 DictParser Dict(
"Completion",
this);
228 Dict.handle(
"AllScopes", [&](Node &N) {
229 if (
auto AllScopes = boolValue(N,
"AllScopes"))
230 F.AllScopes = *AllScopes;
232 Dict.handle(
"ArgumentLists", [&](Node &N) {
233 if (
auto ArgumentLists = scalarValue(N,
"ArgumentLists"))
234 F.ArgumentLists = *ArgumentLists;
239 void parse(Fragment::HoverBlock &F, Node &N) {
240 DictParser Dict(
"Hover",
this);
241 Dict.handle(
"ShowAKA", [&](Node &N) {
242 if (
auto ShowAKA = boolValue(N,
"ShowAKA"))
243 F.ShowAKA = *ShowAKA;
248 void parse(Fragment::InlayHintsBlock &F, Node &N) {
249 DictParser Dict(
"InlayHints",
this);
250 Dict.handle(
"Enabled", [&](Node &N) {
251 if (
auto Value = boolValue(N,
"Enabled"))
254 Dict.handle(
"ParameterNames", [&](Node &N) {
255 if (
auto Value = boolValue(N,
"ParameterNames"))
256 F.ParameterNames = *
Value;
258 Dict.handle(
"DeducedTypes", [&](Node &N) {
259 if (
auto Value = boolValue(N,
"DeducedTypes"))
260 F.DeducedTypes = *
Value;
262 Dict.handle(
"Designators", [&](Node &N) {
263 if (
auto Value = boolValue(N,
"Designators"))
264 F.Designators = *
Value;
266 Dict.handle(
"BlockEnd", [&](Node &N) {
267 if (
auto Value = boolValue(N,
"BlockEnd"))
270 Dict.handle(
"DefaultArguments", [&](Node &N) {
271 if (
auto Value = boolValue(N,
"DefaultArguments"))
272 F.DefaultArguments = *
Value;
274 Dict.handle(
"TypeNameLimit", [&](Node &N) {
275 if (
auto Value = uint32Value(N,
"TypeNameLimit"))
276 F.TypeNameLimit = *
Value;
281 void parse(Fragment::SemanticTokensBlock &F, Node &N) {
282 DictParser Dict(
"SemanticTokens",
this);
283 Dict.handle(
"DisabledKinds", [&](Node &N) {
284 if (
auto Values = scalarValues(N))
285 F.DisabledKinds = std::move(*Values);
287 Dict.handle(
"DisabledModifiers", [&](Node &N) {
288 if (
auto Values = scalarValues(N))
289 F.DisabledModifiers = std::move(*Values);
297 llvm::StringRef Description;
298 std::vector<std::pair<llvm::StringRef, std::function<void(Node &)>>> Keys;
299 std::function<bool(Located<std::string>, Node &)> UnknownHandler;
303 DictParser(llvm::StringRef Description, Parser *Outer)
304 : Description(Description), Outer(Outer) {}
309 void handle(llvm::StringLiteral Key, std::function<
void(Node &)> Parse) {
310 for (
const auto &
Entry : Keys) {
312 assert(
Entry.first != Key &&
"duplicate key handler");
314 Keys.emplace_back(Key, std::move(Parse));
321 unrecognized(std::function<
bool(Located<std::string>, Node &)> Handler) {
322 UnknownHandler = std::move(Handler);
326 void parse(Node &N)
const {
327 if (N.getType() != Node::NK_Mapping) {
328 Outer->error(Description +
" should be a dictionary", N);
331 llvm::SmallSet<std::string, 8> Seen;
332 llvm::SmallVector<Located<std::string>, 0> UnknownKeys;
334 for (
auto &KV : llvm::cast<MappingNode>(N)) {
335 auto *
K = KV.getKey();
338 auto Key = Outer->scalarValue(*
K,
"Dictionary key");
341 if (!Seen.insert(**Key).second) {
342 Outer->warning(
"Duplicate key " + **Key +
" is ignored", *
K);
343 if (
auto *
Value = KV.getValue())
347 auto *
Value = KV.getValue();
350 bool Matched =
false;
351 for (
const auto &Handler : Keys) {
352 if (Handler.first == **Key) {
354 Handler.second(*
Value);
359 bool Warn = !UnknownHandler;
361 Warn = UnknownHandler(
362 Located<std::string>(**Key,
K->getSourceRange()), *
Value);
364 UnknownKeys.push_back(std::move(*Key));
367 if (!UnknownKeys.empty())
368 warnUnknownKeys(UnknownKeys, Seen);
372 void warnUnknownKeys(llvm::ArrayRef<Located<std::string>> UnknownKeys,
373 const llvm::SmallSet<std::string, 8> &SeenKeys)
const {
374 llvm::SmallVector<llvm::StringRef> UnseenKeys;
375 for (
const auto &KeyAndHandler : Keys)
376 if (!SeenKeys.count(KeyAndHandler.first.str()))
377 UnseenKeys.push_back(KeyAndHandler.first);
379 for (
const Located<std::string> &UnknownKey : UnknownKeys)
380 if (
auto BestGuess = bestGuess(*UnknownKey, UnseenKeys))
381 Outer->warning(
"Unknown " + Description +
" key '" + *UnknownKey +
382 "'; did you mean '" + *BestGuess +
"'?",
385 Outer->warning(
"Unknown " + Description +
" key '" + *UnknownKey +
392 std::optional<Located<std::string>> scalarValue(Node &N,
393 llvm::StringRef Desc) {
394 llvm::SmallString<256> Buf;
395 if (
auto *S = llvm::dyn_cast<ScalarNode>(&N))
396 return Located<std::string>(S->getValue(Buf).str(), N.getSourceRange());
397 if (
auto *BS = llvm::dyn_cast<BlockScalarNode>(&N))
398 return Located<std::string>(BS->getValue().str(), N.getSourceRange());
399 warning(Desc +
" should be scalar", N);
403 std::optional<Located<bool>> boolValue(Node &N, llvm::StringRef Desc) {
404 if (
auto Scalar = scalarValue(N, Desc)) {
405 if (
auto Bool = llvm::yaml::parseBool(**Scalar))
406 return Located<bool>(*Bool, Scalar->Range);
407 warning(Desc +
" should be a boolean", N);
412 std::optional<Located<uint32_t>> uint32Value(Node &N, llvm::StringRef Desc) {
413 if (
auto Scalar = scalarValue(N, Desc)) {
414 unsigned long long Num;
415 if (!llvm::getAsUnsignedInteger(**Scalar, 0, Num)) {
416 return Located<uint32_t>(Num, Scalar->Range);
419 warning(Desc +
" invalid number", N);
424 std::optional<std::vector<Located<std::string>>> scalarValues(Node &N) {
425 std::vector<Located<std::string>> Result;
426 if (
auto *S = llvm::dyn_cast<ScalarNode>(&N)) {
427 llvm::SmallString<256> Buf;
428 Result.emplace_back(S->getValue(Buf).str(), N.getSourceRange());
429 }
else if (
auto *S = llvm::dyn_cast<BlockScalarNode>(&N)) {
430 Result.emplace_back(S->getValue().str(), N.getSourceRange());
431 }
else if (
auto *S = llvm::dyn_cast<SequenceNode>(&N)) {
433 for (
auto &Child : *S) {
434 if (
auto Value = scalarValue(Child,
"List item"))
435 Result.push_back(std::move(*
Value));
438 warning(
"Expected scalar or list of scalars", N);
445 void error(
const llvm::Twine &Msg, llvm::SMRange Range) {
447 SM.PrintMessage(
Range.Start, llvm::SourceMgr::DK_Error, Msg, Range);
449 void error(
const llvm::Twine &Msg,
const Node &N) {
450 return error(Msg, N.getSourceRange());
454 void warning(
const llvm::Twine &Msg, llvm::SMRange Range) {
455 SM.PrintMessage(
Range.Start, llvm::SourceMgr::DK_Warning, Msg, Range);
457 void warning(
const llvm::Twine &Msg,
const Node &N) {
458 return warning(Msg, N.getSourceRange());
465 llvm::StringRef BufferName,
469 auto SM = std::make_shared<llvm::SourceMgr>();
470 auto Buf = llvm::MemoryBuffer::getMemBufferCopy(
YAML, BufferName);
474 [](
const llvm::SMDiagnostic &
Diag,
void *Ctx) {
478 std::vector<Fragment> Result;
479 for (
auto &Doc : llvm::yaml::Stream(*Buf, *SM)) {
480 if (
Node *N = Doc.getRoot()) {
485 "Parsing config fragment");
486 if (Parser(*SM).parse(
Fragment, *N))
487 Result.push_back(std::move(
Fragment));
490 SM->PrintMessage(SM->FindLocForLineAndColumn(SM->getMainFileID(), 0, 0),
491 llvm::SourceMgr::DK_Note,
492 "Parsed " + llvm::Twine(Result.size()) +
493 " fragments from file");
496 SM->AddNewSourceBuffer(std::move(Buf), llvm::SMLoc());
CharSourceRange Range
SourceRange for the file name.
::clang::DynTypedNode Node
llvm::function_ref< void(const llvm::SMDiagnostic &)> DiagnosticCallback
Used to report problems in parsing or interpreting a config.
llvm::Error error(std::error_code EC, const char *Fmt, Ts &&... Vals)
===– Representation.cpp - ClangDoc Representation --------—*- C++ -*-===//
A top-level diagnostic that may have Notes and Fixes.
std::shared_ptr< llvm::SourceMgr > Manager
Retains a buffer of the original source this fragment was parsed from.
llvm::SMLoc Location
The start of the original source for this fragment.
A chunk of configuration obtained from a config file, LSP, or elsewhere.
static std::vector< Fragment > parseYAML(llvm::StringRef YAML, llvm::StringRef BufferName, DiagnosticCallback)
Parses fragments from a YAML file (one from each — delimited document).