|  | //===-- CodeCompleteTests.cpp -----------------------------------*- C++ -*-===// | 
|  | // | 
|  | // 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 "ASTSignals.h" | 
|  | #include "Annotations.h" | 
|  | #include "ClangdServer.h" | 
|  | #include "CodeComplete.h" | 
|  | #include "Compiler.h" | 
|  | #include "Feature.h" | 
|  | #include "Matchers.h" | 
|  | #include "Protocol.h" | 
|  | #include "Quality.h" | 
|  | #include "SourceCode.h" | 
|  | #include "SyncAPI.h" | 
|  | #include "TestFS.h" | 
|  | #include "TestIndex.h" | 
|  | #include "TestTU.h" | 
|  | #include "index/Index.h" | 
|  | #include "index/MemIndex.h" | 
|  | #include "index/SymbolOrigin.h" | 
|  | #include "support/Threading.h" | 
|  | #include "clang/Sema/CodeCompleteConsumer.h" | 
|  | #include "clang/Tooling/CompilationDatabase.h" | 
|  | #include "llvm/ADT/StringRef.h" | 
|  | #include "llvm/Support/Error.h" | 
|  | #include "llvm/Support/Path.h" | 
|  | #include "llvm/Testing/Annotations/Annotations.h" | 
|  | #include "llvm/Testing/Support/Error.h" | 
|  | #include "llvm/Testing/Support/SupportHelpers.h" | 
|  | #include "gmock/gmock.h" | 
|  | #include "gtest/gtest.h" | 
|  | #include <condition_variable> | 
|  | #include <functional> | 
|  | #include <mutex> | 
|  | #include <vector> | 
|  |  | 
|  | namespace clang { | 
|  | namespace clangd { | 
|  |  | 
|  | namespace { | 
|  | using ::llvm::Failed; | 
|  | using ::testing::AllOf; | 
|  | using ::testing::Contains; | 
|  | using ::testing::ElementsAre; | 
|  | using ::testing::Field; | 
|  | using ::testing::HasSubstr; | 
|  | using ::testing::IsEmpty; | 
|  | using ::testing::Not; | 
|  | using ::testing::UnorderedElementsAre; | 
|  | using ContextKind = CodeCompletionContext::Kind; | 
|  |  | 
|  | // GMock helpers for matching completion items. | 
|  | MATCHER_P(named, Name, "") { return arg.Name == Name; } | 
|  | MATCHER_P(mainFileRefs, Refs, "") { return arg.MainFileRefs == Refs; } | 
|  | MATCHER_P(scopeRefs, Refs, "") { return arg.ScopeRefsInFile == Refs; } | 
|  | MATCHER_P(nameStartsWith, Prefix, "") { | 
|  | return llvm::StringRef(arg.Name).starts_with(Prefix); | 
|  | } | 
|  | MATCHER_P(filterText, F, "") { return arg.FilterText == F; } | 
|  | MATCHER_P(scope, S, "") { return arg.Scope == S; } | 
|  | MATCHER_P(qualifier, Q, "") { return arg.RequiredQualifier == Q; } | 
|  | MATCHER_P(labeled, Label, "") { | 
|  | return arg.RequiredQualifier + arg.Name + arg.Signature == Label; | 
|  | } | 
|  | MATCHER_P(sigHelpLabeled, Label, "") { return arg.label == Label; } | 
|  | MATCHER_P(kind, K, "") { return arg.Kind == K; } | 
|  | MATCHER_P(doc, D, "") { | 
|  | return arg.Documentation && arg.Documentation->asPlainText() == D; | 
|  | } | 
|  | MATCHER_P(returnType, D, "") { return arg.ReturnType == D; } | 
|  | MATCHER_P(hasInclude, IncludeHeader, "") { | 
|  | return !arg.Includes.empty() && arg.Includes[0].Header == IncludeHeader; | 
|  | } | 
|  | MATCHER_P(insertInclude, IncludeHeader, "") { | 
|  | return !arg.Includes.empty() && arg.Includes[0].Header == IncludeHeader && | 
|  | bool(arg.Includes[0].Insertion); | 
|  | } | 
|  | MATCHER_P(insertIncludeText, InsertedText, "") { | 
|  | return !arg.Includes.empty() && arg.Includes[0].Insertion && | 
|  | arg.Includes[0].Insertion->newText == InsertedText; | 
|  | } | 
|  | MATCHER(insertInclude, "") { | 
|  | return !arg.Includes.empty() && bool(arg.Includes[0].Insertion); | 
|  | } | 
|  | MATCHER_P(snippetSuffix, Text, "") { return arg.SnippetSuffix == Text; } | 
|  | MATCHER_P(origin, OriginSet, "") { return arg.Origin == OriginSet; } | 
|  | MATCHER_P(signature, S, "") { return arg.Signature == S; } | 
|  | MATCHER_P(replacesRange, Range, "") { | 
|  | return arg.CompletionTokenRange == Range; | 
|  | } | 
|  |  | 
|  | // Shorthand for Contains(named(Name)). | 
|  | Matcher<const std::vector<CodeCompletion> &> has(std::string Name) { | 
|  | return Contains(named(std::move(Name))); | 
|  | } | 
|  | Matcher<const std::vector<CodeCompletion> &> has(std::string Name, | 
|  | CompletionItemKind K) { | 
|  | return Contains(AllOf(named(std::move(Name)), kind(K))); | 
|  | } | 
|  | MATCHER(isDocumented, "") { return arg.Documentation.has_value(); } | 
|  | MATCHER(deprecated, "") { return arg.Deprecated; } | 
|  |  | 
|  | std::unique_ptr<SymbolIndex> memIndex(std::vector<Symbol> Symbols) { | 
|  | SymbolSlab::Builder Slab; | 
|  | for (const auto &Sym : Symbols) | 
|  | Slab.insert(Sym); | 
|  | return MemIndex::build(std::move(Slab).build(), RefSlab(), RelationSlab()); | 
|  | } | 
|  |  | 
|  | // Runs code completion. | 
|  | // If IndexSymbols is non-empty, an index will be built and passed to opts. | 
|  | CodeCompleteResult completions(const TestTU &TU, Position Point, | 
|  | std::vector<Symbol> IndexSymbols = {}, | 
|  | clangd::CodeCompleteOptions Opts = {}) { | 
|  | std::unique_ptr<SymbolIndex> OverrideIndex; | 
|  | if (!IndexSymbols.empty()) { | 
|  | assert(!Opts.Index && "both Index and IndexSymbols given!"); | 
|  | OverrideIndex = memIndex(std::move(IndexSymbols)); | 
|  | Opts.Index = OverrideIndex.get(); | 
|  | } | 
|  |  | 
|  | MockFS FS; | 
|  | auto Inputs = TU.inputs(FS); | 
|  | IgnoreDiagnostics Diags; | 
|  | auto CI = buildCompilerInvocation(Inputs, Diags); | 
|  | if (!CI) { | 
|  | ADD_FAILURE() << "Couldn't build CompilerInvocation"; | 
|  | return {}; | 
|  | } | 
|  | auto Preamble = buildPreamble(testPath(TU.Filename), *CI, Inputs, | 
|  | /*InMemory=*/true, /*Callback=*/nullptr); | 
|  | return codeComplete(testPath(TU.Filename), Point, Preamble.get(), Inputs, | 
|  | Opts); | 
|  | } | 
|  |  | 
|  | // Runs code completion. | 
|  | CodeCompleteResult completions(llvm::StringRef Text, | 
|  | std::vector<Symbol> IndexSymbols = {}, | 
|  | clangd::CodeCompleteOptions Opts = {}, | 
|  | PathRef FilePath = "foo.cpp") { | 
|  | Annotations Test(Text); | 
|  | auto TU = TestTU::withCode(Test.code()); | 
|  | // To make sure our tests for completiopns inside templates work on Windows. | 
|  | TU.Filename = FilePath.str(); | 
|  | return completions(TU, Test.point(), std::move(IndexSymbols), | 
|  | std::move(Opts)); | 
|  | } | 
|  |  | 
|  | // Runs code completion without the clang parser. | 
|  | CodeCompleteResult completionsNoCompile(llvm::StringRef Text, | 
|  | std::vector<Symbol> IndexSymbols = {}, | 
|  | clangd::CodeCompleteOptions Opts = {}, | 
|  | PathRef FilePath = "foo.cpp") { | 
|  | std::unique_ptr<SymbolIndex> OverrideIndex; | 
|  | if (!IndexSymbols.empty()) { | 
|  | assert(!Opts.Index && "both Index and IndexSymbols given!"); | 
|  | OverrideIndex = memIndex(std::move(IndexSymbols)); | 
|  | Opts.Index = OverrideIndex.get(); | 
|  | } | 
|  |  | 
|  | MockFS FS; | 
|  | Annotations Test(Text); | 
|  | ParseInputs ParseInput{tooling::CompileCommand(), &FS, Test.code().str()}; | 
|  | return codeComplete(FilePath, Test.point(), /*Preamble=*/nullptr, ParseInput, | 
|  | Opts); | 
|  | } | 
|  |  | 
|  | Symbol withReferences(int N, Symbol S) { | 
|  | S.References = N; | 
|  | return S; | 
|  | } | 
|  |  | 
|  | #if CLANGD_DECISION_FOREST | 
|  | TEST(DecisionForestRankingModel, NameMatchSanityTest) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.RankingModel = CodeCompleteOptions::DecisionForest; | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | struct MemberAccess { | 
|  | int ABG(); | 
|  | int AlphaBetaGamma(); | 
|  | }; | 
|  | int func() { MemberAccess().ABG^ } | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(named("ABG"), named("AlphaBetaGamma"))); | 
|  | } | 
|  |  | 
|  | TEST(DecisionForestRankingModel, ReferencesAffectRanking) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.RankingModel = CodeCompleteOptions::DecisionForest; | 
|  | constexpr int NumReferences = 100000; | 
|  | EXPECT_THAT( | 
|  | completions("int main() { clang^ }", | 
|  | {ns("clangA"), withReferences(NumReferences, func("clangD"))}, | 
|  | Opts) | 
|  | .Completions, | 
|  | ElementsAre(named("clangD"), named("clangA"))); | 
|  | EXPECT_THAT( | 
|  | completions("int main() { clang^ }", | 
|  | {withReferences(NumReferences, ns("clangA")), func("clangD")}, | 
|  | Opts) | 
|  | .Completions, | 
|  | ElementsAre(named("clangA"), named("clangD"))); | 
|  | } | 
|  | #endif // CLANGD_DECISION_FOREST | 
|  |  | 
|  | TEST(DecisionForestRankingModel, DecisionForestScorerCallbackTest) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | constexpr float MagicNumber = 1234.5678f; | 
|  | Opts.RankingModel = CodeCompleteOptions::DecisionForest; | 
|  | Opts.DecisionForestScorer = [&](const SymbolQualitySignals &, | 
|  | const SymbolRelevanceSignals &, float Base) { | 
|  | DecisionForestScores Scores; | 
|  | Scores.Total = MagicNumber; | 
|  | Scores.ExcludingName = MagicNumber; | 
|  | return Scores; | 
|  | }; | 
|  | llvm::StringRef Code = "int func() { int xyz; xy^ }"; | 
|  | auto Results = completions(Code, | 
|  | /*IndexSymbols=*/{}, Opts); | 
|  | ASSERT_EQ(Results.Completions.size(), 1u); | 
|  | EXPECT_EQ(Results.Completions[0].Score.Total, MagicNumber); | 
|  | EXPECT_EQ(Results.Completions[0].Score.ExcludingName, MagicNumber); | 
|  |  | 
|  | // Do not use DecisionForestScorer for heuristics model. | 
|  | Opts.RankingModel = CodeCompleteOptions::Heuristics; | 
|  | Results = completions(Code, | 
|  | /*IndexSymbols=*/{}, Opts); | 
|  | ASSERT_EQ(Results.Completions.size(), 1u); | 
|  | EXPECT_NE(Results.Completions[0].Score.Total, MagicNumber); | 
|  | EXPECT_NE(Results.Completions[0].Score.ExcludingName, MagicNumber); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Limit) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.Limit = 2; | 
|  | auto Results = completions(R"cpp( | 
|  | struct ClassWithMembers { | 
|  | int AAA(); | 
|  | int BBB(); | 
|  | int CCC(); | 
|  | }; | 
|  |  | 
|  | int main() { ClassWithMembers().^ } | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{}, Opts); | 
|  |  | 
|  | EXPECT_TRUE(Results.HasMore); | 
|  | EXPECT_THAT(Results.Completions, ElementsAre(named("AAA"), named("BBB"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Filter) { | 
|  | std::string Body = R"cpp( | 
|  | #define MotorCar | 
|  | int Car; | 
|  | struct S { | 
|  | int FooBar; | 
|  | int FooBaz; | 
|  | int Qux; | 
|  | }; | 
|  | )cpp"; | 
|  |  | 
|  | // Only items matching the fuzzy query are returned. | 
|  | EXPECT_THAT(completions(Body + "int main() { S().Foba^ }").Completions, | 
|  | AllOf(has("FooBar"), has("FooBaz"), Not(has("Qux")))); | 
|  |  | 
|  | // Macros require prefix match, either from index or AST. | 
|  | Symbol Sym = var("MotorCarIndex"); | 
|  | Sym.SymInfo.Kind = index::SymbolKind::Macro; | 
|  | EXPECT_THAT( | 
|  | completions(Body + "int main() { C^ }", {Sym}).Completions, | 
|  | AllOf(has("Car"), Not(has("MotorCar")), Not(has("MotorCarIndex")))); | 
|  | EXPECT_THAT(completions(Body + "int main() { M^ }", {Sym}).Completions, | 
|  | AllOf(has("MotorCar"), has("MotorCarIndex"))); | 
|  | } | 
|  |  | 
|  | void testAfterDotCompletion(clangd::CodeCompleteOptions Opts) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | int global_var; | 
|  |  | 
|  | int global_func(); | 
|  |  | 
|  | // Make sure this is not in preamble. | 
|  | #define MACRO X | 
|  |  | 
|  | struct GlobalClass {}; | 
|  |  | 
|  | struct ClassWithMembers { | 
|  | /// doc for method. | 
|  | int method(); | 
|  |  | 
|  | int field; | 
|  | private: | 
|  | int private_field; | 
|  | }; | 
|  |  | 
|  | int test() { | 
|  | struct LocalClass {}; | 
|  |  | 
|  | /// doc for local_var. | 
|  | int local_var; | 
|  |  | 
|  | ClassWithMembers().^ | 
|  | } | 
|  | )cpp", | 
|  | {cls("IndexClass"), var("index_var"), func("index_func")}, Opts); | 
|  |  | 
|  | EXPECT_TRUE(Results.RanParser); | 
|  | // Class members. The only items that must be present in after-dot | 
|  | // completion. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(has("method"), has("field"), Not(has("ClassWithMembers")), | 
|  | Not(has("operator=")), Not(has("~ClassWithMembers")))); | 
|  | EXPECT_IFF(Opts.IncludeIneligibleResults, Results.Completions, | 
|  | has("private_field")); | 
|  | // Global items. | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | Not(AnyOf(has("global_var"), has("index_var"), has("global_func"), | 
|  | has("global_func()"), has("index_func"), has("GlobalClass"), | 
|  | has("IndexClass"), has("MACRO"), has("LocalClass")))); | 
|  | // There should be no code patterns (aka snippets) in after-dot | 
|  | // completion. At least there aren't any we're aware of. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Not(Contains(kind(CompletionItemKind::Snippet)))); | 
|  | // Check documentation. | 
|  | EXPECT_THAT(Results.Completions, Contains(isDocumented())); | 
|  | } | 
|  |  | 
|  | void testGlobalScopeCompletion(clangd::CodeCompleteOptions Opts) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | int global_var; | 
|  | int global_func(); | 
|  |  | 
|  | // Make sure this is not in preamble. | 
|  | #define MACRO X | 
|  |  | 
|  | struct GlobalClass {}; | 
|  |  | 
|  | struct ClassWithMembers { | 
|  | /// doc for method. | 
|  | int method(); | 
|  | }; | 
|  |  | 
|  | int test() { | 
|  | struct LocalClass {}; | 
|  |  | 
|  | /// doc for local_var. | 
|  | int local_var; | 
|  |  | 
|  | ^ | 
|  | } | 
|  | )cpp", | 
|  | {cls("IndexClass"), var("index_var"), func("index_func")}, Opts); | 
|  |  | 
|  | EXPECT_TRUE(Results.RanParser); | 
|  | // Class members. Should never be present in global completions. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Not(AnyOf(has("method"), has("method()"), has("field")))); | 
|  | // Global items. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(has("global_var"), has("index_var"), has("global_func"), | 
|  | has("index_func" /* our fake symbol doesn't include () */), | 
|  | has("GlobalClass"), has("IndexClass"))); | 
|  | // A macro. | 
|  | EXPECT_THAT(Results.Completions, has("MACRO")); | 
|  | // Local items. Must be present always. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(has("local_var"), has("LocalClass"), | 
|  | Contains(kind(CompletionItemKind::Snippet)))); | 
|  | // Check documentation. | 
|  | EXPECT_THAT(Results.Completions, Contains(isDocumented())); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompletionOptions) { | 
|  | auto Test = [&](const clangd::CodeCompleteOptions &Opts) { | 
|  | testAfterDotCompletion(Opts); | 
|  | testGlobalScopeCompletion(Opts); | 
|  | }; | 
|  | // We used to test every combination of options, but that got too slow (2^N). | 
|  | auto Flags = { | 
|  | &clangd::CodeCompleteOptions::IncludeIneligibleResults, | 
|  | }; | 
|  | // Test default options. | 
|  | Test({}); | 
|  | // Test with one flag flipped. | 
|  | for (auto &F : Flags) { | 
|  | clangd::CodeCompleteOptions O; | 
|  | O.*F ^= true; | 
|  | Test(O); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Accessible) { | 
|  | auto Internal = completions(R"cpp( | 
|  | class Foo { | 
|  | public: void pub(); | 
|  | protected: void prot(); | 
|  | private: void priv(); | 
|  | }; | 
|  | void Foo::pub() { this->^ } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Internal.Completions, | 
|  | AllOf(has("priv"), has("prot"), has("pub"))); | 
|  |  | 
|  | auto External = completions(R"cpp( | 
|  | class Foo { | 
|  | public: void pub(); | 
|  | protected: void prot(); | 
|  | private: void priv(); | 
|  | }; | 
|  | void test() { | 
|  | Foo F; | 
|  | F.^ | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(External.Completions, | 
|  | AllOf(has("pub"), Not(has("prot")), Not(has("priv")))); | 
|  |  | 
|  | auto Results = completions(R"cpp( | 
|  | struct Foo { | 
|  | public: void pub(); | 
|  | protected: void prot(); | 
|  | private: void priv(); | 
|  | }; | 
|  | struct Bar : public Foo { | 
|  | private: using Foo::pub; | 
|  | }; | 
|  | void test() { | 
|  | Bar B; | 
|  | B.^ | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(Not(has("priv")), Not(has("prot")), Not(has("pub")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Qualifiers) { | 
|  | auto Results = completions(R"cpp( | 
|  | class Foo { | 
|  | public: int foo() const; | 
|  | int bar() const; | 
|  | }; | 
|  | class Bar : public Foo { | 
|  | int foo() const; | 
|  | }; | 
|  | void test() { Bar().^ } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(qualifier(""), named("bar")))); | 
|  | // Hidden members are not shown. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Not(Contains(AllOf(qualifier("Foo::"), named("foo"))))); | 
|  | // Private members are not shown. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Not(Contains(AllOf(qualifier(""), named("foo"))))); | 
|  | } | 
|  |  | 
|  | // https://github.com/clangd/clangd/issues/1451 | 
|  | TEST(CompletionTest, QualificationWithInlineNamespace) { | 
|  | auto Results = completions(R"cpp( | 
|  | namespace a { inline namespace b {} } | 
|  | using namespace a::b; | 
|  | void f() { Foo^ } | 
|  | )cpp", | 
|  | {cls("a::Foo")}); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier("a::"), named("Foo")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, InjectedTypename) { | 
|  | // These are suppressed when accessed as a member... | 
|  | EXPECT_THAT(completions("struct X{}; void foo(){ X().^ }").Completions, | 
|  | Not(has("X"))); | 
|  | EXPECT_THAT(completions("struct X{ void foo(){ this->^ } };").Completions, | 
|  | Not(has("X"))); | 
|  | // ...but accessible in other, more useful cases. | 
|  | EXPECT_THAT(completions("struct X{ void foo(){ ^ } };").Completions, | 
|  | has("X")); | 
|  | EXPECT_THAT( | 
|  | completions("struct Y{}; struct X:Y{ void foo(){ ^ } };").Completions, | 
|  | has("Y")); | 
|  | EXPECT_THAT( | 
|  | completions( | 
|  | "template<class> struct Y{}; struct X:Y<int>{ void foo(){ ^ } };") | 
|  | .Completions, | 
|  | has("Y")); | 
|  | // This case is marginal (`using X::X` is useful), we allow it for now. | 
|  | EXPECT_THAT(completions("struct X{}; void foo(){ X::^ }").Completions, | 
|  | has("X")); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, SkipInjectedWhenUnqualified) { | 
|  | EXPECT_THAT(completions("struct X { void f() { X^ }};").Completions, | 
|  | ElementsAre(named("X"), named("~X"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Snippets) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | struct fake { | 
|  | int a; | 
|  | int f(int i, const float f) const; | 
|  | }; | 
|  | int main() { | 
|  | fake f; | 
|  | f.^ | 
|  | } | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{}, Opts); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | HasSubsequence(named("a"), | 
|  | snippetSuffix("(${1:int i}, ${2:const float f})"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, HeuristicsForMemberFunctionCompletion) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.EnableSnippets = true; | 
|  |  | 
|  | Annotations Code(R"cpp( | 
|  | struct Foo { | 
|  | static int staticMethod(int); | 
|  | int method(int) const; | 
|  | template <typename T, typename U, typename V = int> | 
|  | T generic(U, V); | 
|  | template <typename T, int U> | 
|  | static T staticGeneric(); | 
|  | Foo() { | 
|  | this->$canBeCall^ | 
|  | $canBeCall^ | 
|  | Foo::$canBeCall^ | 
|  | } | 
|  | }; | 
|  |  | 
|  | struct Derived : Foo { | 
|  | using Foo::method; | 
|  | using Foo::generic; | 
|  | Derived() { | 
|  | Foo::$canBeCall^ | 
|  | } | 
|  | }; | 
|  |  | 
|  | struct OtherClass { | 
|  | OtherClass() { | 
|  | Foo f; | 
|  | Derived d; | 
|  | f.$canBeCall^ | 
|  | ; // Prevent parsing as 'f.f' | 
|  | f.Foo::$canBeCall^ | 
|  | &Foo::$canNotBeCall^ | 
|  | ; | 
|  | d.Foo::$canBeCall^ | 
|  | ; | 
|  | d.Derived::$canBeCall^ | 
|  | } | 
|  | }; | 
|  |  | 
|  | int main() { | 
|  | Foo f; | 
|  | Derived d; | 
|  | f.$canBeCall^ | 
|  | ; // Prevent parsing as 'f.f' | 
|  | f.Foo::$canBeCall^ | 
|  | &Foo::$canNotBeCall^ | 
|  | ; | 
|  | d.Foo::$canBeCall^ | 
|  | ; | 
|  | d.Derived::$canBeCall^ | 
|  | } | 
|  | )cpp"); | 
|  | auto TU = TestTU::withCode(Code.code()); | 
|  |  | 
|  | for (const auto &P : Code.points("canNotBeCall")) { | 
|  | auto Results = completions(TU, P, /*IndexSymbols*/ {}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("method"), signature("(int) const"), | 
|  | snippetSuffix("")))); | 
|  | // We don't have any arguments to deduce against if this isn't a call. | 
|  | // Thus, we should emit these deducible template arguments explicitly. | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | Contains(AllOf(named("generic"), | 
|  | signature("<typename T, typename U>(U, V)"), | 
|  | snippetSuffix("<${1:typename T}, ${2:typename U}>")))); | 
|  | } | 
|  |  | 
|  | for (const auto &P : Code.points("canBeCall")) { | 
|  | auto Results = completions(TU, P, /*IndexSymbols*/ {}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("method"), signature("(int) const"), | 
|  | snippetSuffix("(${1:int})")))); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | Contains(AllOf(named("generic"), signature("<typename T>(U, V)"), | 
|  | snippetSuffix("<${1:typename T}>(${2:U}, ${3:V})")))); | 
|  | } | 
|  |  | 
|  | // static method will always keep the snippet | 
|  | for (const auto &P : Code.points()) { | 
|  | auto Results = completions(TU, P, /*IndexSymbols*/ {}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("staticMethod"), signature("(int)"), | 
|  | snippetSuffix("(${1:int})")))); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf( | 
|  | named("staticGeneric"), signature("<typename T, int U>()"), | 
|  | snippetSuffix("<${1:typename T}, ${2:int U}>()")))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoSnippetsInUsings) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.EnableSnippets = true; | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { | 
|  | int func(int a, int b); | 
|  | } | 
|  |  | 
|  | using ns::^; | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(AllOf(named("func"), labeled("func(int a, int b)"), | 
|  | snippetSuffix("")))); | 
|  |  | 
|  | // Check index completions too. | 
|  | auto Func = func("ns::func"); | 
|  | Func.CompletionSnippetSuffix = "(${1:int a}, ${2: int b})"; | 
|  | Func.Signature = "(int a, int b)"; | 
|  | Func.ReturnType = "void"; | 
|  |  | 
|  | Results = completions(R"cpp( | 
|  | namespace ns {} | 
|  | using ns::^; | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{Func}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(AllOf(named("func"), labeled("func(int a, int b)"), | 
|  | snippetSuffix("")))); | 
|  |  | 
|  | // Check all-scopes completions too. | 
|  | Opts.AllScopes = true; | 
|  | Results = completions(R"cpp( | 
|  | using ^; | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{Func}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("func"), labeled("ns::func(int a, int b)"), | 
|  | snippetSuffix("")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Kinds) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | int variable; | 
|  | struct Struct {}; | 
|  | int function(); | 
|  | // make sure MACRO is not included in preamble. | 
|  | #define MACRO 10 | 
|  | int X = ^ | 
|  | )cpp", | 
|  | {func("indexFunction"), var("indexVariable"), cls("indexClass"), | 
|  | macro("indexObjMacro"), macro("indexFuncMacro", "(x, y)")}); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(has("function", CompletionItemKind::Function), | 
|  | has("variable", CompletionItemKind::Variable), | 
|  | has("int", CompletionItemKind::Keyword), | 
|  | has("Struct", CompletionItemKind::Struct), | 
|  | has("MACRO", CompletionItemKind::Constant), | 
|  | has("indexFunction", CompletionItemKind::Function), | 
|  | has("indexVariable", CompletionItemKind::Variable), | 
|  | has("indexClass", CompletionItemKind::Class), | 
|  | has("indexObjMacro", CompletionItemKind::Constant), | 
|  | has("indexFuncMacro", CompletionItemKind::Function))); | 
|  |  | 
|  | Results = completions("nam^"); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | has("namespace", CompletionItemKind::Snippet)); | 
|  |  | 
|  | // Members of anonymous unions are of kind 'field'. | 
|  | Results = completions( | 
|  | R"cpp( | 
|  | struct X{ | 
|  | union { | 
|  | void *a; | 
|  | }; | 
|  | }; | 
|  | auto u = X().^ | 
|  | )cpp"); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre(AllOf(named("a"), kind(CompletionItemKind::Field)))); | 
|  |  | 
|  | // Completion kinds for templates should not be unknown. | 
|  | Results = completions( | 
|  | R"cpp( | 
|  | template <class T> struct complete_class {}; | 
|  | template <class T> void complete_function(); | 
|  | template <class T> using complete_type_alias = int; | 
|  | template <class T> int complete_variable = 10; | 
|  |  | 
|  | struct X { | 
|  | template <class T> static int complete_static_member = 10; | 
|  |  | 
|  | static auto x = complete_^ | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(named("complete_class"), kind(CompletionItemKind::Class)), | 
|  | AllOf(named("complete_function"), kind(CompletionItemKind::Function)), | 
|  | AllOf(named("complete_type_alias"), | 
|  | kind(CompletionItemKind::Interface)), | 
|  | AllOf(named("complete_variable"), kind(CompletionItemKind::Variable)), | 
|  | AllOf(named("complete_static_member"), | 
|  | kind(CompletionItemKind::Property)))); | 
|  |  | 
|  | Results = completions( | 
|  | R"cpp( | 
|  | enum Color { | 
|  | Red | 
|  | }; | 
|  | Color u = ^ | 
|  | )cpp"); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | Contains(AllOf(named("Red"), kind(CompletionItemKind::EnumMember)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoDuplicates) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | class Adapter { | 
|  | }; | 
|  |  | 
|  | void f() { | 
|  | Adapter^ | 
|  | } | 
|  | )cpp", | 
|  | {cls("Adapter")}); | 
|  |  | 
|  | // Make sure there are no duplicate entries of 'Adapter'. | 
|  | EXPECT_THAT(Results.Completions, ElementsAre(named("Adapter"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ScopedNoIndex) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace fake { int BigBang, Babble, Box; }; | 
|  | int main() { fake::ba^ } | 
|  | ")cpp"); | 
|  | // Babble is a better match than BigBang. Box doesn't match at all. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(named("Babble"), named("BigBang"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Scoped) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace fake { int Babble, Box; }; | 
|  | int main() { fake::ba^ } | 
|  | ")cpp", | 
|  | {var("fake::BigBang")}); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(named("Babble"), named("BigBang"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ScopedWithFilter) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | void f() { ns::x^ } | 
|  | )cpp", | 
|  | {cls("ns::XYZ"), func("ns::foo")}); | 
|  | EXPECT_THAT(Results.Completions, UnorderedElementsAre(named("XYZ"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ReferencesAffectRanking) { | 
|  | EXPECT_THAT(completions("int main() { abs^ }", {func("absA"), func("absB")}) | 
|  | .Completions, | 
|  | HasSubsequence(named("absA"), named("absB"))); | 
|  | EXPECT_THAT(completions("int main() { abs^ }", | 
|  | {func("absA"), withReferences(1000, func("absB"))}) | 
|  | .Completions, | 
|  | HasSubsequence(named("absB"), named("absA"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ContextWords) { | 
|  | auto Results = completions(R"cpp( | 
|  | enum class Color { RED, YELLOW, BLUE }; | 
|  |  | 
|  | // (blank lines so the definition above isn't "context") | 
|  |  | 
|  | // "It was a yellow car," he said. "Big yellow car, new." | 
|  | auto Finish = Color::^ | 
|  | )cpp"); | 
|  | // Yellow would normally sort last (alphabetic). | 
|  | // But the recent mention should bump it up. | 
|  | ASSERT_THAT(Results.Completions, | 
|  | HasSubsequence(named("YELLOW"), named("BLUE"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, GlobalQualified) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | void f() { ::^ } | 
|  | )cpp", | 
|  | {cls("XYZ")}); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(has("XYZ", CompletionItemKind::Class), | 
|  | has("f", CompletionItemKind::Function))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, FullyQualified) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { void bar(); } | 
|  | void f() { ::ns::^ } | 
|  | )cpp", | 
|  | {cls("ns::XYZ")}); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(has("XYZ", CompletionItemKind::Class), | 
|  | has("bar", CompletionItemKind::Function))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, SemaIndexMerge) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { int local; void both(); } | 
|  | void f() { ::ns::^ } | 
|  | )cpp", | 
|  | {func("ns::both"), cls("ns::Index")}); | 
|  | // We get results from both index and sema, with no duplicates. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(named("local"), origin(SymbolOrigin::AST)), | 
|  | AllOf(named("Index"), origin(SymbolOrigin::Static)), | 
|  | AllOf(named("both"), | 
|  | origin(SymbolOrigin::AST | SymbolOrigin::Static)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, SemaIndexMergeWithLimit) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.Limit = 1; | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { int local; void both(); } | 
|  | void f() { ::ns::^ } | 
|  | )cpp", | 
|  | {func("ns::both"), cls("ns::Index")}, Opts); | 
|  | EXPECT_EQ(Results.Completions.size(), Opts.Limit); | 
|  | EXPECT_TRUE(Results.HasMore); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, IncludeInsertionPreprocessorIntegrationTests) { | 
|  | TestTU TU; | 
|  | TU.ExtraArgs.push_back("-I" + testPath("sub")); | 
|  | TU.AdditionalFiles["sub/bar.h"] = ""; | 
|  | auto BarURI = URI::create(testPath("sub/bar.h")).toString(); | 
|  |  | 
|  | Symbol Sym = cls("ns::X"); | 
|  | Sym.CanonicalDeclaration.FileURI = BarURI.c_str(); | 
|  | Sym.IncludeHeaders.emplace_back(BarURI, 1, Symbol::Include); | 
|  | // Shorten include path based on search directory and insert. | 
|  | Annotations Test("int main() { ns::^ }"); | 
|  | TU.Code = Test.code().str(); | 
|  | auto Results = completions(TU, Test.point(), {Sym}); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(AllOf(named("X"), insertInclude("\"bar.h\"")))); | 
|  | // Can be disabled via option. | 
|  | CodeCompleteOptions NoInsertion; | 
|  | NoInsertion.InsertIncludes = CodeCompleteOptions::NeverInsert; | 
|  | Results = completions(TU, Test.point(), {Sym}, NoInsertion); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(AllOf(named("X"), Not(insertInclude())))); | 
|  | // Duplicate based on inclusions in preamble. | 
|  | Test = Annotations(R"cpp( | 
|  | #include "sub/bar.h"  // not shortest, so should only match resolved. | 
|  | int main() { ns::^ } | 
|  | )cpp"); | 
|  | TU.Code = Test.code().str(); | 
|  | Results = completions(TU, Test.point(), {Sym}); | 
|  | EXPECT_THAT(Results.Completions, ElementsAre(AllOf(named("X"), labeled("X"), | 
|  | Not(insertInclude())))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoIncludeInsertionWhenDeclFoundInFile) { | 
|  | Symbol SymX = cls("ns::X"); | 
|  | Symbol SymY = cls("ns::Y"); | 
|  | std::string BarHeader = testPath("bar.h"); | 
|  | auto BarURI = URI::create(BarHeader).toString(); | 
|  | SymX.CanonicalDeclaration.FileURI = BarURI.c_str(); | 
|  | SymY.CanonicalDeclaration.FileURI = BarURI.c_str(); | 
|  | SymX.IncludeHeaders.emplace_back("<bar>", 1, Symbol::Include); | 
|  | SymY.IncludeHeaders.emplace_back("<bar>", 1, Symbol::Include); | 
|  | // Shorten include path based on search directory and insert. | 
|  | auto Results = completions(R"cpp( | 
|  | namespace ns { | 
|  | class X; | 
|  | class Y {}; | 
|  | } | 
|  | int main() { ns::^ } | 
|  | )cpp", | 
|  | {SymX, SymY}); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(AllOf(named("X"), Not(insertInclude())), | 
|  | AllOf(named("Y"), Not(insertInclude())))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, IndexSuppressesPreambleCompletions) { | 
|  | Annotations Test(R"cpp( | 
|  | #include "bar.h" | 
|  | namespace ns { int local; } | 
|  | void f() { ns::^; } | 
|  | void f2() { ns::preamble().$2^; } | 
|  | )cpp"); | 
|  | auto TU = TestTU::withCode(Test.code()); | 
|  | TU.AdditionalFiles["bar.h"] = | 
|  | R"cpp(namespace ns { struct preamble { int member; }; })cpp"; | 
|  |  | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  | auto I = memIndex({var("ns::index")}); | 
|  | Opts.Index = I.get(); | 
|  | auto WithIndex = completions(TU, Test.point(), {}, Opts); | 
|  | EXPECT_THAT(WithIndex.Completions, | 
|  | UnorderedElementsAre(named("local"), named("index"))); | 
|  | auto ClassFromPreamble = completions(TU, Test.point("2"), {}, Opts); | 
|  | EXPECT_THAT(ClassFromPreamble.Completions, Contains(named("member"))); | 
|  |  | 
|  | Opts.Index = nullptr; | 
|  | auto WithoutIndex = completions(TU, Test.point(), {}, Opts); | 
|  | EXPECT_THAT(WithoutIndex.Completions, | 
|  | UnorderedElementsAre(named("local"), named("preamble"))); | 
|  | } | 
|  |  | 
|  | // This verifies that we get normal preprocessor completions in the preamble. | 
|  | // This is a regression test for an old bug: if we override the preamble and | 
|  | // try to complete inside it, clang kicks our completion point just outside the | 
|  | // preamble, resulting in always getting top-level completions. | 
|  | TEST(CompletionTest, CompletionInPreamble) { | 
|  | auto Results = completions(R"cpp( | 
|  | #ifnd^ef FOO_H_ | 
|  | #define BAR_H_ | 
|  | #include <bar.h> | 
|  | int foo() {} | 
|  | #endif | 
|  | )cpp") | 
|  | .Completions; | 
|  | EXPECT_THAT(Results, ElementsAre(named("ifndef"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompletionRecoveryASTType) { | 
|  | auto Results = completions(R"cpp( | 
|  | struct S { int member; }; | 
|  | S overloaded(int); | 
|  | void foo() { | 
|  | // No overload matches, but we have recovery-expr with the correct type. | 
|  | overloaded().^ | 
|  | })cpp") | 
|  | .Completions; | 
|  | EXPECT_THAT(Results, ElementsAre(named("member"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DynamicIndexIncludeInsertion) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  | ClangdServer::Options Opts = ClangdServer::optsForTest(); | 
|  | Opts.BuildDynamicSymbolIndex = true; | 
|  | ClangdServer Server(CDB, FS, Opts); | 
|  |  | 
|  | FS.Files[testPath("foo_header.h")] = R"cpp( | 
|  | #pragma once | 
|  | struct Foo { | 
|  | // Member doc | 
|  | int foo(); | 
|  | }; | 
|  | )cpp"; | 
|  | const std::string FileContent(R"cpp( | 
|  | #include "foo_header.h" | 
|  | int Foo::foo() { | 
|  | return 42; | 
|  | } | 
|  | )cpp"); | 
|  | Server.addDocument(testPath("foo_impl.cpp"), FileContent); | 
|  | // Wait for the dynamic index being built. | 
|  | ASSERT_TRUE(Server.blockUntilIdleForTest()); | 
|  |  | 
|  | auto File = testPath("foo.cpp"); | 
|  | Annotations Test("Foo^ foo;"); | 
|  | runAddDocument(Server, File, Test.code()); | 
|  | auto CompletionList = | 
|  | llvm::cantFail(runCodeComplete(Server, File, Test.point(), {})); | 
|  |  | 
|  | EXPECT_THAT(CompletionList.Completions, | 
|  | ElementsAre(AllOf(named("Foo"), hasInclude("\"foo_header.h\""), | 
|  | insertInclude()))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DynamicIndexMultiFile) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  | auto Opts = ClangdServer::optsForTest(); | 
|  | Opts.BuildDynamicSymbolIndex = true; | 
|  | ClangdServer Server(CDB, FS, Opts); | 
|  |  | 
|  | FS.Files[testPath("foo.h")] = R"cpp( | 
|  | namespace ns { class XYZ {}; void foo(int x) {} } | 
|  | )cpp"; | 
|  | runAddDocument(Server, testPath("foo.cpp"), R"cpp( | 
|  | #include "foo.h" | 
|  | )cpp"); | 
|  |  | 
|  | auto File = testPath("bar.cpp"); | 
|  | Annotations Test(R"cpp( | 
|  | namespace ns { | 
|  | class XXX {}; | 
|  | /// Doooc | 
|  | void fooooo() {} | 
|  | } | 
|  | void f() { ns::^ } | 
|  | )cpp"); | 
|  | runAddDocument(Server, File, Test.code()); | 
|  |  | 
|  | auto Results = cantFail(runCodeComplete(Server, File, Test.point(), {})); | 
|  | // "XYZ" and "foo" are not included in the file being completed but are still | 
|  | // visible through the index. | 
|  | EXPECT_THAT(Results.Completions, has("XYZ", CompletionItemKind::Class)); | 
|  | EXPECT_THAT(Results.Completions, has("foo", CompletionItemKind::Function)); | 
|  | EXPECT_THAT(Results.Completions, has("XXX", CompletionItemKind::Class)); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains((named("fooooo"), kind(CompletionItemKind::Function), | 
|  | doc("Doooc"), returnType("void")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Documentation) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | // Non-doxygen comment. | 
|  | __attribute__((annotate("custom_annotation"))) int foo(); | 
|  | /// Doxygen comment. | 
|  | /// \param int a | 
|  | int bar(int a); | 
|  | /* Multi-line | 
|  | block comment | 
|  | */ | 
|  | int baz(); | 
|  |  | 
|  | int x = ^ | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf( | 
|  | named("foo"), | 
|  | doc("Annotation: custom_annotation\nNon-doxygen comment.")))); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | Contains(AllOf(named("bar"), doc("Doxygen comment.\n\\param int a")))); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("baz"), doc("Multi-line block comment")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CommentsFromSystemHeaders) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  |  | 
|  | auto Opts = ClangdServer::optsForTest(); | 
|  | Opts.BuildDynamicSymbolIndex = true; | 
|  |  | 
|  | ClangdServer Server(CDB, FS, Opts); | 
|  |  | 
|  | FS.Files[testPath("foo.h")] = R"cpp( | 
|  | #pragma GCC system_header | 
|  |  | 
|  | // This comment should be retained! | 
|  | int foo(); | 
|  | )cpp"; | 
|  |  | 
|  | auto File = testPath("foo.cpp"); | 
|  | Annotations Test(R"cpp( | 
|  | #include "foo.h" | 
|  | int x = foo^ | 
|  | )cpp"); | 
|  | runAddDocument(Server, File, Test.code()); | 
|  | auto CompletionList = | 
|  | llvm::cantFail(runCodeComplete(Server, File, Test.point(), {})); | 
|  |  | 
|  | EXPECT_THAT( | 
|  | CompletionList.Completions, | 
|  | Contains(AllOf(named("foo"), doc("This comment should be retained!")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, GlobalCompletionFiltering) { | 
|  |  | 
|  | Symbol Class = cls("XYZ"); | 
|  | Class.Flags = static_cast<Symbol::SymbolFlag>( | 
|  | Class.Flags & ~(Symbol::IndexedForCodeCompletion)); | 
|  | Symbol Func = func("XYZ::foooo"); | 
|  | Func.Flags = static_cast<Symbol::SymbolFlag>( | 
|  | Func.Flags & ~(Symbol::IndexedForCodeCompletion)); | 
|  |  | 
|  | auto Results = completions(R"(//      void f() { | 
|  | XYZ::foooo^ | 
|  | })", | 
|  | {Class, Func}); | 
|  | EXPECT_THAT(Results.Completions, IsEmpty()); | 
|  | } | 
|  |  | 
|  | TEST(CodeCompleteTest, DisableTypoCorrection) { | 
|  | auto Results = completions(R"cpp( | 
|  | namespace clang { int v; } | 
|  | void f() { clangd::^ | 
|  | )cpp"); | 
|  | EXPECT_TRUE(Results.Completions.empty()); | 
|  | } | 
|  |  | 
|  | TEST(CodeCompleteTest, NoColonColonAtTheEnd) { | 
|  | auto Results = completions(R"cpp( | 
|  | namespace clang { } | 
|  | void f() { | 
|  | clan^ | 
|  | } | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Results.Completions, Contains(labeled("clang"))); | 
|  | EXPECT_THAT(Results.Completions, Not(Contains(labeled("clang::")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTests, EmptySnippetDoesNotCrash) { | 
|  | // See https://github.com/clangd/clangd/issues/1216 | 
|  | auto Results = completions(R"cpp( | 
|  | int main() { | 
|  | auto w = [&](auto &&f) { return f(f); }; | 
|  | auto f = w([&](auto &&f) { | 
|  | return [&](auto &&n) { | 
|  | if (n == 0) { | 
|  | return 1; | 
|  | } | 
|  | return n * ^(f)(n - 1); | 
|  | }; | 
|  | })(10); | 
|  | } | 
|  | )cpp"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Issue1427Crash) { | 
|  | // Need to provide main file signals to ensure that the branch in | 
|  | // SymbolRelevanceSignals::computeASTSignals() that tries to | 
|  | // compute a symbol ID is taken. | 
|  | ASTSignals MainFileSignals; | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.MainFileSignals = &MainFileSignals; | 
|  | completions(R"cpp( | 
|  | auto f = []() { | 
|  | 1.0_^ | 
|  | }; | 
|  | )cpp", | 
|  | {}, Opts); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, BacktrackCrashes) { | 
|  | // Sema calls code completion callbacks twice in these cases. | 
|  | auto Results = completions(R"cpp( | 
|  | namespace ns { | 
|  | struct FooBarBaz {}; | 
|  | } // namespace ns | 
|  |  | 
|  | int foo(ns::FooBar^ | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Results.Completions, ElementsAre(labeled("FooBarBaz"))); | 
|  |  | 
|  | // Check we don't crash in that case too. | 
|  | completions(R"cpp( | 
|  | struct FooBarBaz {}; | 
|  | void test() { | 
|  | if (FooBarBaz * x^) {} | 
|  | } | 
|  | )cpp"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompleteInMacroWithStringification) { | 
|  | auto Results = completions(R"cpp( | 
|  | void f(const char *, int x); | 
|  | #define F(x) f(#x, x) | 
|  |  | 
|  | namespace ns { | 
|  | int X; | 
|  | int Y; | 
|  | }  // namespace ns | 
|  |  | 
|  | int f(int input_num) { | 
|  | F(ns::^) | 
|  | } | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(named("X"), named("Y"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompleteInMacroAndNamespaceWithStringification) { | 
|  | auto Results = completions(R"cpp( | 
|  | void f(const char *, int x); | 
|  | #define F(x) f(#x, x) | 
|  |  | 
|  | namespace ns { | 
|  | int X; | 
|  |  | 
|  | int f(int input_num) { | 
|  | F(^) | 
|  | } | 
|  | }  // namespace ns | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Results.Completions, Contains(named("X"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, IgnoreCompleteInExcludedPPBranchWithRecoveryContext) { | 
|  | auto Results = completions(R"cpp( | 
|  | int bar(int param_in_bar) { | 
|  | } | 
|  |  | 
|  | int foo(int param_in_foo) { | 
|  | #if 0 | 
|  | // In recovery mode, "param_in_foo" will also be suggested among many other | 
|  | // unrelated symbols; however, this is really a special case where this works. | 
|  | // If the #if block is outside of the function, "param_in_foo" is still | 
|  | // suggested, but "bar" and "foo" are missing. So the recovery mode doesn't | 
|  | // really provide useful results in excluded branches. | 
|  | par^ | 
|  | #endif | 
|  | } | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_TRUE(Results.Completions.empty()); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DefaultArgs) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | std::string Context = R"cpp( | 
|  | int X(int A = 0); | 
|  | int Y(int A, int B = 0); | 
|  | int Z(int A, int B = 0, int C = 0, int D = 0); | 
|  | )cpp"; | 
|  | EXPECT_THAT(completions(Context + "int y = X^", {}, Opts).Completions, | 
|  | UnorderedElementsAre(labeled("X(int A = 0)"))); | 
|  | EXPECT_THAT(completions(Context + "int y = Y^", {}, Opts).Completions, | 
|  | UnorderedElementsAre(AllOf(labeled("Y(int A, int B = 0)"), | 
|  | snippetSuffix("(${1:int A})")))); | 
|  | EXPECT_THAT(completions(Context + "int y = Z^", {}, Opts).Completions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(labeled("Z(int A, int B = 0, int C = 0, int D = 0)"), | 
|  | snippetSuffix("(${1:int A})")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoCrashWithTemplateParamsAndPreferredTypes) { | 
|  | auto Completions = completions(R"cpp( | 
|  | template <template <class> class TT> int foo() { | 
|  | int a = ^ | 
|  | } | 
|  | )cpp") | 
|  | .Completions; | 
|  | EXPECT_THAT(Completions, Contains(named("TT"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NestedTemplateHeuristics) { | 
|  | auto Completions = completions(R"cpp( | 
|  | struct Plain { int xxx; }; | 
|  | template <typename T> class Templ { Plain ppp; }; | 
|  | template <typename T> void foo(Templ<T> &t) { | 
|  | // Formally ppp has DependentTy, because Templ may be specialized. | 
|  | // However we sholud be able to see into it using the primary template. | 
|  | t.ppp.^ | 
|  | } | 
|  | )cpp") | 
|  | .Completions; | 
|  | EXPECT_THAT(Completions, Contains(named("xxx"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, RecordCCResultCallback) { | 
|  | std::vector<CodeCompletion> RecordedCompletions; | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.RecordCCResult = [&RecordedCompletions](const CodeCompletion &CC, | 
|  | const SymbolQualitySignals &, | 
|  | const SymbolRelevanceSignals &, | 
|  | float Score) { | 
|  | RecordedCompletions.push_back(CC); | 
|  | }; | 
|  |  | 
|  | completions("int xy1, xy2; int a = xy^", /*IndexSymbols=*/{}, Opts); | 
|  | EXPECT_THAT(RecordedCompletions, | 
|  | UnorderedElementsAre(named("xy1"), named("xy2"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ASTSignals) { | 
|  | struct Completion { | 
|  | std::string Name; | 
|  | unsigned MainFileRefs; | 
|  | unsigned ScopeRefsInFile; | 
|  | }; | 
|  | CodeCompleteOptions Opts; | 
|  | std::vector<Completion> RecordedCompletions; | 
|  | Opts.RecordCCResult = [&RecordedCompletions](const CodeCompletion &CC, | 
|  | const SymbolQualitySignals &, | 
|  | const SymbolRelevanceSignals &R, | 
|  | float Score) { | 
|  | RecordedCompletions.push_back({CC.Name, R.MainFileRefs, R.ScopeRefsInFile}); | 
|  | }; | 
|  | ASTSignals MainFileSignals; | 
|  | MainFileSignals.ReferencedSymbols[var("xy1").ID] = 3; | 
|  | MainFileSignals.ReferencedSymbols[var("xy2").ID] = 1; | 
|  | MainFileSignals.ReferencedSymbols[var("xyindex").ID] = 10; | 
|  | MainFileSignals.RelatedNamespaces["tar::"] = 5; | 
|  | MainFileSignals.RelatedNamespaces["bar::"] = 3; | 
|  | Opts.MainFileSignals = &MainFileSignals; | 
|  | Opts.AllScopes = true; | 
|  | completions( | 
|  | R"cpp( | 
|  | int xy1; | 
|  | int xy2; | 
|  | namespace bar { | 
|  | int xybar = 1; | 
|  | int a = xy^ | 
|  | } | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{var("xyindex"), var("tar::xytar"), var("bar::xybar")}, | 
|  | Opts); | 
|  | EXPECT_THAT(RecordedCompletions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(named("xy1"), mainFileRefs(3u), scopeRefs(0u)), | 
|  | AllOf(named("xy2"), mainFileRefs(1u), scopeRefs(0u)), | 
|  | AllOf(named("xyindex"), mainFileRefs(10u), scopeRefs(0u)), | 
|  | AllOf(named("xytar"), mainFileRefs(0u), scopeRefs(5u)), | 
|  | AllOf(/*both from sema and index*/ named("xybar"), | 
|  | mainFileRefs(0u), scopeRefs(3u)))); | 
|  | } | 
|  |  | 
|  | SignatureHelp | 
|  | signatures(llvm::StringRef Text, Position Point, | 
|  | std::vector<Symbol> IndexSymbols = {}, | 
|  | MarkupKind DocumentationFormat = MarkupKind::PlainText) { | 
|  | std::unique_ptr<SymbolIndex> Index; | 
|  | if (!IndexSymbols.empty()) | 
|  | Index = memIndex(IndexSymbols); | 
|  |  | 
|  | auto TU = TestTU::withCode(Text); | 
|  | MockFS FS; | 
|  | auto Inputs = TU.inputs(FS); | 
|  | Inputs.Index = Index.get(); | 
|  | IgnoreDiagnostics Diags; | 
|  | auto CI = buildCompilerInvocation(Inputs, Diags); | 
|  | if (!CI) { | 
|  | ADD_FAILURE() << "Couldn't build CompilerInvocation"; | 
|  | return {}; | 
|  | } | 
|  | auto Preamble = buildPreamble(testPath(TU.Filename), *CI, Inputs, | 
|  | /*InMemory=*/true, /*Callback=*/nullptr); | 
|  | if (!Preamble) { | 
|  | ADD_FAILURE() << "Couldn't build Preamble"; | 
|  | return {}; | 
|  | } | 
|  | return signatureHelp(testPath(TU.Filename), Point, *Preamble, Inputs, | 
|  | DocumentationFormat); | 
|  | } | 
|  |  | 
|  | SignatureHelp | 
|  | signatures(llvm::StringRef Text, std::vector<Symbol> IndexSymbols = {}, | 
|  | MarkupKind DocumentationFormat = MarkupKind::PlainText) { | 
|  | Annotations Test(Text); | 
|  | return signatures(Test.code(), Test.point(), std::move(IndexSymbols), | 
|  | DocumentationFormat); | 
|  | } | 
|  |  | 
|  | struct ExpectedParameter { | 
|  | std::string Text; | 
|  | std::pair<unsigned, unsigned> Offsets; | 
|  | }; | 
|  | llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, | 
|  | const ExpectedParameter &P) { | 
|  | return OS << P.Text; | 
|  | } | 
|  | MATCHER_P(paramsAre, P, "") { | 
|  | if (P.size() != arg.parameters.size()) | 
|  | return false; | 
|  | for (unsigned I = 0; I < P.size(); ++I) { | 
|  | if (P[I].Text != arg.parameters[I].labelString || | 
|  | P[I].Offsets != arg.parameters[I].labelOffsets) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  | MATCHER_P(sigDoc, doc, "") { return arg.documentation.value == doc; } | 
|  |  | 
|  | /// \p AnnotatedLabel is a signature label with ranges marking parameters, e.g. | 
|  | ///    foo([[int p1]], [[double p2]]) -> void | 
|  | Matcher<SignatureInformation> sig(llvm::StringRef AnnotatedLabel) { | 
|  | llvm::Annotations A(AnnotatedLabel); | 
|  | std::string Label = std::string(A.code()); | 
|  | std::vector<ExpectedParameter> Parameters; | 
|  | for (auto Range : A.ranges()) { | 
|  | Parameters.emplace_back(); | 
|  |  | 
|  | ExpectedParameter &P = Parameters.back(); | 
|  | P.Text = Label.substr(Range.Begin, Range.End - Range.Begin); | 
|  | P.Offsets.first = lspLength(llvm::StringRef(Label).substr(0, Range.Begin)); | 
|  | P.Offsets.second = lspLength(llvm::StringRef(Label).substr(1, Range.End)); | 
|  | } | 
|  | return AllOf(sigHelpLabeled(Label), paramsAre(Parameters)); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, Overloads) { | 
|  | auto Results = signatures(R"cpp( | 
|  | void foo(int x, int y); | 
|  | void foo(int x, float y); | 
|  | void foo(float x, int y); | 
|  | void foo(float x, float y); | 
|  | void bar(int x, int y = 0); | 
|  | int main() { foo(^); } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | UnorderedElementsAre(sig("foo([[float x]], [[float y]]) -> void"), | 
|  | sig("foo([[float x]], [[int y]]) -> void"), | 
|  | sig("foo([[int x]], [[float y]]) -> void"), | 
|  | sig("foo([[int x]], [[int y]]) -> void"))); | 
|  | // We always prefer the first signature. | 
|  | EXPECT_EQ(0, Results.activeSignature); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, FunctionPointers) { | 
|  | llvm::StringLiteral Tests[] = { | 
|  | // Variable of function pointer type | 
|  | R"cpp( | 
|  | void (*foo)(int x, int y); | 
|  | int main() { foo(^); } | 
|  | )cpp", | 
|  | // Wrapped in an AttributedType | 
|  | R"cpp( | 
|  | void (__stdcall *foo)(int x, int y); | 
|  | int main() { foo(^); } | 
|  | )cpp", | 
|  | // Another syntax for an AttributedType | 
|  | R"cpp( | 
|  | void (__attribute__(stdcall) *foo)(int x, int y); | 
|  | int main() { foo(^); }, | 
|  | )cpp", | 
|  | // Wrapped in a typedef | 
|  | R"cpp( | 
|  | typedef void (*fn)(int x, int y); | 
|  | fn foo; | 
|  | int main() { foo(^); } | 
|  | )cpp", | 
|  | // Wrapped in both a typedef and an AttributedTyped | 
|  | R"cpp( | 
|  | typedef void (__stdcall *fn)(int x, int y); | 
|  | fn foo; | 
|  | int main() { foo(^); } | 
|  | )cpp", | 
|  | // Field of function pointer type | 
|  | R"cpp( | 
|  | struct S { | 
|  | void (*foo)(int x, int y); | 
|  | }; | 
|  | S s; | 
|  | int main() { s.foo(^); } | 
|  | )cpp", | 
|  | // Field of function pointer typedef type | 
|  | R"cpp( | 
|  | typedef void (*fn)(int x, int y); | 
|  | struct S { | 
|  | fn foo; | 
|  | }; | 
|  | S s; | 
|  | int main() { s.foo(^); } | 
|  | )cpp"}; | 
|  | for (auto Test : Tests) | 
|  | EXPECT_THAT(signatures(Test).signatures, | 
|  | UnorderedElementsAre(sig("([[int x]], [[int y]]) -> void"))); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, Constructors) { | 
|  | std::string Top = R"cpp( | 
|  | struct S { | 
|  | S(int); | 
|  | S(const S &) = delete; | 
|  | }; | 
|  | )cpp"; | 
|  |  | 
|  | auto CheckParenInit = [&](std::string Init) { | 
|  | EXPECT_THAT(signatures(Top + Init).signatures, | 
|  | UnorderedElementsAre(sig("S([[int]])"))) | 
|  | << Init; | 
|  | }; | 
|  | CheckParenInit("S s(^);"); | 
|  | CheckParenInit("auto s = S(^);"); | 
|  | CheckParenInit("auto s = new S(^);"); | 
|  |  | 
|  | auto CheckBracedInit = [&](std::string Init) { | 
|  | EXPECT_THAT(signatures(Top + Init).signatures, | 
|  | UnorderedElementsAre(sig("S{[[int]]}"))) | 
|  | << Init; | 
|  | }; | 
|  | CheckBracedInit("S s{^};"); | 
|  | CheckBracedInit("S s = {^};"); | 
|  | CheckBracedInit("auto s = S{^};"); | 
|  | // FIXME: doesn't work: no ExpectedType set in ParseCXXNewExpression. | 
|  | // CheckBracedInit("auto s = new S{^};"); | 
|  | CheckBracedInit("int x(S); int i = x({^});"); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, Aggregates) { | 
|  | std::string Top = R"cpp( | 
|  | struct S { | 
|  | int a, b, c, d; | 
|  | }; | 
|  | )cpp"; | 
|  | auto AggregateSig = sig("S{[[int a]], [[int b]], [[int c]], [[int d]]}"); | 
|  | EXPECT_THAT(signatures(Top + "S s{^}").signatures, | 
|  | UnorderedElementsAre(AggregateSig, sig("S{}"), | 
|  | sig("S{[[const S &]]}"), | 
|  | sig("S{[[S &&]]}"))); | 
|  | EXPECT_THAT(signatures(Top + "S s{1,^}").signatures, | 
|  | ElementsAre(AggregateSig)); | 
|  | EXPECT_EQ(signatures(Top + "S s{1,^}").activeParameter, 1); | 
|  | EXPECT_THAT(signatures(Top + "S s{.c=3,^}").signatures, | 
|  | ElementsAre(AggregateSig)); | 
|  | EXPECT_EQ(signatures(Top + "S s{.c=3,^}").activeParameter, 3); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, OverloadInitListRegression) { | 
|  | auto Results = signatures(R"cpp( | 
|  | struct A {int x;}; | 
|  | struct B {B(A);}; | 
|  | void f(); | 
|  | int main() { | 
|  | B b({1}); | 
|  | f(^); | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, UnorderedElementsAre(sig("f() -> void"))); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, DefaultArgs) { | 
|  | auto Results = signatures(R"cpp( | 
|  | void bar(int x, int y = 0); | 
|  | void bar(float x = 0, int y = 42); | 
|  | int main() { bar(^ | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | UnorderedElementsAre( | 
|  | sig("bar([[int x]], [[int y = 0]]) -> void"), | 
|  | sig("bar([[float x = 0]], [[int y = 42]]) -> void"))); | 
|  | EXPECT_EQ(0, Results.activeSignature); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, ActiveArg) { | 
|  | auto Results = signatures(R"cpp( | 
|  | int baz(int a, int b, int c); | 
|  | int main() { baz(baz(1,2,3), ^); } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | ElementsAre(sig("baz([[int a]], [[int b]], [[int c]]) -> int"))); | 
|  | EXPECT_EQ(0, Results.activeSignature); | 
|  | EXPECT_EQ(1, Results.activeParameter); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, OpeningParen) { | 
|  | llvm::StringLiteral Tests[] = { | 
|  | // Recursive function call. | 
|  | R"cpp( | 
|  | int foo(int a, int b, int c); | 
|  | int main() { | 
|  | foo(foo $p^( foo(10, 10, 10), ^ ))); | 
|  | })cpp", | 
|  | // Functional type cast. | 
|  | R"cpp( | 
|  | struct Foo { | 
|  | Foo(int a, int b, int c); | 
|  | }; | 
|  | int main() { | 
|  | Foo $p^( 10, ^ ); | 
|  | })cpp", | 
|  | // New expression. | 
|  | R"cpp( | 
|  | struct Foo { | 
|  | Foo(int a, int b, int c); | 
|  | }; | 
|  | int main() { | 
|  | new Foo $p^( 10, ^ ); | 
|  | })cpp", | 
|  | // Macro expansion. | 
|  | R"cpp( | 
|  | int foo(int a, int b, int c); | 
|  | #define FOO foo( | 
|  |  | 
|  | int main() { | 
|  | // Macro expansions. | 
|  | $p^FOO 10, ^ ); | 
|  | })cpp", | 
|  | // Macro arguments. | 
|  | R"cpp( | 
|  | int foo(int a, int b, int c); | 
|  | int main() { | 
|  | #define ID(X) X | 
|  | // FIXME: figure out why ID(foo (foo(10), )) doesn't work when preserving | 
|  | // the recovery expression. | 
|  | ID(foo $p^( 10, ^ )) | 
|  | })cpp", | 
|  | // Dependent args. | 
|  | R"cpp( | 
|  | int foo(int a, int b); | 
|  | template <typename T> void bar(T t) { | 
|  | foo$p^(t, ^t); | 
|  | })cpp", | 
|  | // Dependent args on templated func. | 
|  | R"cpp( | 
|  | template <typename T> | 
|  | int foo(T, T); | 
|  | template <typename T> void bar(T t) { | 
|  | foo$p^(t, ^t); | 
|  | })cpp", | 
|  | // Dependent args on member. | 
|  | R"cpp( | 
|  | struct Foo { int foo(int, int); }; | 
|  | template <typename T> void bar(T t) { | 
|  | Foo f; | 
|  | f.foo$p^(t, ^t); | 
|  | })cpp", | 
|  | // Dependent args on templated member. | 
|  | R"cpp( | 
|  | struct Foo { template <typename T> int foo(T, T); }; | 
|  | template <typename T> void bar(T t) { | 
|  | Foo f; | 
|  | f.foo$p^(t, ^t); | 
|  | })cpp", | 
|  | }; | 
|  |  | 
|  | for (auto Test : Tests) { | 
|  | Annotations Code(Test); | 
|  | EXPECT_EQ(signatures(Code.code(), Code.point()).argListStart, | 
|  | Code.point("p")) | 
|  | << "Test source:" << Test; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, StalePreamble) { | 
|  | TestTU TU; | 
|  | TU.Code = ""; | 
|  | IgnoreDiagnostics Diags; | 
|  | MockFS FS; | 
|  | auto Inputs = TU.inputs(FS); | 
|  | auto CI = buildCompilerInvocation(Inputs, Diags); | 
|  | ASSERT_TRUE(CI); | 
|  | auto EmptyPreamble = buildPreamble(testPath(TU.Filename), *CI, Inputs, | 
|  | /*InMemory=*/true, /*Callback=*/nullptr); | 
|  | ASSERT_TRUE(EmptyPreamble); | 
|  |  | 
|  | TU.AdditionalFiles["a.h"] = "int foo(int x);"; | 
|  | const Annotations Test(R"cpp( | 
|  | #include "a.h" | 
|  | void bar() { foo(^2); })cpp"); | 
|  | TU.Code = Test.code().str(); | 
|  | auto Results = | 
|  | signatureHelp(testPath(TU.Filename), Test.point(), *EmptyPreamble, | 
|  | TU.inputs(FS), MarkupKind::PlainText); | 
|  | EXPECT_THAT(Results.signatures, ElementsAre(sig("foo([[int x]]) -> int"))); | 
|  | EXPECT_EQ(0, Results.activeSignature); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  |  | 
|  | class IndexRequestCollector : public SymbolIndex { | 
|  | public: | 
|  | IndexRequestCollector(std::vector<Symbol> Syms = {}) : Symbols(Syms) {} | 
|  |  | 
|  | bool | 
|  | fuzzyFind(const FuzzyFindRequest &Req, | 
|  | llvm::function_ref<void(const Symbol &)> Callback) const override { | 
|  | std::unique_lock<std::mutex> Lock(Mut); | 
|  | Requests.push_back(Req); | 
|  | ReceivedRequestCV.notify_one(); | 
|  | for (const auto &Sym : Symbols) | 
|  | Callback(Sym); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void lookup(const LookupRequest &, | 
|  | llvm::function_ref<void(const Symbol &)>) const override {} | 
|  |  | 
|  | bool refs(const RefsRequest &, | 
|  | llvm::function_ref<void(const Ref &)>) const override { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void relations(const RelationsRequest &, | 
|  | llvm::function_ref<void(const SymbolID &, const Symbol &)>) | 
|  | const override {} | 
|  |  | 
|  | llvm::unique_function<IndexContents(llvm::StringRef) const> | 
|  | indexedFiles() const override { | 
|  | return [](llvm::StringRef) { return IndexContents::None; }; | 
|  | } | 
|  |  | 
|  | // This is incorrect, but IndexRequestCollector is not an actual index and it | 
|  | // isn't used in production code. | 
|  | size_t estimateMemoryUsage() const override { return 0; } | 
|  |  | 
|  | const std::vector<FuzzyFindRequest> consumeRequests(size_t Num) const { | 
|  | std::unique_lock<std::mutex> Lock(Mut); | 
|  | EXPECT_TRUE(wait(Lock, ReceivedRequestCV, timeoutSeconds(30), | 
|  | [this, Num] { return Requests.size() == Num; })); | 
|  | auto Reqs = std::move(Requests); | 
|  | Requests = {}; | 
|  | return Reqs; | 
|  | } | 
|  |  | 
|  | private: | 
|  | std::vector<Symbol> Symbols; | 
|  | // We need a mutex to handle async fuzzy find requests. | 
|  | mutable std::condition_variable ReceivedRequestCV; | 
|  | mutable std::mutex Mut; | 
|  | mutable std::vector<FuzzyFindRequest> Requests; | 
|  | }; | 
|  |  | 
|  | // Clients have to consume exactly Num requests. | 
|  | std::vector<FuzzyFindRequest> captureIndexRequests(llvm::StringRef Code, | 
|  | size_t Num = 1) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | IndexRequestCollector Requests; | 
|  | Opts.Index = &Requests; | 
|  | completions(Code, {}, Opts); | 
|  | const auto Reqs = Requests.consumeRequests(Num); | 
|  | EXPECT_EQ(Reqs.size(), Num); | 
|  | return Reqs; | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, UnqualifiedIdQuery) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace std {} | 
|  | using namespace std; | 
|  | namespace ns { | 
|  | void f() { | 
|  | vec^ | 
|  | } | 
|  | } | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, | 
|  | ElementsAre(Field(&FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("", "ns::", "std::")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, EnclosingScopeComesFirst) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace std {} | 
|  | using namespace std; | 
|  | namespace nx { | 
|  | namespace ns { | 
|  | namespace { | 
|  | void f() { | 
|  | vec^ | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, | 
|  | ElementsAre(Field( | 
|  | &FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("", "std::", "nx::ns::", "nx::")))); | 
|  | EXPECT_EQ(Requests[0].Scopes[0], "nx::ns::"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ResolvedQualifiedIdQuery) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace ns1 {} | 
|  | namespace ns2 {} // ignore | 
|  | namespace ns3 { namespace nns3 {} } | 
|  | namespace foo { | 
|  | using namespace ns1; | 
|  | using namespace ns3::nns3; | 
|  | } | 
|  | namespace ns { | 
|  | void f() { | 
|  | foo::^ | 
|  | } | 
|  | } | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, | 
|  | ElementsAre(Field( | 
|  | &FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("foo::", "ns1::", "ns3::nns3::")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, UnresolvedQualifierIdQuery) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace a {} | 
|  | using namespace a; | 
|  | namespace ns { | 
|  | void f() { | 
|  | bar::^ | 
|  | } | 
|  | } // namespace ns | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, | 
|  | ElementsAre(Field( | 
|  | &FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("a::bar::", "ns::bar::", "bar::")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, UnresolvedNestedQualifierIdQuery) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace a {} | 
|  | using namespace a; | 
|  | namespace ns { | 
|  | void f() { | 
|  | ::a::bar::^ | 
|  | } | 
|  | } // namespace ns | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, ElementsAre(Field(&FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("a::bar::")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, EmptyQualifiedQuery) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace ns { | 
|  | void f() { | 
|  | ^ | 
|  | } | 
|  | } // namespace ns | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, ElementsAre(Field(&FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("", "ns::")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, GlobalQualifiedQuery) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace ns { | 
|  | void f() { | 
|  | ::^ | 
|  | } | 
|  | } // namespace ns | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, ElementsAre(Field(&FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoDuplicatedQueryScopes) { | 
|  | auto Requests = captureIndexRequests(R"cpp( | 
|  | namespace {} | 
|  |  | 
|  | namespace na { | 
|  | namespace {} | 
|  | namespace nb { | 
|  | ^ | 
|  | } // namespace nb | 
|  | } // namespace na | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Requests, | 
|  | ElementsAre(Field(&FuzzyFindRequest::Scopes, | 
|  | UnorderedElementsAre("na::", "na::nb::", "")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoIndexCompletionsInsideClasses) { | 
|  | auto Completions = completions( | 
|  | R"cpp( | 
|  | struct Foo { | 
|  | int SomeNameOfField; | 
|  | typedef int SomeNameOfTypedefField; | 
|  | }; | 
|  |  | 
|  | Foo::^)cpp", | 
|  | {func("::SomeNameInTheIndex"), func("::Foo::SomeNameInTheIndex")}); | 
|  |  | 
|  | EXPECT_THAT(Completions.Completions, | 
|  | AllOf(Contains(labeled("SomeNameOfField")), | 
|  | Contains(labeled("SomeNameOfTypedefField")), | 
|  | Not(Contains(labeled("SomeNameInTheIndex"))))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoIndexCompletionsInsideDependentCode) { | 
|  | { | 
|  | auto Completions = completions( | 
|  | R"cpp( | 
|  | template <class T> | 
|  | void foo() { | 
|  | T::^ | 
|  | } | 
|  | )cpp", | 
|  | {func("::SomeNameInTheIndex")}); | 
|  |  | 
|  | EXPECT_THAT(Completions.Completions, | 
|  | Not(Contains(labeled("SomeNameInTheIndex")))); | 
|  | } | 
|  |  | 
|  | { | 
|  | auto Completions = completions( | 
|  | R"cpp( | 
|  | template <class T> | 
|  | void foo() { | 
|  | T::template Y<int>::^ | 
|  | } | 
|  | )cpp", | 
|  | {func("::SomeNameInTheIndex")}); | 
|  |  | 
|  | EXPECT_THAT(Completions.Completions, | 
|  | Not(Contains(labeled("SomeNameInTheIndex")))); | 
|  | } | 
|  |  | 
|  | { | 
|  | auto Completions = completions( | 
|  | R"cpp( | 
|  | template <class T> | 
|  | void foo() { | 
|  | T::foo::^ | 
|  | } | 
|  | )cpp", | 
|  | {func("::SomeNameInTheIndex")}); | 
|  |  | 
|  | EXPECT_THAT(Completions.Completions, | 
|  | Not(Contains(labeled("SomeNameInTheIndex")))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, OverloadBundling) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.BundleOverloads = true; | 
|  |  | 
|  | std::string Context = R"cpp( | 
|  | struct X { | 
|  | // Overload with int | 
|  | int a(int) __attribute__((deprecated("", ""))); | 
|  | // Overload with bool | 
|  | int a(bool); | 
|  | int b(float); | 
|  |  | 
|  | X(int); | 
|  | X(float); | 
|  | }; | 
|  | int GFuncC(int); | 
|  | int GFuncD(int); | 
|  | )cpp"; | 
|  |  | 
|  | // Member completions are bundled. | 
|  | EXPECT_THAT(completions(Context + "int y = X().^", {}, Opts).Completions, | 
|  | UnorderedElementsAre(labeled("a(…)"), labeled("b(float)"))); | 
|  |  | 
|  | // Constructor completions are bundled. | 
|  | EXPECT_THAT(completions(Context + "X z = X^", {}, Opts).Completions, | 
|  | UnorderedElementsAre(labeled("X"), labeled("X(…)"))); | 
|  |  | 
|  | // Non-member completions are bundled, including index+sema. | 
|  | Symbol NoArgsGFunc = func("GFuncC"); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "int y = GFunc^", {NoArgsGFunc}, Opts).Completions, | 
|  | UnorderedElementsAre(labeled("GFuncC(…)"), labeled("GFuncD(int)"))); | 
|  |  | 
|  | // Differences in header-to-insert suppress bundling. | 
|  | std::string DeclFile = URI::create(testPath("foo")).toString(); | 
|  | NoArgsGFunc.CanonicalDeclaration.FileURI = DeclFile.c_str(); | 
|  | NoArgsGFunc.IncludeHeaders.emplace_back("<foo>", 1, Symbol::Include); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "int y = GFunc^", {NoArgsGFunc}, Opts).Completions, | 
|  | UnorderedElementsAre(AllOf(named("GFuncC"), insertInclude("<foo>")), | 
|  | labeled("GFuncC(int)"), labeled("GFuncD(int)"))); | 
|  |  | 
|  | // Examine a bundled completion in detail. | 
|  | auto A = | 
|  | completions(Context + "int y = X().a^", {}, Opts).Completions.front(); | 
|  | EXPECT_EQ(A.Name, "a"); | 
|  | EXPECT_EQ(A.Signature, "(…)"); | 
|  | EXPECT_EQ(A.BundleSize, 2u); | 
|  | EXPECT_EQ(A.Kind, CompletionItemKind::Method); | 
|  | EXPECT_EQ(A.ReturnType, "int"); // All overloads return int. | 
|  | // For now we just return one of the doc strings arbitrarily. | 
|  | ASSERT_TRUE(A.Documentation); | 
|  | ASSERT_FALSE(A.Deprecated); // Not all overloads deprecated. | 
|  | EXPECT_THAT( | 
|  | A.Documentation->asPlainText(), | 
|  | AnyOf(HasSubstr("Overload with int"), HasSubstr("Overload with bool"))); | 
|  | EXPECT_EQ(A.SnippetSuffix, "($0)"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, OverloadBundlingSameFileDifferentURI) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.BundleOverloads = true; | 
|  |  | 
|  | Symbol SymX = sym("ns::X", index::SymbolKind::Function, "@F@\\0#"); | 
|  | Symbol SymY = sym("ns::X", index::SymbolKind::Function, "@F@\\0#I#"); | 
|  | std::string BarHeader = testPath("bar.h"); | 
|  | auto BarURI = URI::create(BarHeader).toString(); | 
|  | SymX.CanonicalDeclaration.FileURI = BarURI.c_str(); | 
|  | SymY.CanonicalDeclaration.FileURI = BarURI.c_str(); | 
|  | // The include header is different, but really it's the same file. | 
|  | SymX.IncludeHeaders.emplace_back("\"bar.h\"", 1, Symbol::Include); | 
|  | SymY.IncludeHeaders.emplace_back(BarURI.c_str(), 1, Symbol::Include); | 
|  |  | 
|  | auto Results = completions("void f() { ::ns::^ }", {SymX, SymY}, Opts); | 
|  | // Expect both results are bundled, despite the different-but-same | 
|  | // IncludeHeader. | 
|  | ASSERT_EQ(1u, Results.Completions.size()); | 
|  | const auto &R = Results.Completions.front(); | 
|  | EXPECT_EQ("X", R.Name); | 
|  | EXPECT_EQ(2u, R.BundleSize); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DocumentationFromChangedFileCrash) { | 
|  | MockFS FS; | 
|  | auto FooH = testPath("foo.h"); | 
|  | auto FooCpp = testPath("foo.cpp"); | 
|  | FS.Files[FooH] = R"cpp( | 
|  | // this is my documentation comment. | 
|  | int func(); | 
|  | )cpp"; | 
|  | FS.Files[FooCpp] = ""; | 
|  |  | 
|  | MockCompilationDatabase CDB; | 
|  | ClangdServer Server(CDB, FS, ClangdServer::optsForTest()); | 
|  |  | 
|  | Annotations Source(R"cpp( | 
|  | #include "foo.h" | 
|  | int func() { | 
|  | // This makes sure we have func from header in the AST. | 
|  | } | 
|  | int a = fun^ | 
|  | )cpp"); | 
|  | Server.addDocument(FooCpp, Source.code(), "null", WantDiagnostics::Yes); | 
|  | // We need to wait for preamble to build. | 
|  | ASSERT_TRUE(Server.blockUntilIdleForTest()); | 
|  |  | 
|  | // Change the header file. Completion will reuse the old preamble! | 
|  | FS.Files[FooH] = R"cpp( | 
|  | int func(); | 
|  | )cpp"; | 
|  |  | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | CodeCompleteResult Completions = | 
|  | cantFail(runCodeComplete(Server, FooCpp, Source.point(), Opts)); | 
|  | // We shouldn't crash. Unfortunately, current workaround is to not produce | 
|  | // comments for symbols from headers. | 
|  | EXPECT_THAT(Completions.Completions, | 
|  | Contains(AllOf(Not(isDocumented()), named("func")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NonDocComments) { | 
|  | const char *Text = R"cpp( | 
|  | // We ignore namespace comments, for rationale see CodeCompletionStrings.h. | 
|  | namespace comments_ns { | 
|  | } | 
|  |  | 
|  | // ------------------ | 
|  | int comments_foo(); | 
|  |  | 
|  | // A comment and a decl are separated by newlines. | 
|  | // Therefore, the comment shouldn't show up as doc comment. | 
|  |  | 
|  | int comments_bar(); | 
|  |  | 
|  | // this comment should be in the results. | 
|  | int comments_baz(); | 
|  |  | 
|  |  | 
|  | template <class T> | 
|  | struct Struct { | 
|  | int comments_qux(); | 
|  | int comments_quux(); | 
|  | }; | 
|  |  | 
|  |  | 
|  | // This comment should not be there. | 
|  |  | 
|  | template <class T> | 
|  | int Struct<T>::comments_qux() { | 
|  | } | 
|  |  | 
|  | // This comment **should** be in results. | 
|  | template <class T> | 
|  | int Struct<T>::comments_quux() { | 
|  | int a = comments^; | 
|  | } | 
|  | )cpp"; | 
|  |  | 
|  | // We should not get any of those comments in completion. | 
|  | EXPECT_THAT( | 
|  | completions(Text).Completions, | 
|  | UnorderedElementsAre(AllOf(Not(isDocumented()), named("comments_foo")), | 
|  | AllOf(isDocumented(), named("comments_baz")), | 
|  | AllOf(isDocumented(), named("comments_quux")), | 
|  | AllOf(Not(isDocumented()), named("comments_ns")), | 
|  | // FIXME(ibiryukov): the following items should have | 
|  | // empty documentation, since they are separated from | 
|  | // a comment with an empty line. Unfortunately, I | 
|  | // couldn't make Sema tests pass if we ignore those. | 
|  | AllOf(isDocumented(), named("comments_bar")), | 
|  | AllOf(isDocumented(), named("comments_qux")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompleteOnInvalidLine) { | 
|  | auto FooCpp = testPath("foo.cpp"); | 
|  |  | 
|  | MockCompilationDatabase CDB; | 
|  | MockFS FS; | 
|  | FS.Files[FooCpp] = "// empty file"; | 
|  |  | 
|  | ClangdServer Server(CDB, FS, ClangdServer::optsForTest()); | 
|  | // Run completion outside the file range. | 
|  | Position Pos; | 
|  | Pos.line = 100; | 
|  | Pos.character = 0; | 
|  | EXPECT_THAT_EXPECTED( | 
|  | runCodeComplete(Server, FooCpp, Pos, clangd::CodeCompleteOptions()), | 
|  | Failed()); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, QualifiedNames) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { int local; void both(); } | 
|  | void f() { ::ns::^ } | 
|  | )cpp", | 
|  | {func("ns::both"), cls("ns::Index")}); | 
|  | // We get results from both index and sema, with no duplicates. | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre(scope("ns::"), scope("ns::"), scope("ns::"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Render) { | 
|  | CodeCompletion C; | 
|  | C.Name = "x"; | 
|  | C.FilterText = "x"; | 
|  | C.Signature = "(bool) const"; | 
|  | C.SnippetSuffix = "(${0:bool})"; | 
|  | C.ReturnType = "int"; | 
|  | C.RequiredQualifier = "Foo::"; | 
|  | C.Scope = "ns::Foo::"; | 
|  | C.Documentation.emplace(); | 
|  | C.Documentation->addParagraph().appendText("This is ").appendCode("x()"); | 
|  | C.Includes.emplace_back(); | 
|  | auto &Include = C.Includes.back(); | 
|  | Include.Header = "\"foo.h\""; | 
|  | C.Kind = CompletionItemKind::Method; | 
|  | C.Score.Total = 1.0; | 
|  | C.Score.ExcludingName = .5; | 
|  | C.Origin = SymbolOrigin::AST | SymbolOrigin::Static; | 
|  |  | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.IncludeIndicator.Insert = "^"; | 
|  | Opts.IncludeIndicator.NoInsert = ""; | 
|  | Opts.EnableSnippets = false; | 
|  |  | 
|  | auto R = C.render(Opts); | 
|  | EXPECT_EQ(R.label, "Foo::x"); | 
|  | EXPECT_EQ(R.labelDetails->detail, "(bool) const"); | 
|  | EXPECT_EQ(R.insertText, "Foo::x"); | 
|  | EXPECT_EQ(R.insertTextFormat, InsertTextFormat::PlainText); | 
|  | EXPECT_EQ(R.filterText, "x"); | 
|  | EXPECT_EQ(R.detail, "int"); | 
|  | EXPECT_EQ(R.documentation->value, "From \"foo.h\"\nThis is x()"); | 
|  | EXPECT_THAT(R.additionalTextEdits, IsEmpty()); | 
|  | EXPECT_EQ(R.sortText, sortText(1.0, "x")); | 
|  | EXPECT_FALSE(R.deprecated); | 
|  | EXPECT_EQ(R.score, .5f); | 
|  |  | 
|  | C.FilterText = "xtra"; | 
|  | R = C.render(Opts); | 
|  | EXPECT_EQ(R.filterText, "xtra"); | 
|  | EXPECT_EQ(R.sortText, sortText(1.0, "xtra")); | 
|  |  | 
|  | Opts.EnableSnippets = true; | 
|  | R = C.render(Opts); | 
|  | EXPECT_EQ(R.insertText, "Foo::x(${0:bool})"); | 
|  | EXPECT_EQ(R.insertTextFormat, InsertTextFormat::Snippet); | 
|  |  | 
|  | C.SnippetSuffix = ""; | 
|  | R = C.render(Opts); | 
|  | EXPECT_EQ(R.insertText, "Foo::x"); | 
|  | EXPECT_EQ(R.insertTextFormat, InsertTextFormat::PlainText); | 
|  |  | 
|  | Include.Insertion.emplace(); | 
|  | R = C.render(Opts); | 
|  | EXPECT_EQ(R.label, "^Foo::x"); | 
|  | EXPECT_EQ(R.labelDetails->detail, "(bool) const"); | 
|  | EXPECT_THAT(R.additionalTextEdits, Not(IsEmpty())); | 
|  |  | 
|  | Opts.ShowOrigins = true; | 
|  | R = C.render(Opts); | 
|  | EXPECT_EQ(R.label, "^[AS]Foo::x"); | 
|  | EXPECT_EQ(R.labelDetails->detail, "(bool) const"); | 
|  |  | 
|  | C.BundleSize = 2; | 
|  | R = C.render(Opts); | 
|  | EXPECT_EQ(R.detail, "[2 overloads]"); | 
|  | EXPECT_EQ(R.documentation->value, "From \"foo.h\"\nThis is x()"); | 
|  |  | 
|  | C.Deprecated = true; | 
|  | R = C.render(Opts); | 
|  | EXPECT_TRUE(R.deprecated); | 
|  |  | 
|  | Opts.DocumentationFormat = MarkupKind::Markdown; | 
|  | R = C.render(Opts); | 
|  | EXPECT_EQ(R.documentation->value, "From `\"foo.h\"`  \nThis is `x()`"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, IgnoreRecoveryResults) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { int NotRecovered() { return 0; } } | 
|  | void f() { | 
|  | // Sema enters recovery mode first and then normal mode. | 
|  | if (auto x = ns::NotRecover^) | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, UnorderedElementsAre(named("NotRecovered"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ScopeOfClassFieldInConstructorInitializer) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { | 
|  | class X { public: X(); int x_; }; | 
|  | X::X() : x_^(0) {} | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(scope("ns::X::"), named("x_")))); | 
|  | } | 
|  |  | 
|  | // Like other class members, constructor init lists have to parse what's below, | 
|  | // after the completion point. | 
|  | // But recovering from an incomplete constructor init list is particularly | 
|  | // tricky because the bulk of the list is not surrounded by brackets. | 
|  | TEST(CompletionTest, ConstructorInitListIncomplete) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { | 
|  | struct X { | 
|  | X() : x^ | 
|  | int xyz_; | 
|  | }; | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, ElementsAre(named("xyz_"))); | 
|  |  | 
|  | Results = completions( | 
|  | R"cpp( | 
|  | int foo(); | 
|  |  | 
|  | namespace ns { | 
|  | struct X { | 
|  | X() : xyz_(fo^ | 
|  | int xyz_; | 
|  | }; | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, ElementsAre(named("foo"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CodeCompletionContext) { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace ns { | 
|  | class X { public: X(); int x_; }; | 
|  | void f() { | 
|  | X x; | 
|  | x.^; | 
|  | } | 
|  | } | 
|  | )cpp"); | 
|  |  | 
|  | EXPECT_THAT(Results.Context, CodeCompletionContext::CCC_DotMemberAccess); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, FixItForArrowToDot) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  |  | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.IncludeFixIts = true; | 
|  | const char *Code = | 
|  | R"cpp( | 
|  | class Auxilary { | 
|  | public: | 
|  | void AuxFunction(); | 
|  | }; | 
|  | class ClassWithPtr { | 
|  | public: | 
|  | void MemberFunction(); | 
|  | Auxilary* operator->() const; | 
|  | Auxilary* Aux; | 
|  | }; | 
|  | void f() { | 
|  | ClassWithPtr x; | 
|  | x[[->]]^; | 
|  | } | 
|  | )cpp"; | 
|  | auto Results = completions(Code, {}, Opts); | 
|  | EXPECT_EQ(Results.Completions.size(), 3u); | 
|  |  | 
|  | TextEdit ReplacementEdit; | 
|  | ReplacementEdit.range = Annotations(Code).range(); | 
|  | ReplacementEdit.newText = "."; | 
|  | for (const auto &C : Results.Completions) { | 
|  | EXPECT_TRUE(C.FixIts.size() == 1u || C.Name == "AuxFunction"); | 
|  | if (!C.FixIts.empty()) { | 
|  | EXPECT_THAT(C.FixIts, ElementsAre(ReplacementEdit)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, FixItForDotToArrow) { | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.IncludeFixIts = true; | 
|  | const char *Code = | 
|  | R"cpp( | 
|  | class Auxilary { | 
|  | public: | 
|  | void AuxFunction(); | 
|  | }; | 
|  | class ClassWithPtr { | 
|  | public: | 
|  | void MemberFunction(); | 
|  | Auxilary* operator->() const; | 
|  | Auxilary* Aux; | 
|  | }; | 
|  | void f() { | 
|  | ClassWithPtr x; | 
|  | x[[.]]^; | 
|  | } | 
|  | )cpp"; | 
|  | auto Results = completions(Code, {}, Opts); | 
|  | EXPECT_EQ(Results.Completions.size(), 3u); | 
|  |  | 
|  | TextEdit ReplacementEdit; | 
|  | ReplacementEdit.range = Annotations(Code).range(); | 
|  | ReplacementEdit.newText = "->"; | 
|  | for (const auto &C : Results.Completions) { | 
|  | EXPECT_TRUE(C.FixIts.empty() || C.Name == "AuxFunction"); | 
|  | if (!C.FixIts.empty()) { | 
|  | EXPECT_THAT(C.FixIts, ElementsAre(ReplacementEdit)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, RenderWithFixItMerged) { | 
|  | TextEdit FixIt; | 
|  | FixIt.range.end.character = 5; | 
|  | FixIt.newText = "->"; | 
|  |  | 
|  | CodeCompletion C; | 
|  | C.Name = "x"; | 
|  | C.RequiredQualifier = "Foo::"; | 
|  | C.FixIts = {FixIt}; | 
|  | C.CompletionTokenRange.start.character = 5; | 
|  |  | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.IncludeFixIts = true; | 
|  |  | 
|  | auto R = C.render(Opts); | 
|  | EXPECT_TRUE(R.textEdit); | 
|  | EXPECT_EQ(R.textEdit->newText, "->Foo::x"); | 
|  | EXPECT_TRUE(R.additionalTextEdits.empty()); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, RenderWithFixItNonMerged) { | 
|  | TextEdit FixIt; | 
|  | FixIt.range.end.character = 4; | 
|  | FixIt.newText = "->"; | 
|  |  | 
|  | CodeCompletion C; | 
|  | C.Name = "x"; | 
|  | C.RequiredQualifier = "Foo::"; | 
|  | C.FixIts = {FixIt}; | 
|  | C.CompletionTokenRange.start.character = 5; | 
|  |  | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.IncludeFixIts = true; | 
|  |  | 
|  | auto R = C.render(Opts); | 
|  | EXPECT_TRUE(R.textEdit); | 
|  | EXPECT_EQ(R.textEdit->newText, "Foo::x"); | 
|  | EXPECT_THAT(R.additionalTextEdits, UnorderedElementsAre(FixIt)); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompletionTokenRange) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  | TestTU TU; | 
|  | TU.AdditionalFiles["foo/abc/foo.h"] = ""; | 
|  |  | 
|  | constexpr const char *TestCodes[] = { | 
|  | R"cpp( | 
|  | class Auxilary { | 
|  | public: | 
|  | void AuxFunction(); | 
|  | }; | 
|  | void f() { | 
|  | Auxilary x; | 
|  | x.[[Aux]]^; | 
|  | } | 
|  | )cpp", | 
|  | R"cpp( | 
|  | class Auxilary { | 
|  | public: | 
|  | void AuxFunction(); | 
|  | }; | 
|  | void f() { | 
|  | Auxilary x; | 
|  | x.[[]]^; | 
|  | } | 
|  | )cpp", | 
|  | R"cpp( | 
|  | #include "foo/[[a^/]]foo.h" | 
|  | )cpp", | 
|  | R"cpp( | 
|  | #include "foo/abc/[[fo^o.h"]] | 
|  | )cpp", | 
|  | }; | 
|  | for (const auto &Text : TestCodes) { | 
|  | Annotations TestCode(Text); | 
|  | TU.Code = TestCode.code().str(); | 
|  | auto Results = completions(TU, TestCode.point()); | 
|  | if (Results.Completions.size() != 1) { | 
|  | ADD_FAILURE() << "Results.Completions.size() != 1" << Text; | 
|  | continue; | 
|  | } | 
|  | EXPECT_THAT(Results.Completions.front().CompletionTokenRange, | 
|  | TestCode.range()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, OverloadsOrdering) { | 
|  | const auto Results = signatures(R"cpp( | 
|  | void foo(int x); | 
|  | void foo(int x, float y); | 
|  | void foo(float x, int y); | 
|  | void foo(float x, float y); | 
|  | void foo(int x, int y = 0); | 
|  | int main() { foo(^); } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | ElementsAre(sig("foo([[int x]]) -> void"), | 
|  | sig("foo([[int x]], [[int y = 0]]) -> void"), | 
|  | sig("foo([[float x]], [[int y]]) -> void"), | 
|  | sig("foo([[int x]], [[float y]]) -> void"), | 
|  | sig("foo([[float x]], [[float y]]) -> void"))); | 
|  | // We always prefer the first signature. | 
|  | EXPECT_EQ(0, Results.activeSignature); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, InstantiatedSignatures) { | 
|  | StringRef Sig0 = R"cpp( | 
|  | template <class T> | 
|  | void foo(T, T, T); | 
|  |  | 
|  | int main() { | 
|  | foo<int>(^); | 
|  | } | 
|  | )cpp"; | 
|  |  | 
|  | EXPECT_THAT(signatures(Sig0).signatures, | 
|  | ElementsAre(sig("foo([[T]], [[T]], [[T]]) -> void"))); | 
|  |  | 
|  | StringRef Sig1 = R"cpp( | 
|  | template <class T> | 
|  | void foo(T, T, T); | 
|  |  | 
|  | int main() { | 
|  | foo(10, ^); | 
|  | })cpp"; | 
|  |  | 
|  | EXPECT_THAT(signatures(Sig1).signatures, | 
|  | ElementsAre(sig("foo([[T]], [[T]], [[T]]) -> void"))); | 
|  |  | 
|  | StringRef Sig2 = R"cpp( | 
|  | template <class ...T> | 
|  | void foo(T...); | 
|  |  | 
|  | int main() { | 
|  | foo<int>(^); | 
|  | } | 
|  | )cpp"; | 
|  |  | 
|  | EXPECT_THAT(signatures(Sig2).signatures, | 
|  | ElementsAre(sig("foo([[T...]]) -> void"))); | 
|  |  | 
|  | // It is debatable whether we should substitute the outer template parameter | 
|  | // ('T') in that case. Currently we don't substitute it in signature help, but | 
|  | // do substitute in code complete. | 
|  | // FIXME: make code complete and signature help consistent, figure out which | 
|  | // way is better. | 
|  | StringRef Sig3 = R"cpp( | 
|  | template <class T> | 
|  | struct X { | 
|  | template <class U> | 
|  | void foo(T, U); | 
|  | }; | 
|  |  | 
|  | int main() { | 
|  | X<int>().foo<double>(^) | 
|  | } | 
|  | )cpp"; | 
|  |  | 
|  | EXPECT_THAT(signatures(Sig3).signatures, | 
|  | ElementsAre(sig("foo([[T]], [[U]]) -> void"))); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, IndexDocumentation) { | 
|  | Symbol Foo0 = sym("foo", index::SymbolKind::Function, "@F@\\0#"); | 
|  | Foo0.Documentation = "doc from the index"; | 
|  | Symbol Foo1 = sym("foo", index::SymbolKind::Function, "@F@\\0#I#"); | 
|  | Foo1.Documentation = "doc from the index"; | 
|  | Symbol Foo2 = sym("foo", index::SymbolKind::Function, "@F@\\0#I#I#"); | 
|  |  | 
|  | StringRef Sig0 = R"cpp( | 
|  | int foo(); | 
|  | int foo(double); | 
|  |  | 
|  | void test() { | 
|  | foo(^); | 
|  | } | 
|  | )cpp"; | 
|  |  | 
|  | EXPECT_THAT( | 
|  | signatures(Sig0, {Foo0}).signatures, | 
|  | ElementsAre(AllOf(sig("foo() -> int"), sigDoc("doc from the index")), | 
|  | AllOf(sig("foo([[double]]) -> int"), sigDoc("")))); | 
|  |  | 
|  | StringRef Sig1 = R"cpp( | 
|  | int foo(); | 
|  | // Overriden doc from sema | 
|  | int foo(int); | 
|  | // doc from sema | 
|  | int foo(int, int); | 
|  |  | 
|  | void test() { | 
|  | foo(^); | 
|  | } | 
|  | )cpp"; | 
|  |  | 
|  | EXPECT_THAT( | 
|  | signatures(Sig1, {Foo0, Foo1, Foo2}).signatures, | 
|  | ElementsAre( | 
|  | AllOf(sig("foo() -> int"), sigDoc("doc from the index")), | 
|  | AllOf(sig("foo([[int]]) -> int"), sigDoc("Overriden doc from sema")), | 
|  | AllOf(sig("foo([[int]], [[int]]) -> int"), sigDoc("doc from sema")))); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, DynamicIndexDocumentation) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  | ClangdServer::Options Opts = ClangdServer::optsForTest(); | 
|  | Opts.BuildDynamicSymbolIndex = true; | 
|  | ClangdServer Server(CDB, FS, Opts); | 
|  |  | 
|  | FS.Files[testPath("foo.h")] = R"cpp( | 
|  | struct Foo { | 
|  | // Member doc | 
|  | int foo(); | 
|  | }; | 
|  | )cpp"; | 
|  | Annotations FileContent(R"cpp( | 
|  | #include "foo.h" | 
|  | void test() { | 
|  | Foo f; | 
|  | f.foo(^); | 
|  | } | 
|  | )cpp"); | 
|  | auto File = testPath("test.cpp"); | 
|  | Server.addDocument(File, FileContent.code()); | 
|  | // Wait for the dynamic index being built. | 
|  | ASSERT_TRUE(Server.blockUntilIdleForTest()); | 
|  | EXPECT_THAT(llvm::cantFail(runSignatureHelp(Server, File, FileContent.point(), | 
|  | MarkupKind::PlainText)) | 
|  | .signatures, | 
|  | ElementsAre(AllOf(sig("foo() -> int"), sigDoc("Member doc")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompletionFunctionArgsDisabled) { | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.EnableSnippets = true; | 
|  | Opts.EnableFunctionArgSnippets = false; | 
|  |  | 
|  | { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | void xfoo(); | 
|  | void xfoo(int x, int y); | 
|  | void f() { xfo^ })cpp", | 
|  | {}, Opts); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix("()")), | 
|  | AllOf(named("xfoo"), snippetSuffix("($0)")))); | 
|  | } | 
|  | { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | void xbar(); | 
|  | void f() { xba^ })cpp", | 
|  | {}, Opts); | 
|  | EXPECT_THAT(Results.Completions, UnorderedElementsAre(AllOf( | 
|  | named("xbar"), snippetSuffix("()")))); | 
|  | } | 
|  | { | 
|  | Opts.BundleOverloads = true; | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | void xfoo(); | 
|  | void xfoo(int x, int y); | 
|  | void f() { xfo^ })cpp", | 
|  | {}, Opts); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix("($0)")))); | 
|  | } | 
|  | { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | template <class T, class U> | 
|  | void xfoo(int a, U b); | 
|  | void f() { xfo^ })cpp", | 
|  | {}, Opts); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix("<$1>($0)")))); | 
|  | } | 
|  | { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | template <class T> | 
|  | class foo_class{}; | 
|  | template <class T> | 
|  | using foo_alias = T**; | 
|  | template <class T> | 
|  | T foo_var = T{}; | 
|  | void f() { foo_^ })cpp", | 
|  | {}, Opts); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre(AllOf(named("foo_class"), snippetSuffix("<$0>")), | 
|  | AllOf(named("foo_alias"), snippetSuffix("<$0>")), | 
|  | AllOf(named("foo_var"), snippetSuffix("<$0>")))); | 
|  | } | 
|  | { | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | #define FOO(x, y) x##f | 
|  | FO^ )cpp", | 
|  | {}, Opts); | 
|  | EXPECT_THAT(Results.Completions, UnorderedElementsAre(AllOf( | 
|  | named("FOO"), snippetSuffix("($0)")))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, SuggestOverrides) { | 
|  | constexpr const char *const Text(R"cpp( | 
|  | class A { | 
|  | public: | 
|  | virtual void vfunc(bool param); | 
|  | virtual void vfunc(bool param, int p); | 
|  | void func(bool param); | 
|  | }; | 
|  | class B : public A { | 
|  | virtual void ttt(bool param) const; | 
|  | void vfunc(bool param, int p) override; | 
|  | }; | 
|  | class C : public B { | 
|  | public: | 
|  | void vfunc(bool param) override; | 
|  | ^ | 
|  | }; | 
|  | )cpp"); | 
|  | const auto Results = completions(Text); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | AllOf(Contains(AllOf(labeled("void vfunc(bool param, int p) override"), | 
|  | nameStartsWith("vfunc"))), | 
|  | Contains(AllOf(labeled("void ttt(bool param) const override"), | 
|  | nameStartsWith("ttt"))), | 
|  | Not(Contains(labeled("void vfunc(bool param) override"))))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, OverridesNonIdentName) { | 
|  | // Check the completions call does not crash. | 
|  | completions(R"cpp( | 
|  | struct Base { | 
|  | virtual ~Base() = 0; | 
|  | virtual operator int() = 0; | 
|  | virtual Base& operator+(Base&) = 0; | 
|  | }; | 
|  |  | 
|  | struct Derived : Base { | 
|  | ^ | 
|  | }; | 
|  | )cpp"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoCrashOnMissingNewLineAtEOF) { | 
|  | auto FooCpp = testPath("foo.cpp"); | 
|  |  | 
|  | MockCompilationDatabase CDB; | 
|  | MockFS FS; | 
|  | Annotations F("#pragma ^ // no new line"); | 
|  | FS.Files[FooCpp] = F.code().str(); | 
|  | ClangdServer Server(CDB, FS, ClangdServer::optsForTest()); | 
|  | runAddDocument(Server, FooCpp, F.code()); | 
|  | // Run completion outside the file range. | 
|  | EXPECT_THAT(cantFail(runCodeComplete(Server, FooCpp, F.point(), | 
|  | clangd::CodeCompleteOptions())) | 
|  | .Completions, | 
|  | IsEmpty()); | 
|  | EXPECT_THAT(cantFail(runSignatureHelp(Server, FooCpp, F.point(), | 
|  | MarkupKind::PlainText)) | 
|  | .signatures, | 
|  | IsEmpty()); | 
|  | } | 
|  |  | 
|  | TEST(GuessCompletionPrefix, Filters) { | 
|  | for (llvm::StringRef Case : { | 
|  | "[[scope::]][[ident]]^", | 
|  | "[[]][[]]^", | 
|  | "\n[[]][[]]^", | 
|  | "[[]][[ab]]^", | 
|  | "x.[[]][[ab]]^", | 
|  | "x.[[]][[]]^", | 
|  | "[[x::]][[ab]]^", | 
|  | "[[x::]][[]]^", | 
|  | "[[::x::]][[ab]]^", | 
|  | "some text [[scope::more::]][[identif]]^ier", | 
|  | "some text [[scope::]][[mor]]^e::identifier", | 
|  | "weird case foo::[[::bar::]][[baz]]^", | 
|  | "/* [[]][[]]^ */", | 
|  | }) { | 
|  | Annotations F(Case); | 
|  | auto Offset = cantFail(positionToOffset(F.code(), F.point())); | 
|  | auto ToStringRef = [&](Range R) { | 
|  | return F.code().slice(cantFail(positionToOffset(F.code(), R.start)), | 
|  | cantFail(positionToOffset(F.code(), R.end))); | 
|  | }; | 
|  | auto WantQualifier = ToStringRef(F.ranges()[0]), | 
|  | WantName = ToStringRef(F.ranges()[1]); | 
|  |  | 
|  | auto Prefix = guessCompletionPrefix(F.code(), Offset); | 
|  | // Even when components are empty, check their offsets are correct. | 
|  | EXPECT_EQ(WantQualifier, Prefix.Qualifier) << Case; | 
|  | EXPECT_EQ(WantQualifier.begin(), Prefix.Qualifier.begin()) << Case; | 
|  | EXPECT_EQ(WantName, Prefix.Name) << Case; | 
|  | EXPECT_EQ(WantName.begin(), Prefix.Name.begin()) << Case; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, EnableSpeculativeIndexRequest) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  | ClangdServer Server(CDB, FS, ClangdServer::optsForTest()); | 
|  |  | 
|  | auto File = testPath("foo.cpp"); | 
|  | Annotations Test(R"cpp( | 
|  | namespace ns1 { int abc; } | 
|  | namespace ns2 { int abc; } | 
|  | void f() { ns1::ab$1^; ns1::ab$2^; } | 
|  | void f2() { ns2::ab$3^; } | 
|  | )cpp"); | 
|  | runAddDocument(Server, File, Test.code()); | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  |  | 
|  | IndexRequestCollector Requests; | 
|  | Opts.Index = &Requests; | 
|  |  | 
|  | auto CompleteAtPoint = [&](StringRef P) { | 
|  | auto CCR = cantFail(runCodeComplete(Server, File, Test.point(P), Opts)); | 
|  | EXPECT_TRUE(CCR.HasMore); | 
|  | }; | 
|  |  | 
|  | CompleteAtPoint("1"); | 
|  | auto Reqs1 = Requests.consumeRequests(1); | 
|  | ASSERT_EQ(Reqs1.size(), 1u); | 
|  | EXPECT_THAT(Reqs1[0].Scopes, UnorderedElementsAre("ns1::")); | 
|  |  | 
|  | CompleteAtPoint("2"); | 
|  | auto Reqs2 = Requests.consumeRequests(1); | 
|  | // Speculation succeeded. Used speculative index result. | 
|  | ASSERT_EQ(Reqs2.size(), 1u); | 
|  | EXPECT_EQ(Reqs2[0], Reqs1[0]); | 
|  |  | 
|  | CompleteAtPoint("3"); | 
|  | // Speculation failed. Sent speculative index request and the new index | 
|  | // request after sema. | 
|  | auto Reqs3 = Requests.consumeRequests(2); | 
|  | ASSERT_EQ(Reqs3.size(), 2u); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, InsertTheMostPopularHeader) { | 
|  | std::string DeclFile = URI::create(testPath("foo")).toString(); | 
|  | Symbol Sym = func("Func"); | 
|  | Sym.CanonicalDeclaration.FileURI = DeclFile.c_str(); | 
|  | Sym.IncludeHeaders.emplace_back("\"foo.h\"", 2, Symbol::Include); | 
|  | Sym.IncludeHeaders.emplace_back("\"bar.h\"", 1000, Symbol::Include); | 
|  |  | 
|  | auto Results = completions("Fun^", {Sym}).Completions; | 
|  | assert(!Results.empty()); | 
|  | EXPECT_THAT(Results[0], AllOf(named("Func"), insertInclude("\"bar.h\""))); | 
|  | EXPECT_EQ(Results[0].Includes.size(), 2u); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, InsertIncludeOrImport) { | 
|  | std::string DeclFile = URI::create(testPath("foo")).toString(); | 
|  | Symbol Sym = func("Func"); | 
|  | Sym.CanonicalDeclaration.FileURI = DeclFile.c_str(); | 
|  | Sym.IncludeHeaders.emplace_back("\"bar.h\"", 1000, | 
|  | Symbol::Include | Symbol::Import); | 
|  | CodeCompleteOptions Opts; | 
|  | // Should only take effect in import contexts. | 
|  | Opts.ImportInsertions = true; | 
|  | auto Results = completions("Fun^", {Sym}, Opts).Completions; | 
|  | assert(!Results.empty()); | 
|  | EXPECT_THAT(Results[0], | 
|  | AllOf(named("Func"), insertIncludeText("#include \"bar.h\"\n"))); | 
|  |  | 
|  | ASTSignals Signals; | 
|  | Signals.InsertionDirective = Symbol::IncludeDirective::Import; | 
|  | Opts.MainFileSignals = &Signals; | 
|  | Results = completions("Fun^", {Sym}, Opts, "Foo.m").Completions; | 
|  | assert(!Results.empty()); | 
|  | EXPECT_THAT(Results[0], | 
|  | AllOf(named("Func"), insertIncludeText("#import \"bar.h\"\n"))); | 
|  |  | 
|  | Sym.IncludeHeaders[0].SupportedDirectives = Symbol::Import; | 
|  | Results = completions("Fun^", {Sym}).Completions; | 
|  | assert(!Results.empty()); | 
|  | EXPECT_THAT(Results[0], AllOf(named("Func"), Not(insertInclude()))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoInsertIncludeIfOnePresent) { | 
|  | Annotations Test(R"cpp( | 
|  | #include "foo.h" | 
|  | Fun^ | 
|  | )cpp"); | 
|  | auto TU = TestTU::withCode(Test.code()); | 
|  | TU.AdditionalFiles["foo.h"] = ""; | 
|  |  | 
|  | std::string DeclFile = URI::create(testPath("foo")).toString(); | 
|  | Symbol Sym = func("Func"); | 
|  | Sym.CanonicalDeclaration.FileURI = DeclFile.c_str(); | 
|  | Sym.IncludeHeaders.emplace_back("\"foo.h\"", 2, Symbol::Include); | 
|  | Sym.IncludeHeaders.emplace_back("\"bar.h\"", 1000, Symbol::Include); | 
|  |  | 
|  | EXPECT_THAT(completions(TU, Test.point(), {Sym}).Completions, | 
|  | UnorderedElementsAre(AllOf(named("Func"), hasInclude("\"foo.h\""), | 
|  | Not(insertInclude())))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, MergeMacrosFromIndexAndSema) { | 
|  | Symbol Sym; | 
|  | Sym.Name = "Clangd_Macro_Test"; | 
|  | Sym.ID = SymbolID("c:foo.cpp@8@macro@Clangd_Macro_Test"); | 
|  | Sym.SymInfo.Kind = index::SymbolKind::Macro; | 
|  | Sym.Flags |= Symbol::IndexedForCodeCompletion; | 
|  | EXPECT_THAT(completions("#define Clangd_Macro_Test\nClangd_Macro_T^", {Sym}) | 
|  | .Completions, | 
|  | UnorderedElementsAre(named("Clangd_Macro_Test"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, MacroFromPreamble) { | 
|  | Annotations Test(R"cpp(#define CLANGD_PREAMBLE_MAIN x | 
|  |  | 
|  | int x = 0; | 
|  | #define CLANGD_MAIN x | 
|  | void f() { CLANGD_^ } | 
|  | )cpp"); | 
|  | auto TU = TestTU::withCode(Test.code()); | 
|  | TU.HeaderCode = "#define CLANGD_PREAMBLE_HEADER x"; | 
|  | auto Results = completions(TU, Test.point(), {func("CLANGD_INDEX")}); | 
|  | // We should get results from the main file, including the preamble section. | 
|  | // However no results from included files (the index should cover them). | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(named("CLANGD_PREAMBLE_MAIN"), | 
|  | named("CLANGD_MAIN"), | 
|  | named("CLANGD_INDEX"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DeprecatedResults) { | 
|  | std::string Body = R"cpp( | 
|  | void TestClangd(); | 
|  | void TestClangc() __attribute__((deprecated("", ""))); | 
|  | )cpp"; | 
|  |  | 
|  | EXPECT_THAT( | 
|  | completions(Body + "int main() { TestClang^ }").Completions, | 
|  | UnorderedElementsAre(AllOf(named("TestClangd"), Not(deprecated())), | 
|  | AllOf(named("TestClangc"), deprecated()))); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, PartialSpec) { | 
|  | const auto Results = signatures(R"cpp( | 
|  | template <typename T> struct Foo {}; | 
|  | template <typename T> struct Foo<T*> { Foo(T); }; | 
|  | Foo<int*> F(^);)cpp"); | 
|  | EXPECT_THAT(Results.signatures, Contains(sig("Foo([[T]])"))); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, InsideArgument) { | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | void foo(int x); | 
|  | void foo(int x, int y); | 
|  | int main() { foo(1+^); } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | ElementsAre(sig("foo([[int x]]) -> void"), | 
|  | sig("foo([[int x]], [[int y]]) -> void"))); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | void foo(int x); | 
|  | void foo(int x, int y); | 
|  | int main() { foo(1^); } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | ElementsAre(sig("foo([[int x]]) -> void"), | 
|  | sig("foo([[int x]], [[int y]]) -> void"))); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | void foo(int x); | 
|  | void foo(int x, int y); | 
|  | int main() { foo(1^0); } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | ElementsAre(sig("foo([[int x]]) -> void"), | 
|  | sig("foo([[int x]], [[int y]]) -> void"))); | 
|  | EXPECT_EQ(0, Results.activeParameter); | 
|  | } | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | void foo(int x); | 
|  | void foo(int x, int y); | 
|  | int bar(int x, int y); | 
|  | int main() { bar(foo(2, 3^)); } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | ElementsAre(sig("foo([[int x]], [[int y]]) -> void"))); | 
|  | EXPECT_EQ(1, Results.activeParameter); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, ConstructorInitializeFields) { | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | struct A { A(int); }; | 
|  | struct B { | 
|  | B() : a_elem(^) {} | 
|  | A a_elem; | 
|  | }; | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | UnorderedElementsAre(sig("A([[int]])"), sig("A([[A &&]])"), | 
|  | sig("A([[const A &]])"))); | 
|  | } | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | struct A { A(int); }; | 
|  | struct B { | 
|  | B() : a_elem(^ | 
|  | A a_elem; | 
|  | }; | 
|  | )cpp"); | 
|  | // FIXME: currently the parser skips over the decl of a_elem as part of the | 
|  | // (broken) init list, so we don't get signatures for the first member. | 
|  | EXPECT_THAT(Results.signatures, IsEmpty()); | 
|  | } | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | struct A { A(int); }; | 
|  | struct B { | 
|  | B() : a_elem(^ | 
|  | int dummy_elem; | 
|  | A a_elem; | 
|  | }; | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | UnorderedElementsAre(sig("A([[int]])"), sig("A([[A &&]])"), | 
|  | sig("A([[const A &]])"))); | 
|  | } | 
|  | { | 
|  | const auto Results = signatures(R"cpp( | 
|  | struct A { | 
|  | A(int); | 
|  | }; | 
|  | struct C { | 
|  | C(int); | 
|  | C(A); | 
|  | }; | 
|  | struct B { | 
|  | B() : c_elem(A(1^)) {} | 
|  | C c_elem; | 
|  | }; | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.signatures, | 
|  | UnorderedElementsAre(sig("A([[int]])"), sig("A([[A &&]])"), | 
|  | sig("A([[const A &]])"))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, Variadic) { | 
|  | const std::string Header = R"cpp( | 
|  | void fun(int x, ...) {} | 
|  | void test() {)cpp"; | 
|  | const std::string ExpectedSig = "fun([[int x]], [[...]]) -> void"; | 
|  |  | 
|  | { | 
|  | const auto Result = signatures(Header + "fun(^);}"); | 
|  | EXPECT_EQ(0, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "fun(1, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "fun(1, 2, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, VariadicTemplate) { | 
|  | const std::string Header = R"cpp( | 
|  | template<typename T, typename ...Args> | 
|  | void fun(T t, Args ...args) {} | 
|  | void test() {)cpp"; | 
|  | const std::string ExpectedSig = "fun([[T t]], [[Args args...]]) -> void"; | 
|  |  | 
|  | { | 
|  | const auto Result = signatures(Header + "fun(^);}"); | 
|  | EXPECT_EQ(0, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "fun(1, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "fun(1, 2, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, VariadicMethod) { | 
|  | const std::string Header = R"cpp( | 
|  | class C { | 
|  | template<typename T, typename ...Args> | 
|  | void fun(T t, Args ...args) {} | 
|  | }; | 
|  | void test() {C c; )cpp"; | 
|  | const std::string ExpectedSig = "fun([[T t]], [[Args args...]]) -> void"; | 
|  |  | 
|  | { | 
|  | const auto Result = signatures(Header + "c.fun(^);}"); | 
|  | EXPECT_EQ(0, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "c.fun(1, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "c.fun(1, 2, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelpTest, VariadicType) { | 
|  | const std::string Header = R"cpp( | 
|  | void fun(int x, ...) {} | 
|  | auto get_fun() { return fun; } | 
|  | void test() { | 
|  | )cpp"; | 
|  | const std::string ExpectedSig = "([[int]], [[...]]) -> void"; | 
|  |  | 
|  | { | 
|  | const auto Result = signatures(Header + "get_fun()(^);}"); | 
|  | EXPECT_EQ(0, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "get_fun()(1, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | { | 
|  | const auto Result = signatures(Header + "get_fun()(1, 2, ^);}"); | 
|  | EXPECT_EQ(1, Result.activeParameter); | 
|  | EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, IncludedCompletionKinds) { | 
|  | Annotations Test(R"cpp(#include "^)cpp"); | 
|  | auto TU = TestTU::withCode(Test.code()); | 
|  | TU.AdditionalFiles["sub/bar.h"] = ""; | 
|  | TU.ExtraArgs.push_back("-I" + testPath("sub")); | 
|  |  | 
|  | auto Results = completions(TU, Test.point()); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | AllOf(has("sub/", CompletionItemKind::Folder), | 
|  | has("bar.h\"", CompletionItemKind::File))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoCrashAtNonAlphaIncludeHeader) { | 
|  | completions( | 
|  | R"cpp( | 
|  | #include "./^" | 
|  | )cpp"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoAllScopesCompletionWhenQualified) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  | Opts.AllScopes = true; | 
|  |  | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | void f() { na::Clangd^ } | 
|  | )cpp", | 
|  | {cls("na::ClangdA"), cls("nx::ClangdX"), cls("Clangd3")}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(qualifier(""), scope("na::"), named("ClangdA")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, AllScopesCompletion) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  | Opts.AllScopes = true; | 
|  |  | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | namespace na { | 
|  | void f() { Clangd^ } | 
|  | } | 
|  | )cpp", | 
|  | {cls("nx::Clangd1"), cls("ny::Clangd2"), cls("Clangd3"), | 
|  | cls("na::nb::Clangd4"), enmConstant("na::C::Clangd5")}, | 
|  | Opts); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier("nx::"), named("Clangd1"), | 
|  | kind(CompletionItemKind::Class)), | 
|  | AllOf(qualifier("ny::"), named("Clangd2"), | 
|  | kind(CompletionItemKind::Class)), | 
|  | AllOf(qualifier(""), scope(""), named("Clangd3"), | 
|  | kind(CompletionItemKind::Class)), | 
|  | AllOf(qualifier("nb::"), named("Clangd4"), | 
|  | kind(CompletionItemKind::Class)), | 
|  | AllOf(qualifier("C::"), named("Clangd5"), | 
|  | kind(CompletionItemKind::EnumMember)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoQualifierIfShadowed) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  | Opts.AllScopes = true; | 
|  |  | 
|  | auto Results = completions(R"cpp( | 
|  | namespace nx { class Clangd1 {}; } | 
|  | using nx::Clangd1; | 
|  | void f() { Clangd^ } | 
|  | )cpp", | 
|  | {cls("nx::Clangd1"), cls("nx::Clangd2")}, Opts); | 
|  | // Although Clangd1 is from another namespace, Sema tells us it's in-scope and | 
|  | // needs no qualifier. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier(""), named("Clangd1")), | 
|  | AllOf(qualifier("nx::"), named("Clangd2")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoCompletionsForNewNames) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.AllScopes = true; | 
|  | auto Results = completions(R"cpp( | 
|  | void f() { int n^ } | 
|  | )cpp", | 
|  | {cls("naber"), cls("nx::naber")}, Opts); | 
|  | EXPECT_THAT(Results.Completions, UnorderedElementsAre()); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Lambda) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  |  | 
|  | auto Results = completions(R"cpp( | 
|  | void function() { | 
|  | auto Lambda = [](int a, const double &b) {return 1.f;}; | 
|  | Lam^ | 
|  | } | 
|  | )cpp", | 
|  | {}, Opts); | 
|  |  | 
|  | ASSERT_EQ(Results.Completions.size(), 1u); | 
|  | const auto &A = Results.Completions.front(); | 
|  | EXPECT_EQ(A.Name, "Lambda"); | 
|  | EXPECT_EQ(A.Signature, "(int a, const double &b) const"); | 
|  | EXPECT_EQ(A.Kind, CompletionItemKind::Variable); | 
|  | EXPECT_EQ(A.ReturnType, "float"); | 
|  | EXPECT_EQ(A.SnippetSuffix, "(${1:int a}, ${2:const double &b})"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, StructuredBinding) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  |  | 
|  | auto Results = completions(R"cpp( | 
|  | struct S { | 
|  | using Float = float; | 
|  | int x; | 
|  | Float y; | 
|  | }; | 
|  | void function() { | 
|  | const auto &[xxx, yyy] = S{}; | 
|  | yyy^ | 
|  | } | 
|  | )cpp", | 
|  | {}, Opts); | 
|  |  | 
|  | ASSERT_EQ(Results.Completions.size(), 1u); | 
|  | const auto &A = Results.Completions.front(); | 
|  | EXPECT_EQ(A.Name, "yyy"); | 
|  | EXPECT_EQ(A.Kind, CompletionItemKind::Variable); | 
|  | EXPECT_EQ(A.ReturnType, "const Float"); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodNoArguments) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | @property(nonatomic, setter=setXToIgnoreComplete:) int value; | 
|  | @end | 
|  | Foo *foo = [Foo new]; int y = [foo v^] | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("value"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(returnType("int"))); | 
|  | EXPECT_THAT(C, ElementsAre(signature(""))); | 
|  | EXPECT_THAT(C, ElementsAre(snippetSuffix(""))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodOneArgument) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | - (int)valueForCharacter:(char)c; | 
|  | @end | 
|  | Foo *foo = [Foo new]; int y = [foo v^] | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("valueForCharacter:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(returnType("int"))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(char)"))); | 
|  | EXPECT_THAT(C, ElementsAre(snippetSuffix("${1:(char)}"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodTwoArgumentsFromBeginning) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | + (id)fooWithValue:(int)value fooey:(unsigned int)fooey; | 
|  | @end | 
|  | id val = [Foo foo^] | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("fooWithValue:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(returnType("id"))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(int) fooey:(unsigned int)"))); | 
|  | EXPECT_THAT( | 
|  | C, ElementsAre(snippetSuffix("${1:(int)} fooey:${2:(unsigned int)}"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodTwoArgumentsFromMiddle) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | + (id)fooWithValue:(int)value fooey:(unsigned int)fooey; | 
|  | @end | 
|  | id val = [Foo fooWithValue:10 f^] | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("fooey:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(returnType("id"))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(unsigned int)"))); | 
|  | EXPECT_THAT(C, ElementsAre(snippetSuffix("${1:(unsigned int)}"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodFilterOnEntireSelector) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | + (id)player:(id)player willRun:(id)run; | 
|  | @end | 
|  | id val = [Foo wi^] | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("player:"))); | 
|  | EXPECT_THAT(C, ElementsAre(filterText("player:willRun:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(returnType("id"))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(id) willRun:(id)"))); | 
|  | EXPECT_THAT(C, ElementsAre(snippetSuffix("${1:(id)} willRun:${2:(id)}"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCSimpleMethodDeclaration) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | - (void)foo; | 
|  | @end | 
|  | @implementation Foo | 
|  | fo^ | 
|  | @end | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("foo"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(qualifier("- (void)"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodDeclaration) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | - (int)valueForCharacter:(char)c secondArgument:(id)object; | 
|  | @end | 
|  | @implementation Foo | 
|  | valueFor^ | 
|  | @end | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("valueForCharacter:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(qualifier("- (int)"))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(char)c secondArgument:(id)object"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodDeclarationFilterOnEntireSelector) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | - (int)valueForCharacter:(char)c secondArgument:(id)object; | 
|  | @end | 
|  | @implementation Foo | 
|  | secondArg^ | 
|  | @end | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("valueForCharacter:"))); | 
|  | EXPECT_THAT(C, ElementsAre(filterText("valueForCharacter:secondArgument:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(qualifier("- (int)"))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(char)c secondArgument:(id)object"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodDeclarationPrefixTyped) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | - (int)valueForCharacter:(char)c; | 
|  | @end | 
|  | @implementation Foo | 
|  | - (int)valueFor^ | 
|  | @end | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("valueForCharacter:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(char)c"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCMethodDeclarationFromMiddle) { | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | - (int)valueForCharacter:(char)c secondArgument:(id)object; | 
|  | @end | 
|  | @implementation Foo | 
|  | - (int)valueForCharacter:(char)c second^ | 
|  | @end | 
|  | )objc", | 
|  | /*IndexSymbols=*/{}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | auto C = Results.Completions; | 
|  | EXPECT_THAT(C, ElementsAre(named("secondArgument:"))); | 
|  | EXPECT_THAT(C, ElementsAre(kind(CompletionItemKind::Method))); | 
|  | EXPECT_THAT(C, ElementsAre(signature("(id)object"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCProtocolFromIndex) { | 
|  | Symbol FoodClass = objcClass("FoodClass"); | 
|  | Symbol SymFood = objcProtocol("Food"); | 
|  | Symbol SymFooey = objcProtocol("Fooey"); | 
|  | auto Results = completions("id<Foo^>", {SymFood, FoodClass, SymFooey}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | // Should only give protocols for ObjC protocol completions. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(named("Food"), kind(CompletionItemKind::Interface)), | 
|  | AllOf(named("Fooey"), kind(CompletionItemKind::Interface)))); | 
|  |  | 
|  | Results = completions("Fo^", {SymFood, FoodClass, SymFooey}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  | // Shouldn't give protocols for non protocol completions. | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | ElementsAre(AllOf(named("FoodClass"), kind(CompletionItemKind::Class)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCProtocolFromIndexSpeculation) { | 
|  | MockFS FS; | 
|  | MockCompilationDatabase CDB; | 
|  | ClangdServer Server(CDB, FS, ClangdServer::optsForTest()); | 
|  |  | 
|  | auto File = testPath("Foo.m"); | 
|  | Annotations Test(R"cpp( | 
|  | @protocol Food | 
|  | @end | 
|  | id<Foo$1^> foo; | 
|  | Foo$2^ bar; | 
|  | )cpp"); | 
|  | runAddDocument(Server, File, Test.code()); | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  |  | 
|  | Symbol FoodClass = objcClass("FoodClass"); | 
|  | IndexRequestCollector Requests({FoodClass}); | 
|  | Opts.Index = &Requests; | 
|  |  | 
|  | auto CompleteAtPoint = [&](StringRef P) { | 
|  | return cantFail(runCodeComplete(Server, File, Test.point(P), Opts)) | 
|  | .Completions; | 
|  | }; | 
|  |  | 
|  | auto C = CompleteAtPoint("1"); | 
|  | auto Reqs1 = Requests.consumeRequests(1); | 
|  | ASSERT_EQ(Reqs1.size(), 1u); | 
|  | EXPECT_THAT(C, ElementsAre(AllOf(named("Food"), | 
|  | kind(CompletionItemKind::Interface)))); | 
|  |  | 
|  | C = CompleteAtPoint("2"); | 
|  | auto Reqs2 = Requests.consumeRequests(1); | 
|  | // Speculation succeeded. Used speculative index result, but filtering now to | 
|  | // now include FoodClass. | 
|  | ASSERT_EQ(Reqs2.size(), 1u); | 
|  | EXPECT_EQ(Reqs2[0], Reqs1[0]); | 
|  | EXPECT_THAT(C, ElementsAre(AllOf(named("FoodClass"), | 
|  | kind(CompletionItemKind::Class)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCCategoryFromIndexIgnored) { | 
|  | Symbol FoodCategory = objcCategory("FoodClass", "Extension"); | 
|  | auto Results = completions(R"objc( | 
|  | @interface Foo | 
|  | @end | 
|  | @interface Foo (^) | 
|  | @end | 
|  | )objc", | 
|  | {FoodCategory}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  | EXPECT_THAT(Results.Completions, IsEmpty()); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjectiveCForwardDeclFromIndex) { | 
|  | Symbol FoodClass = objcClass("FoodClass"); | 
|  | FoodClass.IncludeHeaders.emplace_back("\"Foo.h\"", 2, Symbol::Import); | 
|  | Symbol SymFood = objcProtocol("Food"); | 
|  | auto Results = completions("@class Foo^", {SymFood, FoodClass}, | 
|  | /*Opts=*/{}, "Foo.m"); | 
|  |  | 
|  | // Should only give class names without any include insertion. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(named("FoodClass"), | 
|  | kind(CompletionItemKind::Class), | 
|  | Not(insertInclude())))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CursorInSnippets) { | 
|  | clangd::CodeCompleteOptions Options; | 
|  | Options.EnableSnippets = true; | 
|  | auto Results = completions( | 
|  | R"cpp( | 
|  | void while_foo(int a, int b); | 
|  | void test() { | 
|  | whil^ | 
|  | })cpp", | 
|  | /*IndexSymbols=*/{}, Options); | 
|  |  | 
|  | // Last placeholder in code patterns should be $0 to put the cursor there. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("while"), | 
|  | snippetSuffix(" (${1:condition}) {\n$0\n}")))); | 
|  | // However, snippets for functions must *not* end with $0. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("while_foo"), | 
|  | snippetSuffix("(${1:int a}, ${2:int b})")))); | 
|  |  | 
|  | Results = completions(R"cpp( | 
|  | struct Base { | 
|  | Base(int a, int b) {} | 
|  | }; | 
|  |  | 
|  | struct Derived : Base { | 
|  | Derived() : Base^ | 
|  | }; | 
|  | )cpp", | 
|  | /*IndexSymbols=*/{}, Options); | 
|  | // Constructors from base classes are a kind of pattern that shouldn't end | 
|  | // with $0. | 
|  | EXPECT_THAT(Results.Completions, | 
|  | Contains(AllOf(named("Base"), | 
|  | snippetSuffix("(${1:int a}, ${2:int b})")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, WorksWithNullType) { | 
|  | auto R = completions(R"cpp( | 
|  | int main() { | 
|  | for (auto [loopVar] : y ) { // y has to be unresolved. | 
|  | int z = loopV^; | 
|  | } | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(R.Completions, ElementsAre(named("loopVar"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, UsingDecl) { | 
|  | const char *Header(R"cpp( | 
|  | void foo(int); | 
|  | namespace std { | 
|  | using ::foo; | 
|  | })cpp"); | 
|  | const char *Source(R"cpp( | 
|  | void bar() { | 
|  | std::^; | 
|  | })cpp"); | 
|  | auto Index = TestTU::withHeaderCode(Header).index(); | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.Index = Index.get(); | 
|  | Opts.AllScopes = true; | 
|  | auto R = completions(Source, {}, Opts); | 
|  | EXPECT_THAT(R.Completions, | 
|  | ElementsAre(AllOf(scope("std::"), named("foo"), | 
|  | kind(CompletionItemKind::Reference)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Enums) { | 
|  | const char *Header(R"cpp( | 
|  | namespace ns { | 
|  | enum Unscoped { Clangd1 }; | 
|  | class C { | 
|  | enum Unscoped { Clangd2 }; | 
|  | }; | 
|  | enum class Scoped { Clangd3 }; | 
|  | })cpp"); | 
|  | const char *Source(R"cpp( | 
|  | void bar() { | 
|  | Clangd^ | 
|  | })cpp"); | 
|  | auto Index = TestTU::withHeaderCode(Header).index(); | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.Index = Index.get(); | 
|  | Opts.AllScopes = true; | 
|  | auto R = completions(Source, {}, Opts); | 
|  | EXPECT_THAT(R.Completions, UnorderedElementsAre( | 
|  | AllOf(scope("ns::"), named("Clangd1"), | 
|  | kind(CompletionItemKind::EnumMember)), | 
|  | AllOf(scope("ns::C::"), named("Clangd2"), | 
|  | kind(CompletionItemKind::EnumMember)), | 
|  | AllOf(scope("ns::Scoped::"), named("Clangd3"), | 
|  | kind(CompletionItemKind::EnumMember)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ScopeIsUnresolved) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  | Opts.AllScopes = true; | 
|  |  | 
|  | auto Results = completions(R"cpp( | 
|  | namespace a { | 
|  | void f() { b::X^ } | 
|  | } | 
|  | )cpp", | 
|  | {cls("a::b::XYZ")}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier(""), named("XYZ")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NestedScopeIsUnresolved) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  | Opts.AllScopes = true; | 
|  |  | 
|  | auto Results = completions(R"cpp( | 
|  | namespace a { | 
|  | namespace b {} | 
|  | void f() { b::c::X^ } | 
|  | } | 
|  | )cpp", | 
|  | {cls("a::b::c::XYZ")}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier(""), named("XYZ")))); | 
|  | } | 
|  |  | 
|  | // Clang parser gets confused here and doesn't report the ns:: prefix. | 
|  | // Naive behavior is to insert it again. We examine the source and recover. | 
|  | TEST(CompletionTest, NamespaceDoubleInsertion) { | 
|  | clangd::CodeCompleteOptions Opts = {}; | 
|  |  | 
|  | auto Results = completions(R"cpp( | 
|  | namespace foo { | 
|  | namespace ns {} | 
|  | #define M(X) < X | 
|  | M(ns::ABC^ | 
|  | } | 
|  | )cpp", | 
|  | {cls("foo::ns::ABCDE")}, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier(""), named("ABCDE")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DerivedMethodsAreAlwaysVisible) { | 
|  | // Despite the fact that base method matches the ref-qualifier better, | 
|  | // completion results should only include the derived method. | 
|  | auto Completions = completions(R"cpp( | 
|  | struct deque_base { | 
|  | float size(); | 
|  | double size() const; | 
|  | }; | 
|  | struct deque : deque_base { | 
|  | int size() const; | 
|  | }; | 
|  |  | 
|  | auto x = deque().^ | 
|  | )cpp") | 
|  | .Completions; | 
|  | EXPECT_THAT(Completions, | 
|  | ElementsAre(AllOf(returnType("int"), named("size")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoCrashWithIncompleteLambda) { | 
|  | auto Completions = completions("auto&& x = []{^").Completions; | 
|  | // The completion of x itself can cause a problem: in the code completion | 
|  | // callback, its type is not known, which affects the linkage calculation. | 
|  | // A bad linkage value gets cached, and subsequently updated. | 
|  | EXPECT_THAT(Completions, Contains(named("x"))); | 
|  |  | 
|  | auto Signatures = signatures("auto x() { x(^").signatures; | 
|  | EXPECT_THAT(Signatures, Contains(sig("x() -> auto"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DelayedTemplateParsing) { | 
|  | Annotations Test(R"cpp( | 
|  | int xxx; | 
|  | template <typename T> int foo() { return xx^; } | 
|  | )cpp"); | 
|  | auto TU = TestTU::withCode(Test.code()); | 
|  | // Even though delayed-template-parsing is on, we will disable it to provide | 
|  | // completion in templates. | 
|  | TU.ExtraArgs.push_back("-fdelayed-template-parsing"); | 
|  |  | 
|  | EXPECT_THAT(completions(TU, Test.point()).Completions, | 
|  | Contains(named("xxx"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CompletionRange) { | 
|  | const char *WithRange = "auto x = [[abc]]^"; | 
|  | auto Completions = completions(WithRange); | 
|  | EXPECT_EQ(Completions.CompletionRange, Annotations(WithRange).range()); | 
|  | Completions = completionsNoCompile(WithRange); | 
|  | EXPECT_EQ(Completions.CompletionRange, Annotations(WithRange).range()); | 
|  |  | 
|  | const char *EmptyRange = "auto x = [[]]^"; | 
|  | Completions = completions(EmptyRange); | 
|  | EXPECT_EQ(Completions.CompletionRange, Annotations(EmptyRange).range()); | 
|  | Completions = completionsNoCompile(EmptyRange); | 
|  | EXPECT_EQ(Completions.CompletionRange, Annotations(EmptyRange).range()); | 
|  |  | 
|  | // Sema doesn't trigger at all here, while the no-sema completion runs | 
|  | // heuristics as normal and reports a range. It'd be nice to be consistent. | 
|  | const char *NoCompletion = "/* foo [[]]^ */"; | 
|  | Completions = completions(NoCompletion); | 
|  | EXPECT_EQ(Completions.CompletionRange, std::nullopt); | 
|  | Completions = completionsNoCompile(NoCompletion); | 
|  | EXPECT_EQ(Completions.CompletionRange, Annotations(NoCompletion).range()); | 
|  | } | 
|  |  | 
|  | TEST(NoCompileCompletionTest, Basic) { | 
|  | auto Results = completionsNoCompile(R"cpp( | 
|  | void func() { | 
|  | int xyz; | 
|  | int abc; | 
|  | ^ | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_FALSE(Results.RanParser); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(named("void"), named("func"), named("int"), | 
|  | named("xyz"), named("abc"))); | 
|  | } | 
|  |  | 
|  | TEST(NoCompileCompletionTest, WithFilter) { | 
|  | auto Results = completionsNoCompile(R"cpp( | 
|  | void func() { | 
|  | int sym1; | 
|  | int sym2; | 
|  | int xyz1; | 
|  | int xyz2; | 
|  | sy^ | 
|  | } | 
|  | )cpp"); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(named("sym1"), named("sym2"))); | 
|  | } | 
|  |  | 
|  | TEST(NoCompileCompletionTest, WithIndex) { | 
|  | std::vector<Symbol> Syms = {func("xxx"), func("a::xxx"), func("ns::b::xxx"), | 
|  | func("c::xxx"), func("ns::d::xxx")}; | 
|  | auto Results = completionsNoCompile( | 
|  | R"cpp( | 
|  | // Current-scopes, unqualified completion. | 
|  | using namespace a; | 
|  | namespace ns { | 
|  | using namespace b; | 
|  | void foo() { | 
|  | xx^ | 
|  | } | 
|  | } | 
|  | )cpp", | 
|  | Syms); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier(""), scope("")), | 
|  | AllOf(qualifier(""), scope("a::")), | 
|  | AllOf(qualifier(""), scope("ns::b::")))); | 
|  | CodeCompleteOptions Opts; | 
|  | Opts.AllScopes = true; | 
|  | Results = completionsNoCompile( | 
|  | R"cpp( | 
|  | // All-scopes unqualified completion. | 
|  | using namespace a; | 
|  | namespace ns { | 
|  | using namespace b; | 
|  | void foo() { | 
|  | xx^ | 
|  | } | 
|  | } | 
|  | )cpp", | 
|  | Syms, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(AllOf(qualifier(""), scope("")), | 
|  | AllOf(qualifier(""), scope("a::")), | 
|  | AllOf(qualifier(""), scope("ns::b::")), | 
|  | AllOf(qualifier("c::"), scope("c::")), | 
|  | AllOf(qualifier("d::"), scope("ns::d::")))); | 
|  | Results = completionsNoCompile( | 
|  | R"cpp( | 
|  | // Qualified completion. | 
|  | using namespace a; | 
|  | namespace ns { | 
|  | using namespace b; | 
|  | void foo() { | 
|  | b::xx^ | 
|  | } | 
|  | } | 
|  | )cpp", | 
|  | Syms, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(AllOf(qualifier(""), scope("ns::b::")))); | 
|  | Results = completionsNoCompile( | 
|  | R"cpp( | 
|  | // Absolutely qualified completion. | 
|  | using namespace a; | 
|  | namespace ns { | 
|  | using namespace b; | 
|  | void foo() { | 
|  | ::a::xx^ | 
|  | } | 
|  | } | 
|  | )cpp", | 
|  | Syms, Opts); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | ElementsAre(AllOf(qualifier(""), scope("a::")))); | 
|  | } | 
|  |  | 
|  | TEST(AllowImplicitCompletion, All) { | 
|  | const char *Yes[] = { | 
|  | "foo.^bar", | 
|  | "foo->^bar", | 
|  | "foo::^bar", | 
|  | "  #  include <^foo.h>", | 
|  | "#import <foo/^bar.h>", | 
|  | "#include_next \"^", | 
|  | }; | 
|  | const char *No[] = { | 
|  | "foo>^bar", | 
|  | "foo:^bar", | 
|  | "foo\n^bar", | 
|  | "#include <foo.h> //^", | 
|  | "#include \"foo.h\"^", | 
|  | "#error <^", | 
|  | "#<^", | 
|  | }; | 
|  | for (const char *Test : Yes) { | 
|  | llvm::Annotations A(Test); | 
|  | EXPECT_TRUE(allowImplicitCompletion(A.code(), A.point())) << Test; | 
|  | } | 
|  | for (const char *Test : No) { | 
|  | llvm::Annotations A(Test); | 
|  | EXPECT_FALSE(allowImplicitCompletion(A.code(), A.point())) << Test; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, FunctionArgsExist) { | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.EnableSnippets = true; | 
|  | std::string Context = R"cpp( | 
|  | #define MACRO(x) | 
|  | int foo(int A); | 
|  | int bar(); | 
|  | struct Object { | 
|  | Object(int B) {} | 
|  | }; | 
|  | template <typename T> | 
|  | struct Container { | 
|  | Container(int Size) {} | 
|  | }; | 
|  | )cpp"; | 
|  | EXPECT_THAT(completions(Context + "int y = fo^", {}, Opts).Completions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(labeled("foo(int A)"), snippetSuffix("(${1:int A})")))); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "int y = fo^(42)", {}, Opts).Completions, | 
|  | UnorderedElementsAre(AllOf(labeled("foo(int A)"), snippetSuffix("")))); | 
|  | // FIXME(kirillbobyrev): No snippet should be produced here. | 
|  | EXPECT_THAT(completions(Context + "int y = fo^o(42)", {}, Opts).Completions, | 
|  | UnorderedElementsAre( | 
|  | AllOf(labeled("foo(int A)"), snippetSuffix("(${1:int A})")))); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "int y = ba^", {}, Opts).Completions, | 
|  | UnorderedElementsAre(AllOf(labeled("bar()"), snippetSuffix("()")))); | 
|  | EXPECT_THAT(completions(Context + "int y = ba^()", {}, Opts).Completions, | 
|  | UnorderedElementsAre(AllOf(labeled("bar()"), snippetSuffix("")))); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "Object o = Obj^", {}, Opts).Completions, | 
|  | Contains(AllOf(labeled("Object(int B)"), snippetSuffix("(${1:int B})"), | 
|  | kind(CompletionItemKind::Constructor)))); | 
|  | EXPECT_THAT(completions(Context + "Object o = Obj^()", {}, Opts).Completions, | 
|  | Contains(AllOf(labeled("Object(int B)"), snippetSuffix(""), | 
|  | kind(CompletionItemKind::Constructor)))); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "Container c = Cont^", {}, Opts).Completions, | 
|  | Contains(AllOf(labeled("Container<typename T>(int Size)"), | 
|  | snippetSuffix("<${1:typename T}>(${2:int Size})"), | 
|  | kind(CompletionItemKind::Constructor)))); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "Container c = Cont^()", {}, Opts).Completions, | 
|  | Contains(AllOf(labeled("Container<typename T>(int Size)"), | 
|  | snippetSuffix("<${1:typename T}>"), | 
|  | kind(CompletionItemKind::Constructor)))); | 
|  | EXPECT_THAT( | 
|  | completions(Context + "Container c = Cont^<int>()", {}, Opts).Completions, | 
|  | Contains(AllOf(labeled("Container<typename T>(int Size)"), | 
|  | snippetSuffix(""), | 
|  | kind(CompletionItemKind::Constructor)))); | 
|  | EXPECT_THAT(completions(Context + "MAC^(2)", {}, Opts).Completions, | 
|  | Contains(AllOf(labeled("MACRO(x)"), snippetSuffix(""), | 
|  | kind(CompletionItemKind::Function)))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, FunctionArgsExist_Issue1785) { | 
|  | // This is a scenario where the implementation of our check for | 
|  | // "is there a function argument list right after the cursor" | 
|  | // gave a bogus result. | 
|  | clangd::CodeCompleteOptions Opts; | 
|  | Opts.EnableSnippets = true; | 
|  | // The whitespace in this testcase is important! | 
|  | std::string Code = R"cpp( | 
|  | void waldo(int); | 
|  |  | 
|  | int main() | 
|  | { | 
|  | wal^ | 
|  |  | 
|  |  | 
|  | // (    ) | 
|  | } | 
|  | )cpp"; | 
|  | EXPECT_THAT( | 
|  | completions(Code, {}, Opts).Completions, | 
|  | Contains(AllOf(labeled("waldo(int)"), snippetSuffix("(${1:int})")))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, NoCrashDueToMacroOrdering) { | 
|  | EXPECT_THAT(completions(R"cpp( | 
|  | #define ECHO(X) X | 
|  | #define ECHO2(X) ECHO(X) | 
|  | int finish_preamble = EC^HO(2);)cpp") | 
|  | .Completions, | 
|  | UnorderedElementsAre(labeled("ECHO(X)"), labeled("ECHO2(X)"))); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, ObjCCategoryDecls) { | 
|  | TestTU TU; | 
|  | TU.ExtraArgs.push_back("-xobjective-c"); | 
|  | TU.HeaderCode = R"objc( | 
|  | @interface Foo | 
|  | @end | 
|  |  | 
|  | @interface Foo (FooExt1) | 
|  | @end | 
|  |  | 
|  | @interface Foo (FooExt2) | 
|  | @end | 
|  |  | 
|  | @interface Bar | 
|  | @end | 
|  |  | 
|  | @interface Bar (BarExt) | 
|  | @end)objc"; | 
|  |  | 
|  | { | 
|  | Annotations Test(R"objc( | 
|  | @implementation Foo (^) | 
|  | @end | 
|  | )objc"); | 
|  | TU.Code = Test.code().str(); | 
|  | auto Results = completions(TU, Test.point()); | 
|  | EXPECT_THAT(Results.Completions, | 
|  | UnorderedElementsAre(labeled("FooExt1"), labeled("FooExt2"))); | 
|  | } | 
|  | { | 
|  | Annotations Test(R"objc( | 
|  | @interface Foo (^) | 
|  | @end | 
|  | )objc"); | 
|  | TU.Code = Test.code().str(); | 
|  | auto Results = completions(TU, Test.point()); | 
|  | EXPECT_THAT(Results.Completions, UnorderedElementsAre(labeled("BarExt"))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, PreambleCodeComplete) { | 
|  | llvm::StringLiteral Baseline = "\n#define MACRO 12\nint num = MACRO;"; | 
|  | llvm::StringLiteral ModifiedCC = | 
|  | "#include \"header.h\"\n#define MACRO 12\nint num = MACRO; int num2 = M^"; | 
|  |  | 
|  | Annotations Test(ModifiedCC); | 
|  | auto BaselineTU = TestTU::withCode(Baseline); | 
|  | auto ModifiedTU = TestTU::withCode(Test.code()); | 
|  |  | 
|  | MockFS FS; | 
|  | auto Inputs = ModifiedTU.inputs(FS); | 
|  | auto Result = codeComplete(testPath(ModifiedTU.Filename), Test.point(), | 
|  | BaselineTU.preamble().get(), Inputs, {}); | 
|  | EXPECT_THAT(Result.Completions, Not(testing::IsEmpty())); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, CommentParamName) { | 
|  | const std::string Code = R"cpp( | 
|  | void fun(int foo, int bar); | 
|  | void overloaded(int param_int); | 
|  | void overloaded(int param_int, int param_other); | 
|  | void overloaded(char param_char); | 
|  | int main() { | 
|  | )cpp"; | 
|  |  | 
|  | EXPECT_THAT(completions(Code + "fun(/*^").Completions, | 
|  | UnorderedElementsAre(labeled("foo=*/"))); | 
|  | EXPECT_THAT(completions(Code + "fun(1, /*^").Completions, | 
|  | UnorderedElementsAre(labeled("bar=*/"))); | 
|  | EXPECT_THAT(completions(Code + "/*^").Completions, IsEmpty()); | 
|  | // Test de-duplication. | 
|  | EXPECT_THAT( | 
|  | completions(Code + "overloaded(/*^").Completions, | 
|  | UnorderedElementsAre(labeled("param_int=*/"), labeled("param_char=*/"))); | 
|  | // Comment already has some text in it. | 
|  | EXPECT_THAT(completions(Code + "fun(/*  ^").Completions, | 
|  | UnorderedElementsAre(labeled("foo=*/"))); | 
|  | EXPECT_THAT(completions(Code + "fun(/* f^").Completions, | 
|  | UnorderedElementsAre(labeled("foo=*/"))); | 
|  | EXPECT_THAT(completions(Code + "fun(/* x^").Completions, IsEmpty()); | 
|  | EXPECT_THAT(completions(Code + "fun(/* f ^").Completions, IsEmpty()); | 
|  |  | 
|  | // Test ranges | 
|  | { | 
|  | std::string CompletionRangeTest(Code + "fun(/*[[^]]"); | 
|  | auto Results = completions(CompletionRangeTest); | 
|  | EXPECT_THAT(Results.CompletionRange, | 
|  | llvm::ValueIs(Annotations(CompletionRangeTest).range())); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | testing::Each( | 
|  | AllOf(replacesRange(Annotations(CompletionRangeTest).range()), | 
|  | origin(SymbolOrigin::AST), kind(CompletionItemKind::Text)))); | 
|  | } | 
|  | { | 
|  | std::string CompletionRangeTest(Code + "fun(/*[[fo^]]"); | 
|  | auto Results = completions(CompletionRangeTest); | 
|  | EXPECT_THAT(Results.CompletionRange, | 
|  | llvm::ValueIs(Annotations(CompletionRangeTest).range())); | 
|  | EXPECT_THAT( | 
|  | Results.Completions, | 
|  | testing::Each( | 
|  | AllOf(replacesRange(Annotations(CompletionRangeTest).range()), | 
|  | origin(SymbolOrigin::AST), kind(CompletionItemKind::Text)))); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, Concepts) { | 
|  | Annotations Code(R"cpp( | 
|  | template<class T> | 
|  | concept A = sizeof(T) <= 8; | 
|  |  | 
|  | template<$tparam^A U> | 
|  | int foo(); | 
|  |  | 
|  | template<typename T> | 
|  | int bar(T t) requires $expr^A<int>; | 
|  |  | 
|  | template<class T> | 
|  | concept b = $expr^A && $expr^sizeof(T) % 2 == 0 || $expr^A && sizeof(T) == 1; | 
|  |  | 
|  | $toplevel^A auto i = 19; | 
|  |  | 
|  | template<$toplevel^A auto i> void constrainedNTTP(); | 
|  |  | 
|  | // FIXME: The first parameter should be dropped in this case. | 
|  | void abbreviated($expr^A auto x) {} | 
|  | )cpp"); | 
|  | TestTU TU; | 
|  | TU.Code = Code.code().str(); | 
|  | TU.ExtraArgs = {"-std=c++20"}; | 
|  |  | 
|  | auto Sym = conceptSym("same_as"); | 
|  | Sym.Signature = "<typename Tp, typename Up>"; | 
|  | Sym.CompletionSnippetSuffix = "<${1:typename Tp}, ${2:typename Up}>"; | 
|  | std::vector<Symbol> Syms = {Sym}; | 
|  | for (auto P : Code.points("tparam")) { | 
|  | ASSERT_THAT( | 
|  | completions(TU, P, Syms).Completions, | 
|  | AllOf(Contains(AllOf(named("A"), signature(""), snippetSuffix(""))), | 
|  | Contains(AllOf(named("same_as"), signature("<typename Up>"), | 
|  | snippetSuffix("<${2:typename Up}>"))), | 
|  | Contains(named("class")), Contains(named("typename")))) | 
|  | << "Completing template parameter at position " << P; | 
|  | } | 
|  |  | 
|  | for (auto P : Code.points("toplevel")) { | 
|  | EXPECT_THAT( | 
|  | completions(TU, P, Syms).Completions, | 
|  | AllOf(Contains(AllOf(named("A"), signature(""), snippetSuffix(""))), | 
|  | Contains(AllOf(named("same_as"), signature("<typename Up>"), | 
|  | snippetSuffix("<${2:typename Up}>"))))) | 
|  | << "Completing 'requires' expression at position " << P; | 
|  | } | 
|  |  | 
|  | for (auto P : Code.points("expr")) { | 
|  | EXPECT_THAT( | 
|  | completions(TU, P, Syms).Completions, | 
|  | AllOf(Contains(AllOf(named("A"), signature("<class T>"), | 
|  | snippetSuffix("<${1:class T}>"))), | 
|  | Contains(AllOf( | 
|  | named("same_as"), signature("<typename Tp, typename Up>"), | 
|  | snippetSuffix("<${1:typename Tp}, ${2:typename Up}>"))))) | 
|  | << "Completing 'requires' expression at position " << P; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelp, DocFormat) { | 
|  | Annotations Code(R"cpp( | 
|  | // Comment `with` markup. | 
|  | void foo(int); | 
|  | void bar() { foo(^); } | 
|  | )cpp"); | 
|  | for (auto DocumentationFormat : | 
|  | {MarkupKind::PlainText, MarkupKind::Markdown}) { | 
|  | auto Sigs = signatures(Code.code(), Code.point(), /*IndexSymbols=*/{}, | 
|  | DocumentationFormat); | 
|  | ASSERT_EQ(Sigs.signatures.size(), 1U); | 
|  | EXPECT_EQ(Sigs.signatures[0].documentation.kind, DocumentationFormat); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(SignatureHelp, TemplateArguments) { | 
|  | std::string Top = R"cpp( | 
|  | template <typename T, int> bool foo(char); | 
|  | template <int I, int> bool foo(float); | 
|  | )cpp"; | 
|  |  | 
|  | auto First = signatures(Top + "bool x = foo<^"); | 
|  | EXPECT_THAT( | 
|  | First.signatures, | 
|  | UnorderedElementsAre(sig("foo<[[typename T]], [[int]]>() -> bool"), | 
|  | sig("foo<[[int I]], [[int]]>() -> bool"))); | 
|  | EXPECT_EQ(First.activeParameter, 0); | 
|  |  | 
|  | auto Second = signatures(Top + "bool x = foo<1, ^"); | 
|  | EXPECT_THAT(Second.signatures, | 
|  | ElementsAre(sig("foo<[[int I]], [[int]]>() -> bool"))); | 
|  | EXPECT_EQ(Second.activeParameter, 1); | 
|  | } | 
|  |  | 
|  | TEST(CompletionTest, DoNotCrash) { | 
|  | llvm::StringLiteral Cases[] = { | 
|  | R"cpp( | 
|  | template <typename = int> struct Foo {}; | 
|  | auto a = [x(3)](Foo<^>){}; | 
|  | )cpp", | 
|  | }; | 
|  | for (auto Case : Cases) { | 
|  | SCOPED_TRACE(Case); | 
|  | auto Completions = completions(Case); | 
|  | } | 
|  | } | 
|  | TEST(CompletionTest, PreambleFromDifferentTarget) { | 
|  | constexpr std::string_view PreambleTarget = "x86_64"; | 
|  | constexpr std::string_view Contents = | 
|  | "int foo(int); int num; int num2 = foo(n^"; | 
|  |  | 
|  | Annotations Test(Contents); | 
|  | auto TU = TestTU::withCode(Test.code()); | 
|  | TU.ExtraArgs.emplace_back("-target"); | 
|  | TU.ExtraArgs.emplace_back(PreambleTarget); | 
|  | auto Preamble = TU.preamble(); | 
|  | ASSERT_TRUE(Preamble); | 
|  | // Switch target to wasm. | 
|  | TU.ExtraArgs.pop_back(); | 
|  | TU.ExtraArgs.emplace_back("wasm32"); | 
|  |  | 
|  | MockFS FS; | 
|  | auto Inputs = TU.inputs(FS); | 
|  | auto Result = codeComplete(testPath(TU.Filename), Test.point(), | 
|  | Preamble.get(), Inputs, {}); | 
|  | auto Signatures = | 
|  | signatureHelp(testPath(TU.Filename), Test.point(), *Preamble, Inputs, {}); | 
|  |  | 
|  | // Make sure we don't crash. | 
|  | EXPECT_THAT(Result.Completions, Not(testing::IsEmpty())); | 
|  | EXPECT_THAT(Signatures.signatures, Not(testing::IsEmpty())); | 
|  | } | 
|  | } // namespace | 
|  | } // namespace clangd | 
|  | } // namespace clang |