2020-06-23 17:21:56 +02:00
|
|
|
//===-- ConfigYAMLTests.cpp -----------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "Annotations.h"
|
|
|
|
#include "ConfigFragment.h"
|
2020-06-26 01:49:53 +02:00
|
|
|
#include "ConfigTesting.h"
|
2020-06-23 17:21:56 +02:00
|
|
|
#include "Protocol.h"
|
2020-10-01 12:29:55 +01:00
|
|
|
#include "llvm/ADT/StringRef.h"
|
2020-06-23 17:21:56 +02:00
|
|
|
#include "llvm/Support/SMLoc.h"
|
|
|
|
#include "llvm/Support/SourceMgr.h"
|
2021-03-05 13:57:04 +01:00
|
|
|
#include "llvm/Testing/Support/SupportHelpers.h"
|
2020-06-23 17:21:56 +02:00
|
|
|
#include "gmock/gmock.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace clangd {
|
|
|
|
namespace config {
|
2022-02-01 10:14:07 +00:00
|
|
|
|
|
|
|
// PrintTo is a magic identifier of GTest
|
|
|
|
// NOLINTNEXTLINE (readability-identifier-naming)
|
2020-06-23 17:21:56 +02:00
|
|
|
template <typename T> void PrintTo(const Located<T> &V, std::ostream *OS) {
|
|
|
|
*OS << ::testing::PrintToString(*V);
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
using ::testing::AllOf;
|
|
|
|
using ::testing::ElementsAre;
|
|
|
|
using ::testing::IsEmpty;
|
|
|
|
|
2022-02-01 10:14:07 +00:00
|
|
|
MATCHER_P(val, Value, "") {
|
2020-06-23 17:21:56 +02:00
|
|
|
if (*arg == Value)
|
|
|
|
return true;
|
|
|
|
*result_listener << "value is " << *arg;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-11-22 10:04:00 +00:00
|
|
|
MATCHER_P2(PairVal, Value1, Value2, "") {
|
|
|
|
if (*arg.first == Value1 && *arg.second == Value2)
|
|
|
|
return true;
|
|
|
|
*result_listener << "values are [" << *arg.first << ", " << *arg.second
|
|
|
|
<< "]";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-06-23 17:21:56 +02:00
|
|
|
TEST(ParseYAML, SyntacticForms) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
const char *YAML = R"yaml(
|
|
|
|
If:
|
|
|
|
PathMatch:
|
|
|
|
- 'abc'
|
|
|
|
CompileFlags: { Add: [foo, bar] }
|
|
|
|
---
|
|
|
|
CompileFlags:
|
|
|
|
Add: |
|
|
|
|
b
|
|
|
|
az
|
2020-09-15 20:13:00 +02:00
|
|
|
---
|
|
|
|
Index:
|
|
|
|
Background: Skip
|
2020-11-22 10:04:00 +00:00
|
|
|
---
|
2021-01-25 16:16:22 +01:00
|
|
|
Diagnostics:
|
|
|
|
ClangTidy:
|
|
|
|
CheckOptions:
|
|
|
|
IgnoreMacros: true
|
|
|
|
example-check.ExampleOption: 0
|
2021-10-26 11:53:34 +02:00
|
|
|
UnusedIncludes: Strict
|
2020-06-23 17:21:56 +02:00
|
|
|
)yaml";
|
|
|
|
auto Results = Fragment::parseYAML(YAML, "config.yaml", Diags.callback());
|
|
|
|
EXPECT_THAT(Diags.Diagnostics, IsEmpty());
|
2020-12-05 00:39:21 +01:00
|
|
|
EXPECT_THAT(Diags.Files, ElementsAre("config.yaml"));
|
2020-11-22 10:04:00 +00:00
|
|
|
ASSERT_EQ(Results.size(), 4u);
|
2020-09-15 20:13:00 +02:00
|
|
|
EXPECT_FALSE(Results[0].If.HasUnrecognizedCondition);
|
2022-02-01 10:14:07 +00:00
|
|
|
EXPECT_THAT(Results[0].If.PathMatch, ElementsAre(val("abc")));
|
|
|
|
EXPECT_THAT(Results[0].CompileFlags.Add, ElementsAre(val("foo"), val("bar")));
|
2020-09-15 20:13:00 +02:00
|
|
|
|
2022-02-01 10:14:07 +00:00
|
|
|
EXPECT_THAT(Results[1].CompileFlags.Add, ElementsAre(val("b\naz\n")));
|
2020-06-23 17:21:56 +02:00
|
|
|
|
2020-09-15 20:13:00 +02:00
|
|
|
ASSERT_TRUE(Results[2].Index.Background);
|
2022-12-17 05:11:01 +00:00
|
|
|
EXPECT_EQ("Skip", **Results[2].Index.Background);
|
2021-01-25 16:16:22 +01:00
|
|
|
EXPECT_THAT(Results[3].Diagnostics.ClangTidy.CheckOptions,
|
2020-11-22 10:04:00 +00:00
|
|
|
ElementsAre(PairVal("IgnoreMacros", "true"),
|
|
|
|
PairVal("example-check.ExampleOption", "0")));
|
2021-10-26 11:53:34 +02:00
|
|
|
EXPECT_TRUE(Results[3].Diagnostics.UnusedIncludes);
|
2022-12-17 05:11:01 +00:00
|
|
|
EXPECT_EQ("Strict", **Results[3].Diagnostics.UnusedIncludes);
|
2020-06-23 17:21:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseYAML, Locations) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
If:
|
|
|
|
PathMatch: [['???bad***regex(((']]
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
EXPECT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
ASSERT_NE(Results.front().Source.Manager, nullptr);
|
2020-06-26 01:49:53 +02:00
|
|
|
EXPECT_EQ(toRange(Results.front().If.PathMatch.front().Range,
|
2020-06-23 17:21:56 +02:00
|
|
|
*Results.front().Source.Manager),
|
|
|
|
YAML.range());
|
|
|
|
}
|
|
|
|
|
2021-01-25 14:03:13 +01:00
|
|
|
TEST(ParseYAML, ConfigDiagnostics) {
|
2020-06-23 17:21:56 +02:00
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
If:
|
2020-11-22 10:04:00 +00:00
|
|
|
$unknown[[UnknownCondition]]: "foo"
|
2020-06-23 17:21:56 +02:00
|
|
|
CompileFlags:
|
|
|
|
Add: 'first'
|
|
|
|
---
|
2020-11-22 10:04:00 +00:00
|
|
|
CompileFlags: {$unexpected^
|
2020-06-23 17:21:56 +02:00
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
|
|
|
|
ASSERT_THAT(
|
|
|
|
Diags.Diagnostics,
|
2022-02-01 10:14:07 +00:00
|
|
|
ElementsAre(AllOf(diagMessage("Unknown If key 'UnknownCondition'"),
|
|
|
|
diagKind(llvm::SourceMgr::DK_Warning),
|
|
|
|
diagPos(YAML.range("unknown").start),
|
|
|
|
diagRange(YAML.range("unknown"))),
|
|
|
|
AllOf(diagMessage("Unexpected token. Expected Key, Flow "
|
2020-06-23 17:21:56 +02:00
|
|
|
"Entry, or Flow Mapping End."),
|
2022-02-01 10:14:07 +00:00
|
|
|
diagKind(llvm::SourceMgr::DK_Error),
|
|
|
|
diagPos(YAML.point("unexpected")),
|
2022-12-05 21:49:31 -08:00
|
|
|
diagRange(std::nullopt))));
|
2020-06-23 17:21:56 +02:00
|
|
|
|
2020-07-09 00:13:54 +02:00
|
|
|
ASSERT_EQ(Results.size(), 1u); // invalid fragment discarded.
|
2022-02-01 10:14:07 +00:00
|
|
|
EXPECT_THAT(Results.front().CompileFlags.Add, ElementsAre(val("first")));
|
2020-06-26 01:49:53 +02:00
|
|
|
EXPECT_TRUE(Results.front().If.HasUnrecognizedCondition);
|
2020-07-09 00:13:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseYAML, Invalid) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
const char *YAML = R"yaml(
|
|
|
|
If:
|
|
|
|
|
|
|
|
horrible
|
|
|
|
---
|
|
|
|
- 1
|
|
|
|
)yaml";
|
|
|
|
auto Results = Fragment::parseYAML(YAML, "config.yaml", Diags.callback());
|
|
|
|
EXPECT_THAT(Diags.Diagnostics,
|
2022-02-01 10:14:07 +00:00
|
|
|
ElementsAre(diagMessage("If should be a dictionary"),
|
|
|
|
diagMessage("Config should be a dictionary")));
|
2020-07-09 00:13:54 +02:00
|
|
|
ASSERT_THAT(Results, IsEmpty());
|
2020-06-23 17:21:56 +02:00
|
|
|
}
|
|
|
|
|
2021-04-13 10:26:03 +02:00
|
|
|
TEST(ParseYAML, ExternalBlockNone) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Index:
|
|
|
|
External: None
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
ASSERT_TRUE(Results[0].Index.External);
|
2022-12-17 05:11:01 +00:00
|
|
|
EXPECT_FALSE((*Results[0].Index.External)->File.has_value());
|
|
|
|
EXPECT_FALSE((*Results[0].Index.External)->MountPoint.has_value());
|
|
|
|
EXPECT_FALSE((*Results[0].Index.External)->Server.has_value());
|
|
|
|
EXPECT_THAT(*(*Results[0].Index.External)->IsNone, testing::Eq(true));
|
2021-04-13 10:26:03 +02:00
|
|
|
}
|
|
|
|
|
2020-10-01 12:29:55 +01:00
|
|
|
TEST(ParseYAML, ExternalBlock) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Index:
|
|
|
|
External:
|
|
|
|
File: "foo"
|
|
|
|
Server: ^"bar"
|
|
|
|
MountPoint: "baz"
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
ASSERT_TRUE(Results[0].Index.External);
|
2022-12-17 05:11:01 +00:00
|
|
|
EXPECT_THAT(*(*Results[0].Index.External)->File, val("foo"));
|
|
|
|
EXPECT_THAT(*(*Results[0].Index.External)->MountPoint, val("baz"));
|
2020-10-01 12:29:55 +01:00
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
2022-12-17 05:11:01 +00:00
|
|
|
EXPECT_THAT(*(*Results[0].Index.External)->Server, val("bar"));
|
2020-10-01 12:29:55 +01:00
|
|
|
}
|
|
|
|
|
2021-03-05 13:57:04 +01:00
|
|
|
TEST(ParseYAML, AllScopes) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Completion:
|
|
|
|
AllScopes: True
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
2022-02-01 10:14:07 +00:00
|
|
|
EXPECT_THAT(Results[0].Completion.AllScopes, llvm::ValueIs(val(true)));
|
2021-03-05 13:57:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseYAML, AllScopesWarn) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Completion:
|
|
|
|
AllScopes: $diagrange[[Truex]]
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
EXPECT_THAT(Diags.Diagnostics,
|
2022-02-01 10:14:07 +00:00
|
|
|
ElementsAre(AllOf(diagMessage("AllScopes should be a boolean"),
|
|
|
|
diagKind(llvm::SourceMgr::DK_Warning),
|
|
|
|
diagPos(YAML.range("diagrange").start),
|
|
|
|
diagRange(YAML.range("diagrange")))));
|
2021-03-05 13:57:04 +01:00
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
2022-12-05 21:49:31 -08:00
|
|
|
EXPECT_THAT(Results[0].Completion.AllScopes, testing::Eq(std::nullopt));
|
2021-03-05 13:57:04 +01:00
|
|
|
}
|
2021-12-08 13:52:15 +08:00
|
|
|
|
|
|
|
TEST(ParseYAML, ShowAKA) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Hover:
|
|
|
|
ShowAKA: True
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
2022-02-01 10:14:07 +00:00
|
|
|
EXPECT_THAT(Results[0].Hover.ShowAKA, llvm::ValueIs(val(true)));
|
2021-12-08 13:52:15 +08:00
|
|
|
}
|
2022-01-06 02:01:13 +01:00
|
|
|
|
|
|
|
TEST(ParseYAML, InlayHints) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
InlayHints:
|
|
|
|
Enabled: No
|
|
|
|
ParameterNames: Yes
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
2022-02-01 10:14:07 +00:00
|
|
|
EXPECT_THAT(Results[0].InlayHints.Enabled, llvm::ValueIs(val(false)));
|
|
|
|
EXPECT_THAT(Results[0].InlayHints.ParameterNames, llvm::ValueIs(val(true)));
|
2022-12-05 21:49:31 -08:00
|
|
|
EXPECT_EQ(Results[0].InlayHints.DeducedTypes, std::nullopt);
|
2022-01-06 02:01:13 +01:00
|
|
|
}
|
|
|
|
|
2023-05-26 03:15:10 -04:00
|
|
|
TEST(ParseYAML, SemanticTokens) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
SemanticTokens:
|
|
|
|
DisabledKinds: [ Operator, InactiveCode]
|
|
|
|
DisabledModifiers: Readonly
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
EXPECT_THAT(Results[0].SemanticTokens.DisabledKinds,
|
|
|
|
ElementsAre(val("Operator"), val("InactiveCode")));
|
|
|
|
EXPECT_THAT(Results[0].SemanticTokens.DisabledModifiers,
|
|
|
|
ElementsAre(val("Readonly")));
|
|
|
|
}
|
|
|
|
|
2022-05-06 14:32:04 +02:00
|
|
|
TEST(ParseYAML, IncludesIgnoreHeader) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Diagnostics:
|
|
|
|
Includes:
|
|
|
|
IgnoreHeader: [foo, bar]
|
|
|
|
)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
EXPECT_THAT(Results[0].Diagnostics.Includes.IgnoreHeader,
|
|
|
|
ElementsAre(val("foo"), val("bar")));
|
|
|
|
}
|
2022-06-14 17:08:37 +02:00
|
|
|
|
|
|
|
TEST(ParseYAML, Style) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Style:
|
|
|
|
FullyQualifiedNamespaces: [foo, bar])yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
EXPECT_THAT(Results[0].Style.FullyQualifiedNamespaces,
|
|
|
|
ElementsAre(val("foo"), val("bar")));
|
|
|
|
}
|
2023-01-30 08:06:41 +01:00
|
|
|
|
|
|
|
TEST(ParseYAML, DiagnosticsMode) {
|
|
|
|
CapturedDiags Diags;
|
|
|
|
{
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Diagnostics:
|
|
|
|
AllowStalePreamble: Yes)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
EXPECT_THAT(Results[0].Diagnostics.AllowStalePreamble,
|
|
|
|
llvm::ValueIs(val(true)));
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Annotations YAML(R"yaml(
|
|
|
|
Diagnostics:
|
|
|
|
AllowStalePreamble: No)yaml");
|
|
|
|
auto Results =
|
|
|
|
Fragment::parseYAML(YAML.code(), "config.yaml", Diags.callback());
|
|
|
|
ASSERT_THAT(Diags.Diagnostics, IsEmpty());
|
|
|
|
ASSERT_EQ(Results.size(), 1u);
|
|
|
|
EXPECT_THAT(Results[0].Diagnostics.AllowStalePreamble,
|
|
|
|
llvm::ValueIs(val(false)));
|
|
|
|
}
|
|
|
|
}
|
2020-06-23 17:21:56 +02:00
|
|
|
} // namespace
|
|
|
|
} // namespace config
|
|
|
|
} // namespace clangd
|
|
|
|
} // namespace clang
|