28#include "clang/Sema/CodeCompleteConsumer.h"
29#include "clang/Tooling/CompilationDatabase.h"
30#include "llvm/ADT/StringRef.h"
31#include "llvm/Support/Error.h"
32#include "llvm/Support/Path.h"
33#include "llvm/Testing/Annotations/Annotations.h"
34#include "llvm/Testing/Support/Error.h"
35#include "llvm/Testing/Support/SupportHelpers.h"
36#include "gmock/gmock.h"
37#include "gtest/gtest.h"
38#include <condition_variable>
48using ::testing::AllOf;
49using ::testing::Contains;
50using ::testing::ElementsAre;
51using ::testing::Field;
52using ::testing::HasSubstr;
53using ::testing::IsEmpty;
55using ::testing::UnorderedElementsAre;
56using ContextKind = CodeCompletionContext::Kind;
59MATCHER_P(named, Name,
"") {
return arg.Name == Name; }
60MATCHER_P(mainFileRefs, Refs,
"") {
return arg.MainFileRefs == Refs; }
61MATCHER_P(scopeRefs, Refs,
"") {
return arg.ScopeRefsInFile == Refs; }
63 return llvm::StringRef(arg.Name).starts_with(Prefix);
65MATCHER_P(filterText, F,
"") {
return arg.FilterText == F; }
66MATCHER_P(scope, S,
"") {
return arg.Scope == S; }
67MATCHER_P(qualifier, Q,
"") {
return arg.RequiredQualifier == Q; }
69 return arg.RequiredQualifier + arg.Name + arg.Signature ==
Label;
72MATCHER_P(kind, K,
"") {
return arg.Kind == K; }
74 return arg.Documentation && arg.Documentation->asPlainText() ==
D;
76MATCHER_P(returnType, D,
"") {
return arg.ReturnType ==
D; }
78 return !arg.Includes.empty() && arg.Includes[0].Header == IncludeHeader;
80MATCHER_P(insertInclude, IncludeHeader,
"") {
81 return !arg.Includes.empty() && arg.Includes[0].Header == IncludeHeader &&
82 bool(arg.Includes[0].Insertion);
84MATCHER_P(insertIncludeText, InsertedText,
"") {
85 return !arg.Includes.empty() && arg.Includes[0].Insertion &&
86 arg.Includes[0].Insertion->newText == InsertedText;
89 return !arg.Includes.empty() && bool(arg.Includes[0].Insertion);
92MATCHER_P(origin, OriginSet,
"") {
return arg.Origin == OriginSet; }
93MATCHER_P(signature, S,
"") {
return arg.Signature == S; }
95 return arg.CompletionInsertRange ==
Range;
99Matcher<const std::vector<CodeCompletion> &> has(std::string Name) {
100 return Contains(named(std::move(Name)));
102Matcher<const std::vector<CodeCompletion> &> has(std::string Name,
104 return Contains(AllOf(named(std::move(Name)), kind(K)));
106MATCHER(isDocumented,
"") {
return arg.Documentation.has_value(); }
107MATCHER(deprecated,
"") {
return arg.Deprecated; }
109std::unique_ptr<SymbolIndex> memIndex(std::vector<Symbol>
Symbols) {
111 for (
const auto &Sym :
Symbols)
119 std::vector<Symbol> IndexSymbols = {},
120 clangd::CodeCompleteOptions Opts = {}) {
121 std::unique_ptr<SymbolIndex> OverrideIndex;
122 if (!IndexSymbols.empty()) {
123 assert(!Opts.Index &&
"both Index and IndexSymbols given!");
124 OverrideIndex = memIndex(std::move(IndexSymbols));
125 Opts.Index = OverrideIndex.get();
129 auto Inputs = TU.inputs(FS);
133 ADD_FAILURE() <<
"Couldn't build CompilerInvocation";
143CodeCompleteResult completions(llvm::StringRef
Text,
144 std::vector<Symbol> IndexSymbols = {},
145 clangd::CodeCompleteOptions Opts = {},
146 PathRef FilePath =
"foo.cpp") {
150 TU.Filename = FilePath.str();
151 return completions(TU, Test.point(), std::move(IndexSymbols),
156CodeCompleteResult completionsNoCompile(llvm::StringRef
Text,
157 std::vector<Symbol> IndexSymbols = {},
158 clangd::CodeCompleteOptions Opts = {},
159 PathRef FilePath =
"foo.cpp") {
160 std::unique_ptr<SymbolIndex> OverrideIndex;
161 if (!IndexSymbols.empty()) {
162 assert(!Opts.Index &&
"both Index and IndexSymbols given!");
163 OverrideIndex = memIndex(std::move(IndexSymbols));
164 Opts.Index = OverrideIndex.get();
169 ParseInputs ParseInput{tooling::CompileCommand(), &FS, Test.code().str()};
170 return codeComplete(FilePath, Test.point(),
nullptr, ParseInput,
179#if CLANGD_DECISION_FOREST
180TEST(DecisionForestRankingModel, NameMatchSanityTest) {
181 clangd::CodeCompleteOptions Opts;
182 Opts.RankingModel = CodeCompleteOptions::DecisionForest;
183 auto Results = completions(
187 int AlphaBetaGamma();
189int func() { MemberAccess().ABG^ }
192 EXPECT_THAT(Results.Completions,
193 ElementsAre(named(
"ABG"), named(
"AlphaBetaGamma")));
196TEST(DecisionForestRankingModel, ReferencesAffectRanking) {
197 clangd::CodeCompleteOptions Opts;
198 Opts.RankingModel = CodeCompleteOptions::DecisionForest;
199 constexpr int NumReferences = 100000;
201 completions(
"int main() { clang^ }",
202 {
ns(
"clangA"), withReferences(NumReferences,
func(
"clangD"))},
205 ElementsAre(named(
"clangD"), named(
"clangA")));
207 completions(
"int main() { clang^ }",
208 {withReferences(NumReferences,
ns(
"clangA")),
func(
"clangD")},
211 ElementsAre(named(
"clangA"), named(
"clangD")));
215TEST(DecisionForestRankingModel, DecisionForestScorerCallbackTest) {
216 clangd::CodeCompleteOptions Opts;
217 constexpr float MagicNumber = 1234.5678f;
218 Opts.RankingModel = CodeCompleteOptions::DecisionForest;
222 Scores.
Total = MagicNumber;
223 Scores.ExcludingName = MagicNumber;
226 llvm::StringRef Code =
"int func() { int xyz; xy^ }";
227 auto Results = completions(Code,
229 ASSERT_EQ(Results.Completions.size(), 1u);
230 EXPECT_EQ(Results.Completions[0].Score.Total, MagicNumber);
231 EXPECT_EQ(Results.Completions[0].Score.ExcludingName, MagicNumber);
234 Opts.RankingModel = CodeCompleteOptions::Heuristics;
235 Results = completions(Code,
237 ASSERT_EQ(Results.Completions.size(), 1u);
238 EXPECT_NE(Results.Completions[0].Score.Total, MagicNumber);
239 EXPECT_NE(Results.Completions[0].Score.ExcludingName, MagicNumber);
242TEST(CompletionTest, Limit) {
243 clangd::CodeCompleteOptions Opts;
245 auto Results = completions(R
"cpp(
246struct ClassWithMembers {
252int main() { ClassWithMembers().^ }
256 EXPECT_TRUE(Results.HasMore);
257 EXPECT_THAT(Results.Completions, ElementsAre(named(
"AAA"), named(
"BBB")));
260TEST(CompletionTest, Filter) {
261 std::string Body = R
"cpp(
272 EXPECT_THAT(completions(Body +
"int main() { S().Foba^ }").Completions,
273 AllOf(has(
"FooBar"), has(
"FooBaz"), Not(has(
"Qux"))));
277 Sym.SymInfo.Kind = index::SymbolKind::Macro;
279 completions(Body +
"int main() { C^ }", {Sym}).Completions,
280 AllOf(has(
"Car"), Not(has(
"MotorCar")), Not(has(
"MotorCarIndex"))));
281 EXPECT_THAT(completions(Body +
"int main() { M^ }", {Sym}).Completions,
282 AllOf(has(
"MotorCar"), has(
"MotorCarIndex")));
285void testAfterDotCompletion(clangd::CodeCompleteOptions Opts) {
286 auto Results = completions(
292 // Make sure this is not in preamble.
295 struct GlobalClass {};
297 struct ClassWithMembers {
307 struct LocalClass {};
315 {
cls(
"IndexClass"),
var(
"index_var"),
func(
"index_func")}, Opts);
317 EXPECT_TRUE(Results.RanParser);
320 EXPECT_THAT(Results.Completions,
321 AllOf(has(
"method"), has(
"field"), Not(has(
"ClassWithMembers")),
322 Not(has(
"operator=")), Not(has(
"~ClassWithMembers"))));
323 EXPECT_IFF(Opts.IncludeIneligibleResults, Results.Completions,
324 has(
"private_field"));
328 Not(AnyOf(has(
"global_var"), has(
"index_var"), has(
"global_func"),
329 has(
"global_func()"), has(
"index_func"), has(
"GlobalClass"),
330 has(
"IndexClass"), has(
"MACRO"), has(
"LocalClass"))));
333 EXPECT_THAT(Results.Completions,
336 EXPECT_THAT(Results.Completions, Contains(isDocumented()));
339void testGlobalScopeCompletion(clangd::CodeCompleteOptions Opts) {
340 auto Results = completions(
345 // Make sure this is not in preamble.
348 struct GlobalClass {};
350 struct ClassWithMembers {
356 struct LocalClass {};
364 {
cls(
"IndexClass"),
var(
"index_var"),
func(
"index_func")}, Opts);
366 EXPECT_TRUE(Results.RanParser);
368 EXPECT_THAT(Results.Completions,
369 Not(AnyOf(has(
"method"), has(
"method()"), has(
"field"))));
371 EXPECT_THAT(Results.Completions,
372 AllOf(has(
"global_var"), has(
"index_var"), has(
"global_func"),
374 has(
"GlobalClass"), has(
"IndexClass")));
376 EXPECT_THAT(Results.Completions, has(
"MACRO"));
378 EXPECT_THAT(Results.Completions,
379 AllOf(has(
"local_var"), has(
"LocalClass"),
382 EXPECT_THAT(Results.Completions, Contains(isDocumented()));
385TEST(CompletionTest, CompletionOptions) {
386 auto Test = [&](
const clangd::CodeCompleteOptions &Opts) {
387 testAfterDotCompletion(Opts);
388 testGlobalScopeCompletion(Opts);
392 &clangd::CodeCompleteOptions::IncludeIneligibleResults,
397 for (
auto &F : Flags) {
398 clangd::CodeCompleteOptions O;
404TEST(CompletionTest, Accessible) {
408 protected: void prot();
409 private: void priv();
411 void Foo::pub() { this->^ }
414 AllOf(has("priv"), has(
"prot"), has(
"pub")));
416 auto External = completions(R
"cpp(
419 protected: void prot();
420 private: void priv();
427 EXPECT_THAT(External.Completions,
428 AllOf(has("pub"), Not(has(
"prot")), Not(has(
"priv"))));
430 auto Results = completions(R
"cpp(
433 protected: void prot();
434 private: void priv();
436 struct Bar : public Foo {
437 private: using Foo::pub;
444 EXPECT_THAT(Results.Completions,
445 AllOf(Not(has("priv")), Not(has(
"prot")), Not(has(
"pub"))));
448TEST(CompletionTest, Qualifiers) {
449 auto Results = completions(R
"cpp(
451 public: int foo() const;
454 class Bar : public Foo {
457 void test() { Bar().^ }
459 EXPECT_THAT(Results.Completions,
460 Contains(AllOf(qualifier(""), named(
"bar"))));
462 EXPECT_THAT(Results.Completions,
463 Not(Contains(AllOf(qualifier(
"Foo::"), named(
"foo")))));
465 EXPECT_THAT(Results.Completions,
466 Not(Contains(AllOf(qualifier(
""), named(
"foo")))));
470TEST(CompletionTest, QualificationWithInlineNamespace) {
471 auto Results = completions(R
"cpp(
472 namespace a { inline namespace b {} }
473 using namespace a::b;
477 EXPECT_THAT(Results.Completions,
478 UnorderedElementsAre(AllOf(qualifier(
"a::"), named(
"Foo"))));
481TEST(CompletionTest, InjectedTypename) {
483 EXPECT_THAT(completions(
"struct X{}; void foo(){ X().^ }").Completions,
485 EXPECT_THAT(completions(
"struct X{ void foo(){ this->^ } };").Completions,
488 EXPECT_THAT(completions(
"struct X{ void foo(){ ^ } };").Completions,
491 completions(
"struct Y{}; struct X:Y{ void foo(){ ^ } };").Completions,
495 "template<class> struct Y{}; struct X:Y<int>{ void foo(){ ^ } };")
499 EXPECT_THAT(completions(
"struct X{}; void foo(){ X::^ }").Completions,
503TEST(CompletionTest, SkipInjectedWhenUnqualified) {
504 EXPECT_THAT(completions(
"struct X { void f() { X^ }};").Completions,
505 ElementsAre(named(
"X"), named(
"~X")));
508TEST(CompletionTest, Snippets) {
509 clangd::CodeCompleteOptions Opts;
510 auto Results = completions(
514 int f(int i, const float f) const;
525 snippetSuffix(
"(${1:int i}, ${2:const float f})")));
528TEST(CompletionTest, HeuristicsForMemberFunctionCompletion) {
529 clangd::CodeCompleteOptions Opts;
530 Opts.EnableSnippets =
true;
534 static int staticMethod(int name);
535 int method(int name) const;
536 template <typename T, typename U, typename V = int>
537 T generic(U nameU, V nameV);
538 template <typename T, int U>
539 static T staticGeneric();
541 this->$canBeCallNoStatic^
547 int Foo::$isDefinition^ {
551 int i = Foo::$canBeCallStaticOnly^
554 struct Derived : Foo {
566 f.$canBeCallNoStatic^
567 ; // Prevent parsing as 'f.f'
574 d.Derived::$canBeCall^
582 ; // Prevent parsing as 'f.f'
589 d.Derived::$canBeCall^
594 for (
const auto &P : Code.points(
"canNotBeCall")) {
595 auto Results = completions(TU, P, {}, Opts);
596 EXPECT_THAT(Results.Completions,
597 Contains(AllOf(named(
"method"), signature(
"(int name) const"),
598 snippetSuffix(
""))));
602 Contains(AllOf(named(
"generic"),
603 signature(
"<typename T, typename U>(U nameU, V nameV)"),
604 snippetSuffix(
"<${1:typename T}, ${2:typename U}>"))));
605 EXPECT_THAT(Results.Completions,
606 Contains(AllOf(named(
"staticMethod"), signature(
"(int name)"),
607 snippetSuffix(
""))));
608 EXPECT_THAT(Results.Completions,
610 named(
"staticGeneric"), signature(
"<typename T, int U>()"),
611 snippetSuffix(
"<${1:typename T}, ${2:int U}>"))));
614 for (
const auto &P : Code.points(
"canBeCall")) {
615 auto Results = completions(TU, P, {}, Opts);
616 EXPECT_THAT(Results.Completions,
617 Contains(AllOf(named(
"method"), signature(
"(int name) const"),
618 snippetSuffix(
"(${1:int name})"))));
622 named(
"generic"), signature(
"<typename T>(U nameU, V nameV)"),
623 snippetSuffix(
"<${1:typename T}>(${2:U nameU}, ${3:V nameV})"))));
624 EXPECT_THAT(Results.Completions,
625 Contains(AllOf(named(
"staticMethod"), signature(
"(int name)"),
626 snippetSuffix(
"(${1:int name})"))));
627 EXPECT_THAT(Results.Completions,
629 named(
"staticGeneric"), signature(
"<typename T, int U>()"),
630 snippetSuffix(
"<${1:typename T}, ${2:int U}>()"))));
633 for (
const auto &P : Code.points(
"canBeCallNoStatic")) {
634 auto Results = completions(TU, P, {}, Opts);
635 EXPECT_THAT(Results.Completions,
636 Contains(AllOf(named(
"method"), signature(
"(int name) const"),
637 snippetSuffix(
"(${1:int name})"))));
641 named(
"generic"), signature(
"<typename T>(U nameU, V nameV)"),
642 snippetSuffix(
"<${1:typename T}>(${2:U nameU}, ${3:V nameV})"))));
645 for (
const auto &P : Code.points(
"canBeCallStaticOnly")) {
646 auto Results = completions(TU, P, {}, Opts);
647 EXPECT_THAT(Results.Completions,
648 Contains(AllOf(named(
"method"), signature(
"(int name) const"),
649 snippetSuffix(
""))));
652 Contains(AllOf(named(
"generic"),
653 signature(
"<typename T, typename U>(U nameU, V nameV)"),
654 snippetSuffix(
"<${1:typename T}, ${2:typename U}>"))));
655 EXPECT_THAT(Results.Completions,
656 Contains(AllOf(named(
"staticMethod"), signature(
"(int name)"),
657 snippetSuffix(
"(${1:int name})"))));
658 EXPECT_THAT(Results.Completions,
660 named(
"staticGeneric"), signature(
"<typename T, int U>()"),
661 snippetSuffix(
"<${1:typename T}, ${2:int U}>()"))));
664 for (
const auto &P : Code.points(
"isDefinition")) {
665 auto Results = completions(TU, P, {}, Opts);
667 EXPECT_THAT(Results.Completions,
668 Contains(AllOf(named(
"method"), signature(
"(int name) const"),
669 snippetSuffix(
"(int name) const"))));
672 Contains(AllOf(named(
"generic"),
673 signature(
"<typename T, typename U>(U nameU, V nameV)"),
674 snippetSuffix(
"(U nameU, V nameV)"))));
675 EXPECT_THAT(Results.Completions,
676 Contains(AllOf(named(
"staticMethod"), signature(
"(int name)"),
677 snippetSuffix(
"(int name)"))));
678 EXPECT_THAT(Results.Completions,
679 Contains(AllOf(named(
"staticGeneric"),
680 signature(
"<typename T, int U>()"),
681 snippetSuffix(
"()"))));
685TEST(CompletionTest, PrivateMemberDefinition) {
686 clangd::CodeCompleteOptions Opts;
687 Opts.EnableSnippets =
true;
688 auto Results = completions(
691 int func(int a, int b);
696 EXPECT_THAT(Results.Completions,
697 Contains(AllOf(named(
"func"), signature(
"(int a, int b)"),
698 snippetSuffix(
"(int a, int b)"))));
701TEST(CompletionTest, DefaultArgsWithValues) {
702 clangd::CodeCompleteOptions Opts;
703 Opts.EnableSnippets =
true;
704 auto Results = completions(
710 void foo(int x = 42, int y = 0, Arg arg = Arg(42, 0));
715 EXPECT_THAT(Results.Completions,
718 signature(
"(int x = 42, int y = 0, Arg arg = Arg(42, 0))"),
719 snippetSuffix(
"(int x, int y, Arg arg)"))));
722TEST(CompletionTest, NoSnippetsInUsings) {
723 clangd::CodeCompleteOptions Opts;
724 Opts.EnableSnippets =
true;
725 auto Results = completions(
728 int func(int a, int b);
734 EXPECT_THAT(Results.Completions,
735 ElementsAre(AllOf(named(
"func"), labeled(
"func(int a, int b)"),
736 snippetSuffix(
""))));
739 auto Func =
func(
"ns::func");
740 Func.CompletionSnippetSuffix =
"(${1:int a}, ${2: int b})";
741 Func.Signature =
"(int a, int b)";
742 Func.ReturnType =
"void";
744 Results = completions(R
"cpp(
749 EXPECT_THAT(Results.Completions,
750 ElementsAre(AllOf(named(
"func"), labeled(
"func(int a, int b)"),
751 snippetSuffix(
""))));
754 Opts.AllScopes =
true;
755 Results = completions(R
"cpp(
759 EXPECT_THAT(Results.Completions,
760 Contains(AllOf(named(
"func"), labeled(
"ns::func(int a, int b)"),
761 snippetSuffix(
""))));
764TEST(CompletionTest, Kinds) {
765 auto Results = completions(
770 // make sure MACRO is not included in preamble.
774 {func("indexFunction"),
var(
"indexVariable"),
cls(
"indexClass"),
775 macro(
"indexObjMacro"),
macro(
"indexFuncMacro",
"(x, y)")});
776 EXPECT_THAT(Results.Completions,
788 Results = completions(
"nam^");
789 EXPECT_THAT(Results.Completions,
793 Results = completions(
807 Results = completions(
809 template <class T> struct complete_class {};
810 template <class T> void complete_function();
811 template <class T> using complete_type_alias = int;
812 template <class T> int complete_variable = 10;
815 template <class T> static int complete_static_member = 10;
817 static auto x = complete_^
822 UnorderedElementsAre(
825 AllOf(named(
"complete_type_alias"),
828 AllOf(named(
"complete_static_member"),
831 Results = completions(
843TEST(CompletionTest, NoDuplicates) {
844 auto Results = completions(
856 EXPECT_THAT(Results.Completions, ElementsAre(named(
"Adapter")));
859TEST(CompletionTest, ScopedNoIndex) {
860 auto Results = completions(
862 namespace fake { int BigBang, Babble, Box; };
863 int main() { fake::ba^ }
866 EXPECT_THAT(Results.Completions,
867 ElementsAre(named(
"Babble"), named(
"BigBang")));
870TEST(CompletionTest, Scoped) {
871 auto Results = completions(
873 namespace fake { int Babble, Box; };
874 int main() { fake::ba^ }
876 {var("fake::BigBang")});
877 EXPECT_THAT(Results.Completions,
878 ElementsAre(named(
"Babble"), named(
"BigBang")));
881TEST(CompletionTest, ScopedWithFilter) {
882 auto Results = completions(
886 {cls("ns::XYZ"),
func(
"ns::foo")});
887 EXPECT_THAT(Results.Completions, UnorderedElementsAre(named(
"XYZ")));
890TEST(CompletionTest, ReferencesAffectRanking) {
891 EXPECT_THAT(completions(
"int main() { abs^ }", {
func(
"absA"),
func(
"absB")})
894 EXPECT_THAT(completions(
"int main() { abs^ }",
895 {
func(
"absA"), withReferences(1000,
func(
"absB"))})
900TEST(CompletionTest, ContextWords) {
901 auto Results = completions(R
"cpp(
902 enum class Color { RED, YELLOW, BLUE };
904 // (blank lines so the definition above isn't "context")
906 // "It was a yellow car," he said. "Big yellow car, new."
907 auto Finish = Color::^
911 ASSERT_THAT(Results.Completions,
915TEST(CompletionTest, GlobalQualified) {
916 auto Results = completions(
921 EXPECT_THAT(Results.Completions,
926TEST(CompletionTest, FullyQualified) {
927 auto Results = completions(
929 namespace ns { void bar(); }
933 EXPECT_THAT(Results.Completions,
938TEST(CompletionTest, SemaIndexMerge) {
939 auto Results = completions(
941 namespace ns { int local; void both(); }
944 {func("ns::both"),
cls(
"ns::Index")});
946 EXPECT_THAT(Results.Completions,
947 UnorderedElementsAre(
954TEST(CompletionTest, SemaIndexMergeWithLimit) {
955 clangd::CodeCompleteOptions Opts;
957 auto Results = completions(
959 namespace ns { int local; void both(); }
962 {func("ns::both"),
cls(
"ns::Index")}, Opts);
963 EXPECT_EQ(Results.Completions.size(), Opts.Limit);
964 EXPECT_TRUE(Results.HasMore);
967TEST(CompletionTest, IncludeInsertionPreprocessorIntegrationTests) {
970 TU.AdditionalFiles[
"sub/bar.h"] =
"";
974 Sym.CanonicalDeclaration.FileURI = BarURI.c_str();
978 TU.Code = Test.code().str();
979 auto Results = completions(TU, Test.point(), {Sym});
980 EXPECT_THAT(Results.Completions,
981 ElementsAre(AllOf(named(
"X"), insertInclude(
"\"bar.h\""))));
985 Results = completions(TU, Test.point(), {Sym}, NoInsertion);
986 EXPECT_THAT(Results.Completions,
987 ElementsAre(AllOf(named(
"X"), Not(insertInclude()))));
990 #include "sub/bar.h" // not shortest, so should only match resolved.
993 TU.Code = Test.code().str();
994 Results = completions(TU, Test.point(), {Sym});
995 EXPECT_THAT(Results.Completions, ElementsAre(AllOf(named("X"), labeled(
"X"),
996 Not(insertInclude()))));
999TEST(CompletionTest, NoIncludeInsertionWhenDeclFoundInFile) {
1002 std::string BarHeader =
testPath(
"bar.h");
1004 SymX.CanonicalDeclaration.FileURI = BarURI.c_str();
1005 SymY.CanonicalDeclaration.FileURI = BarURI.c_str();
1009 auto Results = completions(R
"cpp(
1014 int main() { ns::^ }
1017 EXPECT_THAT(Results.Completions,
1018 ElementsAre(AllOf(named("X"), Not(insertInclude())),
1019 AllOf(named(
"Y"), Not(insertInclude()))));
1022TEST(CompletionTest, IncludeInsertionRespectsQuotedAngledConfig) {
1025 TU.AdditionalFiles[
"sub/bar.h"] =
"";
1029 Sym.CanonicalDeclaration.FileURI = BarURI.c_str();
1032 TU.Code = Test.code().str();
1033 auto Results = completions(TU, Test.point(), {Sym});
1035 EXPECT_THAT(Results.Completions,
1036 ElementsAre(AllOf(named(
"X"), insertInclude(
"\"bar.h\""))));
1039 C.Style.AngledHeaders.push_back(
1040 [](
auto header) {
return header.contains(
"bar.h"); });
1042 Results = completions(TU, Test.point(), {Sym});
1043 EXPECT_THAT(Results.Completions,
1044 ElementsAre(AllOf(named(
"X"), insertInclude(
"<bar.h>"))));
1048 C.Style.QuotedHeaders.push_back(
1049 [](
auto header) {
return header.contains(
"bar.h"); });
1051 Results = completions(TU, Test.point(), {Sym});
1052 EXPECT_THAT(Results.Completions,
1053 ElementsAre(AllOf(named(
"X"), insertInclude(
"\"bar.h\""))));
1057TEST(CompletionTest, IndexSuppressesPreambleCompletions) {
1060 namespace ns { int local; }
1062 void f2() { ns::preamble().$2^; }
1065 TU.AdditionalFiles[
"bar.h"] =
1066 R
"cpp(namespace ns { struct preamble { int member; }; })cpp";
1068 clangd::CodeCompleteOptions Opts = {};
1069 auto I = memIndex({
var(
"ns::index")});
1070 Opts.Index = I.get();
1071 auto WithIndex = completions(TU, Test.point(), {}, Opts);
1072 EXPECT_THAT(WithIndex.Completions,
1073 UnorderedElementsAre(named(
"local"), named(
"index")));
1074 auto ClassFromPreamble = completions(TU, Test.point(
"2"), {}, Opts);
1075 EXPECT_THAT(ClassFromPreamble.Completions, Contains(named(
"member")));
1077 Opts.Index =
nullptr;
1078 auto WithoutIndex = completions(TU, Test.point(), {}, Opts);
1079 EXPECT_THAT(WithoutIndex.Completions,
1080 UnorderedElementsAre(named(
"local"), named(
"preamble")));
1087TEST(CompletionTest, CompletionInPreamble) {
1088 auto Results = completions(R
"cpp(
1096 EXPECT_THAT(Results, ElementsAre(named("ifndef")));
1099TEST(CompletionTest, CompletionRecoveryASTType) {
1100 auto Results = completions(R
"cpp(
1101 struct S { int member; };
1104 // No overload matches, but we have recovery-expr with the correct type.
1108 EXPECT_THAT(Results, ElementsAre(named("member")));
1111TEST(CompletionTest, DynamicIndexIncludeInsertion) {
1115 Opts.BuildDynamicSymbolIndex =
true;
1118 FS.Files[
testPath(
"foo_header.h")] = R
"cpp(
1125 const std::string FileContent(R
"cpp(
1126 #include "foo_header.h"
1131 Server.addDocument(testPath("foo_impl.cpp"), FileContent);
1133 ASSERT_TRUE(Server.blockUntilIdleForTest());
1142 ElementsAre(AllOf(named(
"Foo"), hasInclude(
"\"foo_header.h\""),
1146TEST(CompletionTest, DynamicIndexMultiFile) {
1150 Opts.BuildDynamicSymbolIndex =
true;
1153 FS.Files[
testPath(
"foo.h")] = R
"cpp(
1154 namespace ns { class XYZ {}; void foo(int x) {} }
1177 EXPECT_THAT(Results.Completions,
1179 doc(
"Doooc"), returnType(
"void"))));
1182TEST(CompletionTest, Documentation) {
1183 auto Results = completions(
1185 // Non-doxygen comment.
1186 __attribute__((annotate("custom_annotation"))) int foo();
1198 Results.Completions,
1201 doc(
"Annotation: custom_annotation\n\nNon-doxygen comment."))));
1203 Results.Completions,
1204 Contains(AllOf(named(
"bar"), doc(
"Doxygen comment.\n\\param int a"))));
1205 EXPECT_THAT(Results.Completions,
1206 Contains(AllOf(named(
"baz"), doc(
"Multi-line block comment"))));
1209TEST(CompletionTest, CommentsFromSystemHeaders) {
1214 Opts.BuildDynamicSymbolIndex =
true;
1218 FS.Files[
testPath(
"foo.h")] = R
"cpp(
1219 #pragma GCC system_header
1221 // This comment should be retained!
1236 Contains(AllOf(named(
"foo"), doc(
"This comment should be retained!"))));
1239TEST(CompletionTest, CommentsOnMembersFromHeader) {
1244 Opts.BuildDynamicSymbolIndex =
true;
1248 FS.Files[
testPath(
"foo.h")] = R
"cpp(
1257 template <typename T>
1280 Contains(AllOf(named(
"gamma"), doc(
"This is a member field."))));
1283 Contains(AllOf(named(
"delta"), doc(
"This is a member function."))));
1289 Contains(AllOf(named(
"omega")
1293 Contains(AllOf(named(
"epsilon"),
1294 doc(
"This is a member function inside a template."))));
1297TEST(CompletionTest, CommentsOnMembersFromHeaderOverloadBundling) {
1298 using testing::AnyOf;
1303 Opts.BuildDynamicSymbolIndex =
true;
1307 FS.Files[
testPath(
"foo.h")] = R
"cpp(
1315 void epsilon(long l);
1318 void epsilon(int i);
1329 clangd::CodeCompleteOptions CCOpts;
1330 CCOpts.BundleOverloads = true;
1336 Contains(AllOf(named(
"epsilon"), doc(
"This one has a comment."))));
1338 Contains(AllOf(named(
"delta"), AnyOf(doc(
"bool overload."),
1339 doc(
"int overload.")))));
1342TEST(CompletionTest, GlobalCompletionFiltering) {
1351 auto Results = completions(R
"(// void f() {
1355 EXPECT_THAT(Results.Completions, IsEmpty());
1358TEST(CodeCompleteTest, DisableTypoCorrection) {
1359 auto Results = completions(R
"cpp(
1360 namespace clang { int v; }
1361 void f() { clangd::^
1363 EXPECT_TRUE(Results.Completions.empty());
1366TEST(CodeCompleteTest, NoColonColonAtTheEnd) {
1367 auto Results = completions(R
"cpp(
1374 EXPECT_THAT(Results.Completions, Contains(labeled("clang")));
1375 EXPECT_THAT(Results.Completions, Not(Contains(labeled(
"clang::"))));
1378TEST(CompletionTests, EmptySnippetDoesNotCrash) {
1380 auto Results = completions(R
"cpp(
1382 auto w = [&](auto &&f) { return f(f); };
1383 auto f = w([&](auto &&f) {
1384 return [&](auto &&n) {
1388 return n * ^(f)(n - 1);
1395TEST(CompletionTest, Issue1427Crash) {
1401 Opts.MainFileSignals = &MainFileSignals;
1410TEST(CompletionTest, BacktrackCrashes) {
1412 auto Results = completions(R
"cpp(
1414 struct FooBarBaz {};
1420 EXPECT_THAT(Results.Completions, ElementsAre(labeled("FooBarBaz")));
1424 struct FooBarBaz {};
1426 if (FooBarBaz * x^) {}
1431TEST(CompletionTest, CompleteInMacroWithStringification) {
1432 auto Results = completions(R
"cpp(
1433void f(const char *, int x);
1434#define F(x) f(#x, x)
1441int f(int input_num) {
1446 EXPECT_THAT(Results.Completions,
1447 UnorderedElementsAre(named("X"), named(
"Y")));
1450TEST(CompletionTest, CompleteInMacroAndNamespaceWithStringification) {
1451 auto Results = completions(R
"cpp(
1452void f(const char *, int x);
1453#define F(x) f(#x, x)
1458int f(int input_num) {
1464 EXPECT_THAT(Results.Completions, Contains(named("X")));
1467TEST(CompletionTest, IgnoreCompleteInExcludedPPBranchWithRecoveryContext) {
1468 auto Results = completions(R
"cpp(
1469 int bar(int param_in_bar) {
1472 int foo(int param_in_foo) {
1474 // In recovery mode, "param_in_foo" will also be suggested among many other
1475 // unrelated symbols; however, this is really a special case where this works.
1476 // If the #if block is outside of the function, "param_in_foo" is still
1477 // suggested, but "bar" and "foo" are missing. So the recovery mode doesn't
1478 // really provide useful results in excluded branches.
1484 EXPECT_TRUE(Results.Completions.empty());
1487TEST(CompletionTest, DefaultArgs) {
1488 clangd::CodeCompleteOptions Opts;
1491 int Y(int A, int B = 0);
1492 int Z(int A, int B = 0, int C = 0, int D = 0);
1494 EXPECT_THAT(completions(Context + "int y = X^", {}, Opts).Completions,
1495 UnorderedElementsAre(labeled(
"X(int A = 0)")));
1496 EXPECT_THAT(completions(
Context +
"int y = Y^", {}, Opts).Completions,
1497 UnorderedElementsAre(AllOf(labeled(
"Y(int A, int B = 0)"),
1498 snippetSuffix(
"(${1:int A})"))));
1499 EXPECT_THAT(completions(
Context +
"int y = Z^", {}, Opts).Completions,
1500 UnorderedElementsAre(
1501 AllOf(labeled(
"Z(int A, int B = 0, int C = 0, int D = 0)"),
1502 snippetSuffix(
"(${1:int A})"))));
1505TEST(CompletionTest, NoCrashWithTemplateParamsAndPreferredTypes) {
1506 auto Completions = completions(R
"cpp(
1507template <template <class> class TT> int foo() {
1512 EXPECT_THAT(Completions, Contains(named("TT")));
1515TEST(CompletionTest, NestedTemplateHeuristics) {
1516 auto Completions = completions(R
"cpp(
1517struct Plain { int xxx; };
1518template <typename T> class Templ { Plain ppp; };
1519template <typename T> void foo(Templ<T> &t) {
1520 // Formally ppp has DependentTy, because Templ may be specialized.
1521 // However we sholud be able to see into it using the primary template.
1526 EXPECT_THAT(Completions, Contains(named("xxx")));
1529TEST(CompletionTest, RecordCCResultCallback) {
1530 std::vector<CodeCompletion> RecordedCompletions;
1532 Opts.RecordCCResult = [&RecordedCompletions](
const CodeCompletion &CC,
1536 RecordedCompletions.push_back(CC);
1539 completions(
"int xy1, xy2; int a = xy^", {}, Opts);
1540 EXPECT_THAT(RecordedCompletions,
1541 UnorderedElementsAre(named(
"xy1"), named(
"xy2")));
1547 unsigned MainFileRefs;
1548 unsigned ScopeRefsInFile;
1551 std::vector<Completion> RecordedCompletions;
1552 Opts.RecordCCResult = [&RecordedCompletions](
const CodeCompletion &CC,
1556 RecordedCompletions.push_back({CC.Name, R.MainFileRefs, R.ScopeRefsInFile});
1560 MainFileSignals.ReferencedSymbols[
var(
"xy2").
ID] = 1;
1561 MainFileSignals.ReferencedSymbols[
var(
"xyindex").
ID] = 10;
1562 MainFileSignals.RelatedNamespaces[
"tar::"] = 5;
1563 MainFileSignals.RelatedNamespaces[
"bar::"] = 3;
1564 Opts.MainFileSignals = &MainFileSignals;
1565 Opts.AllScopes =
true;
1575 {
var(
"xyindex"),
var(
"tar::xytar"),
var(
"bar::xybar")},
1577 EXPECT_THAT(RecordedCompletions,
1578 UnorderedElementsAre(
1579 AllOf(named(
"xy1"), mainFileRefs(3u), scopeRefs(0u)),
1580 AllOf(named(
"xy2"), mainFileRefs(1u), scopeRefs(0u)),
1581 AllOf(named(
"xyindex"), mainFileRefs(10u), scopeRefs(0u)),
1582 AllOf(named(
"xytar"), mainFileRefs(0u), scopeRefs(5u)),
1583 AllOf( named(
"xybar"),
1584 mainFileRefs(0u), scopeRefs(3u))));
1589 std::vector<Symbol> IndexSymbols = {},
1591 std::unique_ptr<SymbolIndex> Index;
1592 if (!IndexSymbols.empty())
1593 Index = memIndex(IndexSymbols);
1597 auto Inputs = TU.inputs(FS);
1598 Inputs.Index = Index.get();
1602 ADD_FAILURE() <<
"Couldn't build CompilerInvocation";
1608 ADD_FAILURE() <<
"Couldn't build Preamble";
1612 DocumentationFormat);
1616signatures(llvm::StringRef
Text, std::vector<Symbol> IndexSymbols = {},
1619 return signatures(Test.code(), Test.point(), std::move(IndexSymbols),
1620 DocumentationFormat);
1623struct ExpectedParameter {
1625 std::pair<unsigned, unsigned> Offsets;
1627llvm::raw_ostream &
operator<<(llvm::raw_ostream &OS,
1628 const ExpectedParameter &P) {
1629 return OS <<
P.Text;
1632 if (
P.size() != arg.parameters.size())
1634 for (
unsigned I = 0; I <
P.size(); ++I) {
1635 if (P[I].
Text != arg.parameters[I].labelString ||
1636 P[I].Offsets != arg.parameters[I].labelOffsets)
1641MATCHER_P(sigDoc, doc,
"") {
return arg.documentation.value == doc; }
1645Matcher<SignatureInformation> sig(llvm::StringRef AnnotatedLabel) {
1646 llvm::Annotations
A(AnnotatedLabel);
1647 std::string
Label = std::string(
A.code());
1648 std::vector<ExpectedParameter> Parameters;
1649 for (
auto Range :
A.ranges()) {
1650 Parameters.emplace_back();
1652 ExpectedParameter &
P = Parameters.back();
1657 return AllOf(sigHelpLabeled(
Label), paramsAre(Parameters));
1660TEST(SignatureHelpTest, Overloads) {
1661 auto Results = signatures(R
"cpp(
1662 void foo(int x, int y);
1663 void foo(int x, float y);
1664 void foo(float x, int y);
1665 void foo(float x, float y);
1666 void bar(int x, int y = 0);
1667 int main() { foo(^); }
1669 EXPECT_THAT(Results.signatures,
1670 UnorderedElementsAre(sig("foo([[float x]], [[float y]]) -> void"),
1671 sig(
"foo([[float x]], [[int y]]) -> void"),
1672 sig(
"foo([[int x]], [[float y]]) -> void"),
1673 sig(
"foo([[int x]], [[int y]]) -> void")));
1675 EXPECT_EQ(0, Results.activeSignature);
1676 EXPECT_EQ(0, Results.activeParameter);
1679TEST(SignatureHelpTest, FunctionPointers) {
1680 llvm::StringLiteral Tests[] = {
1683 void (*foo)(int x, int y);
1684 int main() { foo(^); }
1688 void (__stdcall *foo)(int x, int y);
1689 int main() { foo(^); }
1693 void (__attribute__(stdcall) *foo)(int x, int y);
1694 int main() { foo(^); },
1698 typedef void (*fn)(int x, int y);
1700 int main() { foo(^); }
1704 typedef void (__stdcall *fn)(int x, int y);
1706 int main() { foo(^); }
1711 void (*foo)(int x, int y);
1714 int main() { s.foo(^); }
1718 typedef void (*fn)(int x, int y);
1723 int main() { s.foo(^); }
1725 for (
auto Test : Tests)
1726 EXPECT_THAT(signatures(Test).signatures,
1727 UnorderedElementsAre(sig(
"([[int x]], [[int y]]) -> void")));
1730TEST(SignatureHelpTest, Constructors) {
1731 std::string Top = R
"cpp(
1734 S(const S &) = delete;
1738 auto CheckParenInit = [&](std::string Init) {
1739 EXPECT_THAT(signatures(Top + Init).signatures,
1740 UnorderedElementsAre(sig(
"S([[int]])")))
1743 CheckParenInit(
"S s(^);");
1744 CheckParenInit(
"auto s = S(^);");
1745 CheckParenInit(
"auto s = new S(^);");
1747 auto CheckBracedInit = [&](std::string Init) {
1748 EXPECT_THAT(signatures(Top + Init).signatures,
1749 UnorderedElementsAre(sig(
"S{[[int]]}")))
1752 CheckBracedInit(
"S s{^};");
1753 CheckBracedInit(
"S s = {^};");
1754 CheckBracedInit(
"auto s = S{^};");
1757 CheckBracedInit(
"int x(S); int i = x({^});");
1760TEST(SignatureHelpTest, Aggregates) {
1761 std::string Top = R
"cpp(
1766 auto AggregateSig = sig(
"S{[[int a]], [[int b]], [[int c]], [[int d]]}");
1767 EXPECT_THAT(signatures(Top +
"S s{^}").signatures,
1768 UnorderedElementsAre(AggregateSig, sig(
"S{}"),
1769 sig(
"S{[[const S &]]}"),
1770 sig(
"S{[[S &&]]}")));
1771 EXPECT_THAT(signatures(Top +
"S s{1,^}").signatures,
1772 ElementsAre(AggregateSig));
1773 EXPECT_EQ(signatures(Top +
"S s{1,^}").activeParameter, 1);
1774 EXPECT_THAT(signatures(Top +
"S s{.c=3,^}").signatures,
1775 ElementsAre(AggregateSig));
1776 EXPECT_EQ(signatures(Top +
"S s{.c=3,^}").activeParameter, 3);
1779TEST(SignatureHelpTest, OverloadInitListRegression) {
1780 auto Results = signatures(R
"cpp(
1789 EXPECT_THAT(Results.signatures, UnorderedElementsAre(sig("f() -> void")));
1792TEST(SignatureHelpTest, DefaultArgs) {
1793 auto Results = signatures(R
"cpp(
1794 void bar(int x, int y = 0);
1795 void bar(float x = 0, int y = 42);
1798 EXPECT_THAT(Results.signatures,
1799 UnorderedElementsAre(
1800 sig("bar([[int x]], [[int y = 0]]) -> void"),
1801 sig(
"bar([[float x = 0]], [[int y = 42]]) -> void")));
1802 EXPECT_EQ(0, Results.activeSignature);
1803 EXPECT_EQ(0, Results.activeParameter);
1806TEST(SignatureHelpTest, ActiveArg) {
1807 auto Results = signatures(R
"cpp(
1808 int baz(int a, int b, int c);
1809 int main() { baz(baz(1,2,3), ^); }
1811 EXPECT_THAT(Results.signatures,
1812 ElementsAre(sig("baz([[int a]], [[int b]], [[int c]]) -> int")));
1813 EXPECT_EQ(0, Results.activeSignature);
1814 EXPECT_EQ(1, Results.activeParameter);
1817TEST(SignatureHelpTest, OpeningParen) {
1818 llvm::StringLiteral Tests[] = {
1821 int foo(int a, int b, int c);
1823 foo(foo $p^( foo(10, 10, 10), ^ )));
1828 Foo(int a, int b, int c);
1836 Foo(int a, int b, int c);
1839 new Foo $p^( 10, ^ );
1843 int foo(int a, int b, int c);
1847 // Macro expansions.
1852 int foo(int a, int b, int c);
1855 // FIXME: figure out why ID(foo (foo(10), )) doesn't work when preserving
1856 // the recovery expression.
1857 ID(foo $p^( 10, ^ ))
1861 int foo(int a, int b);
1862 template <typename T> void bar(T t) {
1867 template <typename T>
1869 template <typename T> void bar(T t) {
1874 struct Foo { int foo(int, int); };
1875 template <typename T> void bar(T t) {
1881 struct Foo { template <typename T> int foo(T, T); };
1882 template <typename T> void bar(T t) {
1888 for (
auto Test : Tests) {
1890 EXPECT_EQ(signatures(Code.code(), Code.point()).argListStart,
1892 <<
"Test source:" << Test;
1896TEST(SignatureHelpTest, StalePreamble) {
1901 auto Inputs = TU.inputs(FS);
1906 ASSERT_TRUE(EmptyPreamble);
1908 TU.AdditionalFiles[
"a.h"] =
"int foo(int x);";
1911 void bar() { foo(^2); })cpp");
1912 TU.Code = Test.code().str();
1916 EXPECT_THAT(Results.signatures, ElementsAre(sig(
"foo([[int x]]) -> int")));
1917 EXPECT_EQ(0, Results.activeSignature);
1918 EXPECT_EQ(0, Results.activeParameter);
1921TEST(SignatureHelpTest, EOFInSkippedFunctionBody) {
1924void frameSizeBlocksWarning() {
1934 auto Results = signatures(Test.code(), Test.point());
1935 EXPECT_THAT(Results.signatures, IsEmpty());
1940 IndexRequestCollector(std::vector<Symbol> Syms = {}) : Symbols(Syms) {}
1943 fuzzyFind(
const FuzzyFindRequest &Req,
1944 llvm::function_ref<
void(
const Symbol &)>
Callback)
const override {
1945 std::unique_lock<std::mutex> Lock(Mut);
1946 Requests.push_back(Req);
1947 ReceivedRequestCV.notify_one();
1948 for (
const auto &Sym : Symbols)
1953 void lookup(
const LookupRequest &,
1954 llvm::function_ref<
void(
const Symbol &)>)
const override {}
1956 bool refs(
const RefsRequest &,
1957 llvm::function_ref<
void(
const Ref &)>)
const override {
1962 const ContainedRefsRequest &,
1963 llvm::function_ref<
void(
const ContainedRefsResult &)>)
const override {
1967 void relations(
const RelationsRequest &,
1968 llvm::function_ref<
void(
const SymbolID &,
const Symbol &)>)
1972 reverseRelations(
const RelationsRequest &,
1973 llvm::function_ref<
void(
const SymbolID &,
const Symbol &)>)
1977 indexedFiles()
const override {
1983 size_t estimateMemoryUsage()
const override {
return 0; }
1985 const std::vector<FuzzyFindRequest> consumeRequests(
size_t Num)
const {
1986 std::unique_lock<std::mutex> Lock(Mut);
1988 [
this, Num] {
return Requests.size() == Num; }));
1989 auto Reqs = std::move(Requests);
1995 std::vector<Symbol> Symbols;
1997 mutable std::condition_variable ReceivedRequestCV;
1998 mutable std::mutex Mut;
1999 mutable std::vector<FuzzyFindRequest> Requests;
2003std::vector<FuzzyFindRequest> captureIndexRequests(llvm::StringRef Code,
2005 clangd::CodeCompleteOptions Opts;
2006 IndexRequestCollector Requests;
2007 Opts.Index = &Requests;
2008 completions(Code, {}, Opts);
2009 const auto Reqs = Requests.consumeRequests(Num);
2010 EXPECT_EQ(Reqs.size(), Num);
2014TEST(CompletionTest, UnqualifiedIdQuery) {
2015 auto Requests = captureIndexRequests(R
"cpp(
2017 using namespace std;
2025 EXPECT_THAT(Requests,
2027 UnorderedElementsAre("",
"ns::",
"std::"))));
2030TEST(CompletionTest, EnclosingScopeComesFirst) {
2031 auto Requests = captureIndexRequests(R
"cpp(
2033 using namespace std;
2045 EXPECT_THAT(Requests,
2048 UnorderedElementsAre("",
"std::",
"nx::ns::",
"nx::"))));
2049 EXPECT_EQ(Requests[0].Scopes[0],
"nx::ns::");
2052TEST(CompletionTest, ResolvedQualifiedIdQuery) {
2053 auto Requests = captureIndexRequests(R
"cpp(
2055 namespace ns2 {} // ignore
2056 namespace ns3 { namespace nns3 {} }
2058 using namespace ns1;
2059 using namespace ns3::nns3;
2068 EXPECT_THAT(Requests,
2071 UnorderedElementsAre("foo::",
"ns1::",
"ns3::nns3::"))));
2074TEST(CompletionTest, UnresolvedQualifierIdQuery) {
2075 auto Requests = captureIndexRequests(R
"cpp(
2085 EXPECT_THAT(Requests,
2088 UnorderedElementsAre("a::bar::",
"ns::bar::",
"bar::"))));
2091TEST(CompletionTest, UnresolvedNestedQualifierIdQuery) {
2092 auto Requests = captureIndexRequests(R
"cpp(
2103 UnorderedElementsAre("a::bar::"))));
2106TEST(CompletionTest, EmptyQualifiedQuery) {
2107 auto Requests = captureIndexRequests(R
"cpp(
2116 UnorderedElementsAre("",
"ns::"))));
2119TEST(CompletionTest, GlobalQualifiedQuery) {
2120 auto Requests = captureIndexRequests(R
"cpp(
2129 UnorderedElementsAre(""))));
2132TEST(CompletionTest, NoDuplicatedQueryScopes) {
2133 auto Requests = captureIndexRequests(R
"cpp(
2144 EXPECT_THAT(Requests,
2146 UnorderedElementsAre("na::",
"na::nb::",
""))));
2149TEST(CompletionTest, NoIndexCompletionsInsideClasses) {
2150 auto Completions = completions(
2153 int SomeNameOfField;
2154 typedef int SomeNameOfTypedefField;
2158 {func("::SomeNameInTheIndex"),
func(
"::Foo::SomeNameInTheIndex")});
2160 EXPECT_THAT(Completions.Completions,
2161 AllOf(Contains(labeled(
"SomeNameOfField")),
2162 Contains(labeled(
"SomeNameOfTypedefField")),
2163 Not(Contains(labeled(
"SomeNameInTheIndex")))));
2166TEST(CompletionTest, NoIndexCompletionsInsideDependentCode) {
2168 auto Completions = completions(
2175 {func("::SomeNameInTheIndex")});
2177 EXPECT_THAT(Completions.Completions,
2178 Not(Contains(labeled(
"SomeNameInTheIndex"))));
2182 auto Completions = completions(
2186 T::template Y<int>::^
2189 {func("::SomeNameInTheIndex")});
2191 EXPECT_THAT(Completions.Completions,
2192 Not(Contains(labeled(
"SomeNameInTheIndex"))));
2196 auto Completions = completions(
2203 {func("::SomeNameInTheIndex")});
2205 EXPECT_THAT(Completions.Completions,
2206 Not(Contains(labeled(
"SomeNameInTheIndex"))));
2210TEST(CompletionTest, OverloadBundling) {
2211 clangd::CodeCompleteOptions Opts;
2212 Opts.BundleOverloads =
true;
2216 // Overload with int
2217 int a(int) __attribute__((deprecated("", "")));
2218 // Overload with bool
2230 EXPECT_THAT(completions(
Context +
"int y = X().^", {}, Opts).Completions,
2231 UnorderedElementsAre(labeled(
"a(…)"), labeled(
"b(float)")));
2234 EXPECT_THAT(completions(
Context +
"X z = X^", {}, Opts).Completions,
2235 UnorderedElementsAre(labeled(
"X"), labeled(
"X(…)")));
2240 completions(
Context +
"int y = GFunc^", {NoArgsGFunc}, Opts).Completions,
2241 UnorderedElementsAre(labeled(
"GFuncC(…)"), labeled(
"GFuncD(int)")));
2245 NoArgsGFunc.CanonicalDeclaration.FileURI = DeclFile.c_str();
2248 completions(
Context +
"int y = GFunc^", {NoArgsGFunc}, Opts).Completions,
2249 UnorderedElementsAre(AllOf(named(
"GFuncC"), insertInclude(
"<foo>")),
2250 labeled(
"GFuncC(int)"), labeled(
"GFuncD(int)")));
2254 completions(
Context +
"int y = X().a^", {}, Opts).Completions.front();
2255 EXPECT_EQ(
A.Name,
"a");
2256 EXPECT_EQ(
A.Signature,
"(…)");
2257 EXPECT_EQ(
A.BundleSize, 2u);
2259 EXPECT_EQ(
A.ReturnType,
"int");
2261 ASSERT_TRUE(
A.Documentation);
2262 ASSERT_FALSE(
A.Deprecated);
2264 A.Documentation->asPlainText(),
2265 AnyOf(HasSubstr(
"Overload with int"), HasSubstr(
"Overload with bool")));
2266 EXPECT_EQ(
A.SnippetSuffix,
"($0)");
2269TEST(CompletionTest, OverloadBundlingSameFileDifferentURI) {
2270 clangd::CodeCompleteOptions Opts;
2271 Opts.BundleOverloads =
true;
2273 Symbol SymX =
sym(
"ns::X", index::SymbolKind::Function,
"@F@\\0#");
2274 Symbol SymY =
sym(
"ns::X", index::SymbolKind::Function,
"@F@\\0#I#");
2275 std::string BarHeader =
testPath(
"bar.h");
2277 SymX.CanonicalDeclaration.FileURI = BarURI.c_str();
2278 SymY.CanonicalDeclaration.FileURI = BarURI.c_str();
2283 auto Results = completions(
"void f() { ::ns::^ }", {SymX, SymY}, Opts);
2286 ASSERT_EQ(1u, Results.Completions.size());
2287 const auto &R = Results.Completions.front();
2288 EXPECT_EQ(
"X", R.Name);
2289 EXPECT_EQ(2u, R.BundleSize);
2292TEST(CompletionTest, DocumentationFromChangedFileCrash) {
2296 FS.Files[FooH] = R
"cpp(
2297 // this is my documentation comment.
2300 FS.Files[FooCpp] = "";
2308 // This makes sure we have func from header in the AST.
2314 ASSERT_TRUE(Server.blockUntilIdleForTest());
2317 FS.Files[FooH] = R
"cpp(
2321 clangd::CodeCompleteOptions Opts;
2322 CodeCompleteResult Completions =
2326 EXPECT_THAT(Completions.Completions,
2327 Contains(AllOf(Not(isDocumented()), named(
"func"))));
2330TEST(CompletionTest, NonDocComments) {
2331 const char *
Text = R
"cpp(
2332 // We ignore namespace comments, for rationale see CodeCompletionStrings.h.
2333 namespace comments_ns {
2336 // ------------------
2339 // A comment and a decl are separated by newlines.
2340 // Therefore, the comment shouldn't show up as doc comment.
2344 // this comment should be in the results.
2351 int comments_quux();
2355 // This comment should not be there.
2358 int Struct<T>::comments_qux() {
2361 // This comment **should** be in results.
2363 int Struct<T>::comments_quux() {
2370 completions(
Text).Completions,
2371 UnorderedElementsAre(AllOf(Not(isDocumented()), named(
"comments_foo")),
2372 AllOf(isDocumented(), named(
"comments_baz")),
2373 AllOf(isDocumented(), named(
"comments_quux")),
2374 AllOf(Not(isDocumented()), named(
"comments_ns")),
2379 AllOf(isDocumented(), named(
"comments_bar")),
2380 AllOf(isDocumented(), named(
"comments_qux"))));
2383TEST(CompletionTest, CompleteOnInvalidLine) {
2388 FS.
Files[FooCpp] =
"// empty file";
2395 EXPECT_THAT_EXPECTED(
2400TEST(CompletionTest, QualifiedNames) {
2401 auto Results = completions(
2403 namespace ns { int local; void both(); }
2404 void f() { ::ns::^ }
2406 {func("ns::both"),
cls(
"ns::Index")});
2409 Results.Completions,
2410 UnorderedElementsAre(scope(
"ns::"), scope(
"ns::"), scope(
"ns::")));
2413TEST(CompletionTest, Render) {
2417 C.Signature =
"(bool) const";
2418 C.SnippetSuffix =
"(${0:bool})";
2419 C.ReturnType =
"int";
2420 C.RequiredQualifier =
"Foo::";
2421 C.Scope =
"ns::Foo::";
2422 C.Documentation.emplace();
2423 C.Documentation->addParagraph().appendText(
"This is ").appendCode(
"x()");
2424 C.Includes.emplace_back();
2425 auto &Include =
C.Includes.back();
2426 Include.Header =
"\"foo.h\"";
2428 C.Score.Total = 1.0;
2429 C.Score.ExcludingName = .5;
2433 Opts.IncludeIndicator.Insert =
"^";
2434 Opts.IncludeIndicator.NoInsert =
"";
2435 Opts.EnableSnippets =
false;
2437 auto R =
C.render(Opts);
2438 EXPECT_EQ(R.label,
"Foo::x");
2439 EXPECT_EQ(R.labelDetails->detail,
"(bool) const");
2440 EXPECT_EQ(R.insertText,
"Foo::x");
2442 EXPECT_EQ(R.filterText,
"x");
2443 EXPECT_EQ(R.detail,
"int");
2444 EXPECT_EQ(R.documentation->value,
"From \"foo.h\"\n\nThis is x()");
2445 EXPECT_THAT(R.additionalTextEdits, IsEmpty());
2446 EXPECT_EQ(R.sortText,
sortText(1.0,
"x"));
2447 EXPECT_FALSE(R.deprecated);
2448 EXPECT_EQ(R.score, .5f);
2450 C.FilterText =
"xtra";
2452 EXPECT_EQ(R.filterText,
"xtra");
2453 EXPECT_EQ(R.sortText,
sortText(1.0,
"xtra"));
2455 Opts.EnableSnippets =
true;
2457 EXPECT_EQ(R.insertText,
"Foo::x(${0:bool})");
2460 C.SnippetSuffix =
"";
2462 EXPECT_EQ(R.insertText,
"Foo::x");
2465 Include.Insertion.emplace();
2467 EXPECT_EQ(R.label,
"^Foo::x");
2468 EXPECT_EQ(R.labelDetails->detail,
"(bool) const");
2469 EXPECT_THAT(R.additionalTextEdits, Not(IsEmpty()));
2471 Opts.ShowOrigins =
true;
2473 EXPECT_EQ(R.label,
"^[AS]Foo::x");
2474 EXPECT_EQ(R.labelDetails->detail,
"(bool) const");
2478 EXPECT_EQ(R.detail,
"[2 overloads]");
2479 EXPECT_EQ(R.documentation->value,
"From \"foo.h\"\n\nThis is x()");
2481 C.Deprecated =
true;
2483 EXPECT_TRUE(R.deprecated);
2487 EXPECT_EQ(R.documentation->value,
"From `\"foo.h\"`\n\nThis is `x()`");
2490TEST(CompletionTest, IgnoreRecoveryResults) {
2491 auto Results = completions(
2493 namespace ns { int NotRecovered() { return 0; } }
2495 // Sema enters recovery mode first and then normal mode.
2496 if (auto x = ns::NotRecover^)
2499 EXPECT_THAT(Results.Completions, UnorderedElementsAre(named("NotRecovered")));
2502TEST(CompletionTest, ScopeOfClassFieldInConstructorInitializer) {
2503 auto Results = completions(
2506 class X { public: X(); int x_; };
2510 EXPECT_THAT(Results.Completions,
2511 UnorderedElementsAre(AllOf(scope("ns::X::"), named(
"x_"))));
2518TEST(CompletionTest, ConstructorInitListIncomplete) {
2519 auto Results = completions(
2528 EXPECT_THAT(Results.Completions, ElementsAre(named("xyz_")));
2530 Results = completions(
2541 EXPECT_THAT(Results.Completions, ElementsAre(named("foo")));
2544TEST(CompletionTest, CodeCompletionContext) {
2545 auto Results = completions(
2548 class X { public: X(); int x_; };
2556 EXPECT_THAT(Results.Context, CodeCompletionContext::CCC_DotMemberAccess);
2559TEST(CompletionTest, OffsetOfDesignator) {
2560 auto Results = completions(R
"cpp(
2561 struct S { int field; int other; void fieldFn(); };
2562 int x = __builtin_offsetof(S, fiel^d);
2565 Results.Completions,
2567 EXPECT_THAT(Results.Context, CodeCompletionContext::CCC_DotMemberAccess);
2569 Results = completions(R
"cpp(
2570 struct Inner { int field; void fieldFn(); };
2571 struct Outer { Inner inner; };
2572 int x = __builtin_offsetof(Outer, inner.fiel^d);
2575 Results.Completions,
2578 Results = completions(R
"cpp(
2579 struct Inner { int field; void fieldFn(); };
2580 struct Outer { Inner inner[2]; };
2582 int x = __builtin_offsetof(Outer, inner[i].fiel^d);
2585 Results.Completions,
2588 Results = completions(R
"cpp(
2589 struct Base { int field; void fieldFn(); };
2590 struct Derived : Base {};
2591 int x = __builtin_offsetof(Derived, fiel^d);
2594 Results.Completions,
2598TEST(CompletionTest, FixItForArrowToDot) {
2603 Opts.IncludeFixIts =
true;
2610 class ClassWithPtr {
2612 void MemberFunction();
2613 Auxilary* operator->() const;
2621 auto Results = completions(Code, {}, Opts);
2622 EXPECT_EQ(Results.Completions.size(), 3u);
2626 ReplacementEdit.newText =
".";
2627 for (
const auto &C : Results.Completions) {
2628 EXPECT_TRUE(
C.FixIts.size() == 1u ||
C.Name ==
"AuxFunction");
2629 if (!
C.FixIts.empty()) {
2630 EXPECT_THAT(
C.FixIts, ElementsAre(ReplacementEdit));
2635TEST(CompletionTest, FixItForDotToArrow) {
2637 Opts.IncludeFixIts =
true;
2644 class ClassWithPtr {
2646 void MemberFunction();
2647 Auxilary* operator->() const;
2655 auto Results = completions(Code, {}, Opts);
2656 EXPECT_EQ(Results.Completions.size(), 3u);
2660 ReplacementEdit.newText =
"->";
2661 for (
const auto &C : Results.Completions) {
2662 EXPECT_TRUE(
C.FixIts.empty() ||
C.Name ==
"AuxFunction");
2663 if (!
C.FixIts.empty()) {
2664 EXPECT_THAT(
C.FixIts, ElementsAre(ReplacementEdit));
2669TEST(CompletionTest, RenderWithFixItMerged) {
2672 FixIt.newText =
"->";
2676 C.RequiredQualifier =
"Foo::";
2678 C.CompletionInsertRange.start.character = 5;
2681 Opts.IncludeFixIts =
true;
2683 auto R =
C.render(Opts);
2684 EXPECT_TRUE(R.textEdit);
2685 EXPECT_EQ(std::get<TextEdit>(*R.textEdit).newText,
"->Foo::x");
2686 EXPECT_TRUE(R.additionalTextEdits.empty());
2689TEST(CompletionTest, RenderWithFixItNonMerged) {
2692 FixIt.newText =
"->";
2696 C.RequiredQualifier =
"Foo::";
2698 C.CompletionInsertRange.start.character = 5;
2701 Opts.IncludeFixIts =
true;
2703 auto R =
C.render(Opts);
2704 EXPECT_TRUE(R.textEdit);
2705 EXPECT_EQ(std::get<TextEdit>(*R.textEdit).newText,
"Foo::x");
2706 EXPECT_THAT(R.additionalTextEdits, UnorderedElementsAre(FixIt));
2709TEST(CompletionTest, CompletionInsertRange) {
2715 constexpr const char *TestCodes[] = {
2737 #include "foo/[[a^/]]foo.h"
2740 #include "foo/abc/[[fo^o.h"]]
2743 for (
const auto &
Text : TestCodes) {
2745 TU.Code = TestCode.code().str();
2746 auto Results = completions(TU, TestCode.point());
2747 if (Results.Completions.size() != 1) {
2748 ADD_FAILURE() <<
"Results.Completions.size() != 1" <<
Text;
2751 EXPECT_THAT(Results.Completions.front().CompletionInsertRange,
2756TEST(SignatureHelpTest, OverloadsOrdering) {
2757 const auto Results = signatures(R
"cpp(
2759 void foo(int x, float y);
2760 void foo(float x, int y);
2761 void foo(float x, float y);
2762 void foo(int x, int y = 0);
2763 int main() { foo(^); }
2765 EXPECT_THAT(Results.signatures,
2766 ElementsAre(sig("foo([[int x]]) -> void"),
2767 sig(
"foo([[int x]], [[int y = 0]]) -> void"),
2768 sig(
"foo([[float x]], [[int y]]) -> void"),
2769 sig(
"foo([[int x]], [[float y]]) -> void"),
2770 sig(
"foo([[float x]], [[float y]]) -> void")));
2772 EXPECT_EQ(0, Results.activeSignature);
2773 EXPECT_EQ(0, Results.activeParameter);
2776TEST(SignatureHelpTest, InstantiatedSignatures) {
2777 StringRef Sig0 = R
"cpp(
2786 EXPECT_THAT(signatures(Sig0).signatures,
2787 ElementsAre(sig("foo([[T]], [[T]], [[T]]) -> void")));
2789 StringRef Sig1 = R
"cpp(
2797 EXPECT_THAT(signatures(Sig1).signatures,
2798 ElementsAre(sig("foo([[T]], [[T]], [[T]]) -> void")));
2800 StringRef Sig2 = R
"cpp(
2801 template <class ...T>
2809 EXPECT_THAT(signatures(Sig2).signatures,
2810 ElementsAre(sig("foo([[T...]]) -> void")));
2817 StringRef Sig3 = R
"cpp(
2825 X<int>().foo<double>(^)
2829 EXPECT_THAT(signatures(Sig3).signatures,
2830 ElementsAre(sig("foo([[T]], [[U]]) -> void")));
2833TEST(SignatureHelpTest, IndexDocumentation) {
2834 Symbol Foo0 =
sym(
"foo", index::SymbolKind::Function,
"@F@\\0#");
2835 Foo0.Documentation =
"doc from the index";
2836 Symbol Foo1 =
sym(
"foo", index::SymbolKind::Function,
"@F@\\0#I#");
2837 Foo1.Documentation =
"doc from the index";
2838 Symbol Foo2 =
sym(
"foo", index::SymbolKind::Function,
"@F@\\0#I#I#");
2840 StringRef Sig0 = R
"cpp(
2850 signatures(Sig0, {Foo0}).signatures,
2851 ElementsAre(AllOf(sig("foo() -> int"), sigDoc(
"doc from the index")),
2852 AllOf(sig(
"foo([[double]]) -> int"), sigDoc(
""))));
2854 StringRef Sig1 = R
"cpp(
2856 // Overriden doc from sema
2867 signatures(Sig1, {Foo0, Foo1, Foo2}).signatures,
2869 AllOf(sig("foo() -> int"), sigDoc(
"doc from the index")),
2870 AllOf(sig(
"foo([[int]]) -> int"), sigDoc(
"Overriden doc from sema")),
2871 AllOf(sig(
"foo([[int]], [[int]]) -> int"), sigDoc(
"doc from sema"))));
2874TEST(SignatureHelpTest, DynamicIndexDocumentation) {
2878 Opts.BuildDynamicSymbolIndex =
true;
2881 FS.Files[
testPath(
"foo.h")] = R
"cpp(
2895 Server.addDocument(
File, FileContent.code());
2897 ASSERT_TRUE(Server.blockUntilIdleForTest());
2901 ElementsAre(AllOf(sig(
"foo() -> int"), sigDoc(
"Member doc"))));
2904TEST(CompletionTest, ArgumentListsPolicy) {
2906 Opts.EnableSnippets =
true;
2910 auto Results = completions(
2913 void xfoo(int x, int y);
2914 void f() { xfo^ })cpp",
2917 Results.Completions,
2918 UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix(
"()")),
2919 AllOf(named(
"xfoo"), snippetSuffix(
"($0)"))));
2922 auto Results = completions(
2925 void f() { xba^ })cpp",
2927 EXPECT_THAT(Results.Completions, UnorderedElementsAre(AllOf(
2928 named("xbar"), snippetSuffix(
"()"))));
2931 Opts.BundleOverloads =
true;
2932 auto Results = completions(
2935 void xfoo(int x, int y);
2936 void f() { xfo^ })cpp",
2939 Results.Completions,
2940 UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix(
"($0)"))));
2943 auto Results = completions(
2945 template <class T, class U>
2946 void xfoo(int a, U b);
2947 void f() { xfo^ })cpp",
2950 Results.Completions,
2951 UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix(
"<$1>($0)"))));
2954 auto Results = completions(
2959 using foo_alias = T**;
2962 void f() { foo_^ })cpp",
2965 Results.Completions,
2966 UnorderedElementsAre(AllOf(named("foo_class"), snippetSuffix(
"<$0>")),
2967 AllOf(named(
"foo_alias"), snippetSuffix(
"<$0>")),
2968 AllOf(named(
"foo_var"), snippetSuffix(
"<$0>"))));
2971 auto Results = completions(
2973 #define FOO(x, y) x##f
2976 EXPECT_THAT(Results.Completions, UnorderedElementsAre(AllOf(
2977 named("FOO"), snippetSuffix(
"($0)"))));
2980 auto Results = completions(
2983 auto Lambda = [](int a, const double &b) {return 1.f;};
2988 Results.Completions,
2989 UnorderedElementsAre(AllOf(named("Lambda"), snippetSuffix(
"($0)"))));
2993 auto Results = completions(
2995 void xfoo(int x, int y);
2996 void f() { xfo^ })cpp",
2998 EXPECT_THAT(Results.Completions,
2999 UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix(
""))));
3003 auto Results = completions(
3005 void xfoo(int x, int y);
3006 void f() { xfo^ })cpp",
3008 EXPECT_THAT(Results.Completions,
3009 UnorderedElementsAre(AllOf(named("xfoo"), snippetSuffix(
"("))));
3013TEST(CompletionTest, SuggestOverrides) {
3014 constexpr const char *
const Text(R
"cpp(
3017 virtual void vfunc(bool param);
3018 virtual void vfunc(bool param, int p);
3019 void func(bool param);
3021 class B : public A {
3022 virtual void ttt(bool param) const;
3023 void vfunc(bool param, int p) override;
3025 class C : public B {
3027 void vfunc(bool param) override;
3031 const auto Results = completions(
Text);
3033 Results.Completions,
3034 AllOf(Contains(AllOf(labeled(
"void vfunc(bool param, int p) override"),
3035 nameStartsWith(
"vfunc"))),
3036 Contains(AllOf(labeled(
"void ttt(bool param) const override"),
3037 nameStartsWith(
"ttt"))),
3038 Not(Contains(labeled(
"void vfunc(bool param) override")))));
3041TEST(CompletionTest, OverridesNonIdentName) {
3045 virtual ~Base() = 0;
3046 virtual operator int() = 0;
3047 virtual Base& operator+(Base&) = 0;
3050 struct Derived : Base {
3056TEST(CompletionTest, NoCrashOnMissingNewLineAtEOF) {
3062 FS.Files[FooCpp] = F.code().str();
3067 clangd::CodeCompleteOptions()))
3076TEST(GuessCompletionPrefix, Filters) {
3077 for (llvm::StringRef Case : {
3078 "[[scope::]][[ident]]^",
3087 "some text [[scope::more::]][[identif]]^ier",
3088 "some text [[scope::]][[mor]]^e::identifier",
3089 "weird case foo::[[::bar::]][[baz]]^",
3094 auto ToStringRef = [&](
Range R) {
3098 auto WantQualifier = ToStringRef(F.ranges()[0]),
3099 WantName = ToStringRef(F.ranges()[1]);
3103 EXPECT_EQ(WantQualifier, Prefix.Qualifier) << Case;
3104 EXPECT_EQ(WantQualifier.begin(), Prefix.Qualifier.begin()) << Case;
3105 EXPECT_EQ(WantName, Prefix.Name) << Case;
3106 EXPECT_EQ(WantName.begin(), Prefix.Name.begin()) << Case;
3110TEST(CompletionTest, EnableSpeculativeIndexRequest) {
3117 namespace ns1 { int abc; }
3118 namespace ns2 { int abc; }
3119 void f() { ns1::ab$1^; ns1::ab$2^; }
3120 void f2() { ns2::ab$3^; }
3123 clangd::CodeCompleteOptions Opts = {};
3125 IndexRequestCollector Requests;
3126 Opts.Index = &Requests;
3128 auto CompleteAtPoint = [&](StringRef
P) {
3130 EXPECT_TRUE(CCR.HasMore);
3133 CompleteAtPoint(
"1");
3134 auto Reqs1 = Requests.consumeRequests(1);
3135 ASSERT_EQ(Reqs1.size(), 1u);
3136 EXPECT_THAT(Reqs1[0].Scopes, UnorderedElementsAre(
"ns1::"));
3138 CompleteAtPoint(
"2");
3139 auto Reqs2 = Requests.consumeRequests(1);
3141 ASSERT_EQ(Reqs2.size(), 1u);
3142 EXPECT_EQ(Reqs2[0], Reqs1[0]);
3144 CompleteAtPoint(
"3");
3147 auto Reqs3 = Requests.consumeRequests(2);
3148 ASSERT_EQ(Reqs3.size(), 2u);
3151TEST(CompletionTest, InsertTheMostPopularHeader) {
3154 Sym.CanonicalDeclaration.FileURI = DeclFile.c_str();
3158 auto Results = completions(
"Fun^", {Sym}).Completions;
3159 assert(!Results.empty());
3160 EXPECT_THAT(Results[0], AllOf(named(
"Func"), insertInclude(
"\"bar.h\"")));
3161 EXPECT_EQ(Results[0].Includes.size(), 2u);
3164TEST(CompletionTest, InsertIncludeOrImport) {
3167 Sym.CanonicalDeclaration.FileURI = DeclFile.c_str();
3168 Sym.IncludeHeaders.emplace_back(
"\"bar.h\"", 1000,
3172 Opts.ImportInsertions =
true;
3173 auto Results = completions(
"Fun^", {Sym}, Opts).Completions;
3174 assert(!Results.empty());
3175 EXPECT_THAT(Results[0],
3176 AllOf(named(
"Func"), insertIncludeText(
"#include \"bar.h\"\n")));
3180 Opts.MainFileSignals = &Signals;
3181 Results = completions(
"Fun^", {Sym}, Opts,
"Foo.m").Completions;
3182 assert(!Results.empty());
3183 EXPECT_THAT(Results[0],
3184 AllOf(named(
"Func"), insertIncludeText(
"#import \"bar.h\"\n")));
3187 Results = completions(
"Fun^", {Sym}).Completions;
3188 assert(!Results.empty());
3189 EXPECT_THAT(Results[0], AllOf(named(
"Func"), Not(insertInclude())));
3192TEST(CompletionTest, NoInsertIncludeIfOnePresent) {
3198 TU.AdditionalFiles[
"foo.h"] =
"";
3202 Sym.CanonicalDeclaration.FileURI = DeclFile.c_str();
3206 EXPECT_THAT(completions(TU, Test.point(), {Sym}).Completions,
3207 UnorderedElementsAre(AllOf(named(
"Func"), hasInclude(
"\"foo.h\""),
3208 Not(insertInclude()))));
3211TEST(CompletionTest, MergeMacrosFromIndexAndSema) {
3213 Sym.Name =
"Clangd_Macro_Test";
3214 Sym.ID =
SymbolID(
"c:foo.cpp@8@macro@Clangd_Macro_Test");
3215 Sym.SymInfo.Kind = index::SymbolKind::Macro;
3217 EXPECT_THAT(completions(
"#define Clangd_Macro_Test\nClangd_Macro_T^", {Sym})
3219 UnorderedElementsAre(named(
"Clangd_Macro_Test")));
3222TEST(CompletionTest, MacroFromPreamble) {
3223 Annotations Test(R
"cpp(#define CLANGD_PREAMBLE_MAIN x
3226 #define CLANGD_MAIN x
3227 void f() { CLANGD_^ }
3230 TU.HeaderCode =
"#define CLANGD_PREAMBLE_HEADER x";
3231 auto Results = completions(TU, Test.point(), {func(
"CLANGD_INDEX")});
3234 EXPECT_THAT(Results.Completions,
3235 UnorderedElementsAre(named(
"CLANGD_PREAMBLE_MAIN"),
3236 named(
"CLANGD_MAIN"),
3237 named(
"CLANGD_INDEX")));
3240TEST(CompletionTest, DeprecatedResults) {
3241 std::string Body = R
"cpp(
3243 void TestClangc() __attribute__((deprecated("", "")));
3247 completions(Body + "int main() { TestClang^ }").Completions,
3248 UnorderedElementsAre(AllOf(named(
"TestClangd"), Not(deprecated())),
3249 AllOf(named(
"TestClangc"), deprecated())));
3252TEST(SignatureHelpTest, PartialSpec) {
3253 const auto Results = signatures(R
"cpp(
3254 template <typename T> struct Foo {};
3255 template <typename T> struct Foo<T*> { Foo(T); };
3256 Foo<int*> F(^);)cpp");
3257 EXPECT_THAT(Results.signatures, Contains(sig("Foo([[T]])")));
3258 EXPECT_EQ(0, Results.activeParameter);
3261TEST(SignatureHelpTest, InsideArgument) {
3263 const auto Results = signatures(R
"cpp(
3265 void foo(int x, int y);
3266 int main() { foo(1+^); }
3268 EXPECT_THAT(Results.signatures,
3269 ElementsAre(sig("foo([[int x]]) -> void"),
3270 sig(
"foo([[int x]], [[int y]]) -> void")));
3271 EXPECT_EQ(0, Results.activeParameter);
3274 const auto Results = signatures(R
"cpp(
3276 void foo(int x, int y);
3277 int main() { foo(1^); }
3279 EXPECT_THAT(Results.signatures,
3280 ElementsAre(sig("foo([[int x]]) -> void"),
3281 sig(
"foo([[int x]], [[int y]]) -> void")));
3282 EXPECT_EQ(0, Results.activeParameter);
3285 const auto Results = signatures(R
"cpp(
3287 void foo(int x, int y);
3288 int main() { foo(1^0); }
3290 EXPECT_THAT(Results.signatures,
3291 ElementsAre(sig("foo([[int x]]) -> void"),
3292 sig(
"foo([[int x]], [[int y]]) -> void")));
3293 EXPECT_EQ(0, Results.activeParameter);
3296 const auto Results = signatures(R
"cpp(
3298 void foo(int x, int y);
3299 int bar(int x, int y);
3300 int main() { bar(foo(2, 3^)); }
3302 EXPECT_THAT(Results.signatures,
3303 ElementsAre(sig("foo([[int x]], [[int y]]) -> void")));
3304 EXPECT_EQ(1, Results.activeParameter);
3308TEST(SignatureHelpTest, ConstructorInitializeFields) {
3310 const auto Results = signatures(R
"cpp(
3311 struct A { A(int); };
3317 EXPECT_THAT(Results.signatures,
3318 UnorderedElementsAre(sig("A([[int]])"), sig(
"A([[A &&]])"),
3319 sig(
"A([[const A &]])")));
3322 const auto Results = signatures(R
"cpp(
3323 struct A { A(int); };
3331 EXPECT_THAT(Results.signatures, IsEmpty());
3334 const auto Results = signatures(R
"cpp(
3335 struct A { A(int); };
3342 EXPECT_THAT(Results.signatures,
3343 UnorderedElementsAre(sig("A([[int]])"), sig(
"A([[A &&]])"),
3344 sig(
"A([[const A &]])")));
3347 const auto Results = signatures(R
"cpp(
3356 B() : c_elem(A(1^)) {}
3360 EXPECT_THAT(Results.signatures,
3361 UnorderedElementsAre(sig("A([[int]])"), sig(
"A([[A &&]])"),
3362 sig(
"A([[const A &]])")));
3366TEST(SignatureHelpTest, Variadic) {
3367 const std::string Header = R
"cpp(
3368 void fun(int x, ...) {}
3370 const std::string ExpectedSig =
"fun([[int x]], [[...]]) -> void";
3373 const auto Result = signatures(Header +
"fun(^);}");
3374 EXPECT_EQ(0, Result.activeParameter);
3375 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3378 const auto Result = signatures(Header +
"fun(1, ^);}");
3379 EXPECT_EQ(1, Result.activeParameter);
3380 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3383 const auto Result = signatures(Header +
"fun(1, 2, ^);}");
3384 EXPECT_EQ(1, Result.activeParameter);
3385 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3389TEST(SignatureHelpTest, VariadicTemplate) {
3390 const std::string Header = R
"cpp(
3391 template<typename T, typename ...Args>
3392 void fun(T t, Args ...args) {}
3394 const std::string ExpectedSig =
"fun([[T t]], [[Args args...]]) -> void";
3397 const auto Result = signatures(Header +
"fun(^);}");
3398 EXPECT_EQ(0, Result.activeParameter);
3399 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3402 const auto Result = signatures(Header +
"fun(1, ^);}");
3403 EXPECT_EQ(1, Result.activeParameter);
3404 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3407 const auto Result = signatures(Header +
"fun(1, 2, ^);}");
3408 EXPECT_EQ(1, Result.activeParameter);
3409 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3413TEST(SignatureHelpTest, VariadicMethod) {
3414 const std::string Header = R
"cpp(
3416 template<typename T, typename ...Args>
3417 void fun(T t, Args ...args) {}
3419 void test() {C c; )cpp";
3420 const std::string ExpectedSig =
"fun([[T t]], [[Args args...]]) -> void";
3423 const auto Result = signatures(Header +
"c.fun(^);}");
3424 EXPECT_EQ(0, Result.activeParameter);
3425 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3428 const auto Result = signatures(Header +
"c.fun(1, ^);}");
3429 EXPECT_EQ(1, Result.activeParameter);
3430 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3433 const auto Result = signatures(Header +
"c.fun(1, 2, ^);}");
3434 EXPECT_EQ(1, Result.activeParameter);
3435 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3439TEST(SignatureHelpTest, VariadicType) {
3440 const std::string Header = R
"cpp(
3441 void fun(int x, ...) {}
3442 auto get_fun() { return fun; }
3445 const std::string ExpectedSig =
"([[int]], [[...]]) -> void";
3448 const auto Result = signatures(Header +
"get_fun()(^);}");
3449 EXPECT_EQ(0, Result.activeParameter);
3450 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3453 const auto Result = signatures(Header +
"get_fun()(1, ^);}");
3454 EXPECT_EQ(1, Result.activeParameter);
3455 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3458 const auto Result = signatures(Header +
"get_fun()(1, 2, ^);}");
3459 EXPECT_EQ(1, Result.activeParameter);
3460 EXPECT_THAT(Result.signatures, UnorderedElementsAre(sig(ExpectedSig)));
3464TEST(SignatureHelpTest, SkipExplicitObjectParameter) {
3467 void foo(this auto&& self, int arg);
3468 void bar(this A self, int arg);
3479 TU.ExtraArgs = {
"-std=c++23"};
3482 auto Inputs = TU.inputs(FS);
3491 EXPECT_EQ(1U, Result.signatures.size());
3493 EXPECT_THAT(Result.signatures[0], AllOf(sig(
"foo([[int arg]]) -> void")));
3499 EXPECT_EQ(1U, Result.signatures.size());
3501 EXPECT_THAT(Result.signatures[0], AllOf(sig(
"([[A]], [[int]]) -> void")));
3508 EXPECT_EQ(0U, Result.signatures.size());
3514TEST(CompletionTest, IncludedCompletionKinds) {
3517 TU.AdditionalFiles[
"sub/bar.h"] =
"";
3518 TU.ExtraArgs.push_back(
"-I" +
testPath(
"sub"));
3520 auto Results = completions(TU, Test.point());
3521 EXPECT_THAT(Results.Completions,
3526TEST(CompletionTest, NoCrashAtNonAlphaIncludeHeader) {
3533TEST(CompletionTest, NoAllScopesCompletionWhenQualified) {
3534 clangd::CodeCompleteOptions Opts = {};
3535 Opts.AllScopes = true;
3537 auto Results = completions(
3539 void f() { na::Clangd^ }
3541 {cls("na::ClangdA"),
cls(
"nx::ClangdX"),
cls(
"Clangd3")}, Opts);
3542 EXPECT_THAT(Results.Completions,
3543 UnorderedElementsAre(
3544 AllOf(qualifier(
""), scope(
"na::"), named(
"ClangdA"))));
3547TEST(CompletionTest, AllScopesCompletion) {
3548 clangd::CodeCompleteOptions Opts = {};
3549 Opts.AllScopes =
true;
3551 auto Results = completions(
3554 void f() { Clangd^ }
3557 {cls("nx::Clangd1"),
cls(
"ny::Clangd2"),
cls(
"Clangd3"),
3561 Results.Completions,
3562 UnorderedElementsAre(AllOf(qualifier(
"nx::"), named(
"Clangd1"),
3564 AllOf(qualifier(
"ny::"), named(
"Clangd2"),
3566 AllOf(qualifier(
""), scope(
""), named(
"Clangd3"),
3568 AllOf(qualifier(
"nb::"), named(
"Clangd4"),
3570 AllOf(qualifier(
"C::"), named(
"Clangd5"),
3574TEST(CompletionTest, NoCodePatternsIfDisabled) {
3575 clangd::CodeCompleteOptions Opts = {};
3576 Opts.EnableSnippets =
true;
3579 auto Results = completions(R
"cpp(
3588 EXPECT_THAT(Results.Completions,
3592TEST(CompletionTest, CompleteIncludeIfCodePatternsNone) {
3593 clangd::CodeCompleteOptions Opts = {};
3594 Opts.EnableSnippets =
true;
3599 TU.AdditionalFiles[
"foo/bar.h"] =
"";
3600 TU.ExtraArgs.push_back(
"-I" +
testPath(
"foo"));
3602 auto Results = completions(TU, Test.point(), {}, Opts);
3603 EXPECT_THAT(Results.Completions,
3608TEST(CompletionTest, NoQualifierIfShadowed) {
3609 clangd::CodeCompleteOptions Opts = {};
3610 Opts.AllScopes =
true;
3612 auto Results = completions(R
"cpp(
3613 namespace nx { class Clangd1 {}; }
3615 void f() { Clangd^ }
3617 {cls("nx::Clangd1"),
cls(
"nx::Clangd2")}, Opts);
3620 EXPECT_THAT(Results.Completions,
3621 UnorderedElementsAre(AllOf(qualifier(
""), named(
"Clangd1")),
3622 AllOf(qualifier(
"nx::"), named(
"Clangd2"))));
3625TEST(CompletionTest, NoCompletionsForNewNames) {
3626 clangd::CodeCompleteOptions Opts;
3627 Opts.AllScopes =
true;
3628 auto Results = completions(R
"cpp(
3631 {cls("naber"),
cls(
"nx::naber")}, Opts);
3632 EXPECT_THAT(Results.Completions, UnorderedElementsAre());
3635TEST(CompletionTest, Lambda) {
3636 clangd::CodeCompleteOptions Opts = {};
3638 auto Results = completions(R
"cpp(
3640 auto Lambda = [](int a, const double &b) {return 1.f;};
3646 ASSERT_EQ(Results.Completions.size(), 1u);
3647 const auto &
A = Results.Completions.front();
3648 EXPECT_EQ(
A.Name,
"Lambda");
3649 EXPECT_EQ(
A.Signature,
"(int a, const double &b) const");
3651 EXPECT_EQ(
A.ReturnType,
"float");
3652 EXPECT_EQ(
A.SnippetSuffix,
"(${1:int a}, ${2:const double &b})");
3655TEST(CompletionTest, StructuredBinding) {
3656 clangd::CodeCompleteOptions Opts = {};
3658 auto Results = completions(R
"cpp(
3660 using Float = float;
3665 const auto &[xxx, yyy] = S{};
3671 ASSERT_EQ(Results.Completions.size(), 1u);
3672 const auto &
A = Results.Completions.front();
3673 EXPECT_EQ(
A.Name,
"yyy");
3675 EXPECT_EQ(
A.ReturnType,
"const Float");
3678TEST(CompletionTest, ObjectiveCMethodNoArguments) {
3679 auto Results = completions(R
"objc(
3681 @property(nonatomic, setter=setXToIgnoreComplete:) int value;
3683 Foo *foo = [Foo new]; int y = [foo v^]
3688 auto C = Results.Completions;
3689 EXPECT_THAT(C, ElementsAre(named(
"value")));
3691 EXPECT_THAT(C, ElementsAre(returnType(
"int")));
3692 EXPECT_THAT(C, ElementsAre(signature(
"")));
3693 EXPECT_THAT(C, ElementsAre(snippetSuffix(
"")));
3696TEST(CompletionTest, ObjectiveCMethodOneArgument) {
3697 auto Results = completions(R
"objc(
3699 - (int)valueForCharacter:(char)c;
3701 Foo *foo = [Foo new]; int y = [foo v^]
3706 auto C = Results.Completions;
3707 EXPECT_THAT(C, ElementsAre(named(
"valueForCharacter:")));
3709 EXPECT_THAT(C, ElementsAre(returnType(
"int")));
3710 EXPECT_THAT(C, ElementsAre(signature(
"(char)")));
3711 EXPECT_THAT(C, ElementsAre(snippetSuffix(
"${1:(char)}")));
3714TEST(CompletionTest, ObjectiveCMethodTwoArgumentsFromBeginning) {
3715 auto Results = completions(R
"objc(
3717 + (id)fooWithValue:(int)value fooey:(unsigned int)fooey;
3724 auto C = Results.Completions;
3725 EXPECT_THAT(C, ElementsAre(named(
"fooWithValue:")));
3727 EXPECT_THAT(C, ElementsAre(returnType(
"id")));
3728 EXPECT_THAT(C, ElementsAre(signature(
"(int) fooey:(unsigned int)")));
3730 C, ElementsAre(snippetSuffix(
"${1:(int)} fooey:${2:(unsigned int)}")));
3733TEST(CompletionTest, ObjectiveCMethodTwoArgumentsFromMiddle) {
3734 auto Results = completions(R
"objc(
3736 + (id)fooWithValue:(int)value fooey:(unsigned int)fooey;
3738 id val = [Foo fooWithValue:10 f^]
3743 auto C = Results.Completions;
3744 EXPECT_THAT(C, ElementsAre(named(
"fooey:")));
3746 EXPECT_THAT(C, ElementsAre(returnType(
"id")));
3747 EXPECT_THAT(C, ElementsAre(signature(
"(unsigned int)")));
3748 EXPECT_THAT(C, ElementsAre(snippetSuffix(
"${1:(unsigned int)}")));
3751TEST(CompletionTest, ObjectiveCMethodFilterOnEntireSelector) {
3752 auto Results = completions(R
"objc(
3754 + (id)player:(id)player willRun:(id)run;
3761 auto C = Results.Completions;
3762 EXPECT_THAT(C, ElementsAre(named(
"player:")));
3763 EXPECT_THAT(C, ElementsAre(filterText(
"player:willRun:")));
3765 EXPECT_THAT(C, ElementsAre(returnType(
"id")));
3766 EXPECT_THAT(C, ElementsAre(signature(
"(id) willRun:(id)")));
3767 EXPECT_THAT(C, ElementsAre(snippetSuffix(
"${1:(id)} willRun:${2:(id)}")));
3770TEST(CompletionTest, ObjectiveCSimpleMethodDeclaration) {
3771 auto Results = completions(R
"objc(
3782 auto C = Results.Completions;
3783 EXPECT_THAT(C, ElementsAre(named(
"foo")));
3785 EXPECT_THAT(C, ElementsAre(qualifier(
"- (void)")));
3788TEST(CompletionTest, ObjectiveCMethodDeclaration) {
3789 auto Results = completions(R
"objc(
3791 - (int)valueForCharacter:(char)c secondArgument:(id)object;
3800 auto C = Results.Completions;
3801 EXPECT_THAT(C, ElementsAre(named(
"valueForCharacter:")));
3803 EXPECT_THAT(C, ElementsAre(qualifier(
"- (int)")));
3804 EXPECT_THAT(C, ElementsAre(signature(
"(char)c secondArgument:(id)object")));
3807TEST(CompletionTest, ObjectiveCMethodDeclarationFilterOnEntireSelector) {
3808 auto Results = completions(R
"objc(
3810 - (int)valueForCharacter:(char)c secondArgument:(id)object;
3819 auto C = Results.Completions;
3820 EXPECT_THAT(C, ElementsAre(named(
"valueForCharacter:")));
3821 EXPECT_THAT(C, ElementsAre(filterText(
"valueForCharacter:secondArgument:")));
3823 EXPECT_THAT(C, ElementsAre(qualifier(
"- (int)")));
3824 EXPECT_THAT(C, ElementsAre(signature(
"(char)c secondArgument:(id)object")));
3827TEST(CompletionTest, ObjectiveCMethodDeclarationPrefixTyped) {
3828 auto Results = completions(R
"objc(
3830 - (int)valueForCharacter:(char)c;
3839 auto C = Results.Completions;
3840 EXPECT_THAT(C, ElementsAre(named(
"valueForCharacter:")));
3842 EXPECT_THAT(C, ElementsAre(signature(
"(char)c")));
3845TEST(CompletionTest, ObjectiveCMethodDeclarationFromMiddle) {
3846 auto Results = completions(R
"objc(
3848 - (int)valueForCharacter:(char)c secondArgument:(id)object;
3851 - (int)valueForCharacter:(char)c second^
3857 auto C = Results.Completions;
3858 EXPECT_THAT(C, ElementsAre(named(
"secondArgument:")));
3860 EXPECT_THAT(C, ElementsAre(signature(
"(id)object")));
3863TEST(CompletionTest, ObjectiveCProtocolFromIndex) {
3867 auto Results = completions(
"id<Foo^>", {SymFood, FoodClass, SymFooey},
3871 EXPECT_THAT(Results.Completions,
3872 UnorderedElementsAre(
3876 Results = completions(
"Fo^", {SymFood, FoodClass, SymFooey},
3880 Results.Completions,
3884TEST(CompletionTest, ObjectiveCProtocolFromIndexSpeculation) {
3897 clangd::CodeCompleteOptions Opts = {};
3900 IndexRequestCollector Requests({FoodClass});
3901 Opts.Index = &Requests;
3903 auto CompleteAtPoint = [&](StringRef
P) {
3908 auto C = CompleteAtPoint(
"1");
3909 auto Reqs1 = Requests.consumeRequests(1);
3910 ASSERT_EQ(Reqs1.size(), 1u);
3911 EXPECT_THAT(C, ElementsAre(AllOf(named(
"Food"),
3914 C = CompleteAtPoint(
"2");
3915 auto Reqs2 = Requests.consumeRequests(1);
3918 ASSERT_EQ(Reqs2.size(), 1u);
3919 EXPECT_EQ(Reqs2[0], Reqs1[0]);
3920 EXPECT_THAT(C, ElementsAre(AllOf(named(
"FoodClass"),
3924TEST(CompletionTest, ObjectiveCCategoryFromIndexIgnored) {
3926 auto Results = completions(R
"objc(
3934 EXPECT_THAT(Results.Completions, IsEmpty());
3937TEST(CompletionTest, ObjectiveCForwardDeclFromIndex) {
3939 FoodClass.IncludeHeaders.emplace_back(
"\"Foo.h\"", 2,
Symbol::Import);
3941 auto Results = completions(
"@class Foo^", {SymFood, FoodClass},
3945 EXPECT_THAT(Results.Completions,
3946 UnorderedElementsAre(AllOf(named(
"FoodClass"),
3948 Not(insertInclude()))));
3951TEST(CompletionTest, CursorInSnippets) {
3952 clangd::CodeCompleteOptions Options;
3953 Options.EnableSnippets =
true;
3954 auto Results = completions(
3956 void while_foo(int a, int b);
3963 EXPECT_THAT(Results.Completions,
3964 Contains(AllOf(named(
"while"),
3965 snippetSuffix(
" (${1:condition}) {\n$0\n}"))));
3967 EXPECT_THAT(Results.Completions,
3968 Contains(AllOf(named(
"while_foo"),
3969 snippetSuffix(
"(${1:int a}, ${2:int b})"))));
3971 Results = completions(R
"cpp(
3973 Base(int a, int b) {}
3976 struct Derived : Base {
3983 EXPECT_THAT(Results.Completions,
3984 Contains(AllOf(named(
"Base"),
3985 snippetSuffix(
"(${1:int a}, ${2:int b})"))));
3988TEST(CompletionTest, WorksWithNullType) {
3989 auto R = completions(R
"cpp(
3991 for (auto [loopVar] : y ) { // y has to be unresolved.
3996 EXPECT_THAT(R.Completions, ElementsAre(named("loopVar")));
3999TEST(CompletionTest, UsingDecl) {
4000 const char *Header(R
"cpp(
4005 const char *Source(R
"cpp(
4010 clangd::CodeCompleteOptions Opts;
4011 Opts.Index = Index.get();
4012 Opts.AllScopes =
true;
4013 auto R = completions(Source, {}, Opts);
4014 EXPECT_THAT(R.Completions,
4015 ElementsAre(AllOf(scope(
"std::"), named(
"foo"),
4019TEST(CompletionTest, Enums) {
4020 const char *Header(R
"cpp(
4022 enum Unscoped { Clangd1 };
4024 enum Unscoped { Clangd2 };
4026 enum class Scoped { Clangd3 };
4028 const char *Source(R
"cpp(
4033 clangd::CodeCompleteOptions Opts;
4034 Opts.Index = Index.get();
4035 Opts.AllScopes =
true;
4036 auto R = completions(Source, {}, Opts);
4037 EXPECT_THAT(R.Completions, UnorderedElementsAre(
4038 AllOf(scope(
"ns::"), named(
"Clangd1"),
4040 AllOf(scope(
"ns::C::"), named(
"Clangd2"),
4042 AllOf(scope(
"ns::Scoped::"), named(
"Clangd3"),
4046TEST(CompletionTest, ScopeIsUnresolved) {
4047 clangd::CodeCompleteOptions Opts = {};
4048 Opts.AllScopes =
true;
4050 auto Results = completions(R
"cpp(
4055 {cls("a::b::XYZ")}, Opts);
4056 EXPECT_THAT(Results.Completions,
4057 UnorderedElementsAre(AllOf(qualifier(
""), named(
"XYZ"))));
4060TEST(CompletionTest, NestedScopeIsUnresolved) {
4061 clangd::CodeCompleteOptions Opts = {};
4062 Opts.AllScopes =
true;
4064 auto Results = completions(R
"cpp(
4067 void f() { b::c::X^ }
4070 {cls("a::b::c::XYZ")}, Opts);
4071 EXPECT_THAT(Results.Completions,
4072 UnorderedElementsAre(AllOf(qualifier(
""), named(
"XYZ"))));
4077TEST(CompletionTest, NamespaceDoubleInsertion) {
4078 clangd::CodeCompleteOptions Opts = {};
4080 auto Results = completions(R
"cpp(
4087 {cls("foo::ns::ABCDE")}, Opts);
4088 EXPECT_THAT(Results.Completions,
4089 UnorderedElementsAre(AllOf(qualifier(
""), named(
"ABCDE"))));
4092TEST(CompletionTest, DerivedMethodsAreAlwaysVisible) {
4095 auto Completions = completions(R
"cpp(
4098 double size() const;
4100 struct deque : deque_base {
4107 EXPECT_THAT(Completions,
4108 ElementsAre(AllOf(returnType("int"), named(
"size"))));
4111TEST(CompletionTest, NoCrashWithIncompleteLambda) {
4112 auto Completions = completions(
"auto&& x = []{^").Completions;
4116 EXPECT_THAT(Completions, Contains(named(
"x")));
4118 auto Signatures = signatures(
"auto x() { x(^").signatures;
4119 EXPECT_THAT(Signatures, Contains(sig(
"x() -> auto")));
4122TEST(CompletionTest, DelayedTemplateParsing) {
4125 template <typename T> int foo() { return xx^; }
4130 TU.ExtraArgs.push_back(
"-fdelayed-template-parsing");
4132 EXPECT_THAT(completions(TU, Test.point()).Completions,
4133 Contains(named(
"xxx")));
4136TEST(CompletionTest, CompletionRange) {
4137 const char *WithRange =
"auto x = [[abc]]^";
4138 auto Completions = completions(WithRange);
4139 EXPECT_EQ(Completions.InsertRange,
Annotations(WithRange).range());
4140 Completions = completionsNoCompile(WithRange);
4141 EXPECT_EQ(Completions.InsertRange,
Annotations(WithRange).range());
4143 const char *EmptyRange =
"auto x = [[]]^";
4144 Completions = completions(EmptyRange);
4145 EXPECT_EQ(Completions.InsertRange,
Annotations(EmptyRange).range());
4146 Completions = completionsNoCompile(EmptyRange);
4147 EXPECT_EQ(Completions.InsertRange,
Annotations(EmptyRange).range());
4151 const char *NoCompletion =
"/* foo [[]]^ */";
4152 Completions = completions(NoCompletion);
4153 EXPECT_EQ(Completions.InsertRange, std::nullopt);
4154 Completions = completionsNoCompile(NoCompletion);
4155 EXPECT_EQ(Completions.InsertRange,
Annotations(NoCompletion).range());
4158TEST(CompletionTest, ReplaceRange) {
4159 clangd::CodeCompleteOptions Opts;
4160 Opts.EnableInsertReplace =
true;
4163 const char *EndOfToken =
4164 "struct S { int abc; }; void f() { S s; s.[[abc]]^; }";
4165 CodeCompleteResult Completions =
4166 completions(EndOfToken, {}, Opts);
4168 EXPECT_EQ(Completions.InsertRange,
A.range());
4169 EXPECT_EQ(Completions.ReplaceRange,
A.range());
4172 const char *MidWord =
"struct S { int abcd; }; void f() { S s; "
4173 "s.$replace[[$insert[[ab^]]cd]]; }";
4174 Completions = completions(MidWord, {}, Opts);
4176 EXPECT_EQ(Completions.InsertRange,
A.range(
"insert"));
4177 EXPECT_EQ(Completions.ReplaceRange,
A.range(
"replace"));
4180 const char *EmptyPrefix =
"struct S { int abcd; }; void f() { S s; "
4181 "s.$replace[[$insert[[^]]abcd]]; }";
4182 Completions = completions(EmptyPrefix, {}, Opts);
4184 EXPECT_EQ(Completions.InsertRange,
A.range(
"insert"));
4185 EXPECT_EQ(Completions.ReplaceRange,
A.range(
"replace"));
4188 const char *MidWordUTF8 =
"struct S { int naïve; }; void f() { S s; "
4189 "s.$replace[[$insert[[na^]]ïve]]; }";
4190 Completions = completions(MidWordUTF8, {}, Opts);
4192 EXPECT_EQ(Completions.InsertRange,
A.range(
"insert"));
4193 EXPECT_EQ(Completions.ReplaceRange,
A.range(
"replace"));
4196 const char *BeforeParen =
"struct S { int abcd(); }; void f() { S s; "
4197 "s.$replace[[$insert[[ab^]]cd]](123); }";
4198 Completions = completions(BeforeParen, {}, Opts);
4200 EXPECT_EQ(Completions.InsertRange,
A.range(
"insert"));
4201 EXPECT_EQ(Completions.ReplaceRange,
A.range(
"replace"));
4204 const char *BeforeAngle =
4205 "struct S { template <typename T> int abcd(); }; void f() { S s; "
4206 "s.$replace[[$insert[[ab^]]cd]]<int>(); }";
4207 Completions = completions(BeforeAngle, {}, Opts);
4209 EXPECT_EQ(Completions.InsertRange,
A.range(
"insert"));
4210 EXPECT_EQ(Completions.ReplaceRange,
A.range(
"replace"));
4213 const char *BeforeEquals =
4214 "void f() { int $replace[[$insert[[ab^]]cd]] = 1; }";
4215 Completions = completions(BeforeEquals, {}, Opts);
4217 EXPECT_EQ(Completions.InsertRange,
A.range(
"insert"));
4218 EXPECT_EQ(Completions.ReplaceRange,
A.range(
"replace"));
4221 Opts.EnableInsertReplace =
false;
4222 const char *NoReplace =
"auto x = [[abc]]^";
4223 Completions = completions(NoReplace, {}, Opts);
4224 EXPECT_EQ(Completions.InsertRange,
Annotations(NoReplace).range());
4225 EXPECT_EQ(Completions.ReplaceRange, std::nullopt);
4228TEST(CompletionTest, ReplaceRangeNoCompile) {
4229 clangd::CodeCompleteOptions Opts;
4230 Opts.EnableInsertReplace =
true;
4233 const char *EndOfToken =
"auto x = [[abc]]^";
4235 CodeCompleteResult Results =
4236 completionsNoCompile(EndOfToken, {}, Opts);
4237 EXPECT_EQ(Results.InsertRange,
A.range());
4238 EXPECT_EQ(Results.ReplaceRange,
A.range());
4241 const char *MidWord =
"auto x = $replace[[$insert[[ab^]]cd]]";
4242 Results = completionsNoCompile(MidWord, {}, Opts);
4244 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4245 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4248 const char *EmptyPrefix =
"auto x = $replace[[$insert[[^]]abcd]]";
4249 Results = completionsNoCompile(EmptyPrefix, {}, Opts);
4251 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4252 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4255 const char *MidWordUTF8 =
"auto x = $replace[[$insert[[na^]]]]ïve";
4256 Results = completionsNoCompile(MidWordUTF8, {}, Opts);
4258 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4259 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4262 const char *BeforeParen =
"auto x = $replace[[$insert[[ab^]]cd]](123);";
4263 Results = completionsNoCompile(BeforeParen, {}, Opts);
4265 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4266 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4269 const char *BeforeAngle =
"auto x = $replace[[$insert[[ab^]]cd]]<int>();";
4270 Results = completionsNoCompile(BeforeAngle, {}, Opts);
4272 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4273 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4276 const char *BeforeEquals =
"auto $replace[[$insert[[ab^]]cd]] = 1;";
4277 Results = completionsNoCompile(BeforeEquals, {}, Opts);
4279 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4280 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4283 Opts.EnableInsertReplace =
false;
4284 const char *NoReplace =
"auto x = [[abc]]^";
4285 Results = completionsNoCompile(NoReplace, {}, Opts);
4286 EXPECT_EQ(Results.InsertRange,
Annotations(NoReplace).range());
4287 EXPECT_EQ(Results.ReplaceRange, std::nullopt);
4290TEST(NoCompileCompletionTest, Basic) {
4291 auto Results = completionsNoCompile(R
"cpp(
4298 EXPECT_FALSE(Results.RanParser);
4299 EXPECT_THAT(Results.Completions,
4300 UnorderedElementsAre(named("void"), named(
"func"), named(
"int"),
4301 named(
"xyz"), named(
"abc")));
4304TEST(NoCompileCompletionTest, WithFilter) {
4305 auto Results = completionsNoCompile(R
"cpp(
4314 EXPECT_THAT(Results.Completions,
4315 UnorderedElementsAre(named("sym1"), named(
"sym2")));
4318TEST(NoCompileCompletionTest, WithIndex) {
4319 std::vector<Symbol> Syms = {
func(
"xxx"),
func(
"a::xxx"),
func(
"ns::b::xxx"),
4320 func(
"c::xxx"),
func(
"ns::d::xxx")};
4321 auto Results = completionsNoCompile(
4323 // Current-scopes, unqualified completion.
4333 EXPECT_THAT(Results.Completions,
4334 UnorderedElementsAre(AllOf(qualifier(""), scope(
"")),
4335 AllOf(qualifier(
""), scope(
"a::")),
4336 AllOf(qualifier(
""), scope(
"ns::b::"))));
4338 Opts.AllScopes =
true;
4339 Results = completionsNoCompile(
4341 // All-scopes unqualified completion.
4351 EXPECT_THAT(Results.Completions,
4352 UnorderedElementsAre(AllOf(qualifier(""), scope(
"")),
4353 AllOf(qualifier(
""), scope(
"a::")),
4354 AllOf(qualifier(
""), scope(
"ns::b::")),
4355 AllOf(qualifier(
"c::"), scope(
"c::")),
4356 AllOf(qualifier(
"d::"), scope(
"ns::d::"))));
4357 Results = completionsNoCompile(
4359 // Qualified completion.
4369 EXPECT_THAT(Results.Completions,
4370 ElementsAre(AllOf(qualifier(""), scope(
"ns::b::"))));
4371 Results = completionsNoCompile(
4373 // Absolutely qualified completion.
4383 EXPECT_THAT(Results.Completions,
4384 ElementsAre(AllOf(qualifier(""), scope(
"a::"))));
4387TEST(AllowImplicitCompletion,
All) {
4388 const char *
Yes[] = {
4392 " # include <^foo.h>",
4393 "#import <foo/^bar.h>",
4394 "#include_next \"^",
4396 const char *
No[] = {
4400 "#include <foo.h> //^",
4401 "#include \"foo.h\"^",
4405 for (
const char *Test :
Yes) {
4406 llvm::Annotations
A(Test);
4409 for (
const char *Test :
No) {
4410 llvm::Annotations
A(Test);
4415TEST(CompletionTest, FunctionArgsExist) {
4416 clangd::CodeCompleteOptions Opts;
4417 Opts.EnableSnippets =
true;
4425 template <typename T>
4427 Container(int Size) {}
4430 EXPECT_THAT(completions(Context + "int y = fo^", {}, Opts).Completions,
4431 UnorderedElementsAre(
4432 AllOf(labeled(
"foo(int A)"), snippetSuffix(
"(${1:int A})"))));
4434 completions(
Context +
"int y = fo^(42)", {}, Opts).Completions,
4435 UnorderedElementsAre(AllOf(labeled(
"foo(int A)"), snippetSuffix(
""))));
4437 completions(
Context +
"int y = fo^o(42)", {}, Opts).Completions,
4438 UnorderedElementsAre(AllOf(labeled(
"foo(int A)"), snippetSuffix(
""))));
4440 completions(
Context +
"int y = ba^", {}, Opts).Completions,
4441 UnorderedElementsAre(AllOf(labeled(
"bar()"), snippetSuffix(
"()"))));
4442 EXPECT_THAT(completions(
Context +
"int y = ba^()", {}, Opts).Completions,
4443 UnorderedElementsAre(AllOf(labeled(
"bar()"), snippetSuffix(
""))));
4444 EXPECT_THAT(completions(
Context +
"int y = ba^r()", {}, Opts).Completions,
4445 UnorderedElementsAre(AllOf(labeled(
"bar()"), snippetSuffix(
""))));
4447 completions(
Context +
"Object o = Obj^", {}, Opts).Completions,
4448 Contains(AllOf(labeled(
"Object(int B)"), snippetSuffix(
"(${1:int B})"),
4450 EXPECT_THAT(completions(
Context +
"Object o = Obj^()", {}, Opts).Completions,
4451 Contains(AllOf(labeled(
"Object(int B)"), snippetSuffix(
""),
4454 completions(
Context +
"Container c = Cont^", {}, Opts).Completions,
4455 Contains(AllOf(labeled(
"Container<typename T>(int Size)"),
4456 snippetSuffix(
"<${1:typename T}>(${2:int Size})"),
4459 completions(
Context +
"Container c = Cont^()", {}, Opts).Completions,
4460 Contains(AllOf(labeled(
"Container<typename T>(int Size)"),
4461 snippetSuffix(
"<${1:typename T}>"),
4464 completions(
Context +
"Container c = Cont^<int>()", {}, Opts).Completions,
4465 Contains(AllOf(labeled(
"Container<typename T>(int Size)"),
4469 completions(
Context +
"Container c = Cont^ainer()", {}, Opts).Completions,
4470 Contains(AllOf(labeled(
"Container<typename T>(int Size)"),
4471 snippetSuffix(
"<${1:typename T}>"),
4473 EXPECT_THAT(completions(
Context +
"MAC^(2)", {}, Opts).Completions,
4474 Contains(AllOf(labeled(
"MACRO(x)"), snippetSuffix(
""),
4476 EXPECT_THAT(completions(
Context +
"MAC^RO(2)", {}, Opts).Completions,
4477 Contains(AllOf(labeled(
"MACRO(x)"), snippetSuffix(
""),
4481TEST(CompletionTest, FunctionArgsExist_Issue1785) {
4485 clangd::CodeCompleteOptions Opts;
4486 Opts.EnableSnippets =
true;
4488 std::string Code = R
"cpp(
4500 completions(Code, {}, Opts).Completions,
4501 Contains(AllOf(labeled("waldo(int)"), snippetSuffix(
"(${1:int})"))));
4504TEST(CompletionTest, NoCrashDueToMacroOrdering) {
4505 EXPECT_THAT(completions(R
"cpp(
4507 #define ECHO2(X) ECHO(X)
4508 int finish_preamble = EC^HO(2);)cpp")
4510 UnorderedElementsAre(labeled("ECHO(X)"), labeled(
"ECHO2(X)")));
4513TEST(CompletionTest, ObjCCategoryDecls) {
4515 TU.
ExtraArgs.push_back(
"-xobjective-c");
4516 TU.HeaderCode = R
"objc(
4520 @interface Foo (FooExt1)
4523 @interface Foo (FooExt2)
4529 @interface Bar (BarExt)
4534 @implementation Foo (^)
4537 TU.Code = Test.code().str();
4538 auto Results = completions(TU, Test.point());
4539 EXPECT_THAT(Results.Completions,
4540 UnorderedElementsAre(labeled(
"FooExt1"), labeled(
"FooExt2")));
4547 TU.Code = Test.code().str();
4548 auto Results = completions(TU, Test.point());
4549 EXPECT_THAT(Results.Completions, UnorderedElementsAre(labeled(
"BarExt")));
4553TEST(CompletionTest, PreambleCodeComplete) {
4554 llvm::StringLiteral Baseline =
"\n#define MACRO 12\nint num = MACRO;";
4555 llvm::StringLiteral ModifiedCC =
4556 "#include \"header.h\"\n#define MACRO 12\nint num = MACRO; int num2 = M^";
4563 auto Inputs = ModifiedTU.inputs(FS);
4565 BaselineTU.preamble().get(), Inputs, {});
4566 EXPECT_THAT(Result.Completions, Not(testing::IsEmpty()));
4569TEST(CompletionTest, CommentParamName) {
4570 const std::string Code = R
"cpp(
4571 void fun(int foo, int bar);
4572 void overloaded(int param_int);
4573 void overloaded(int param_int, int param_other);
4574 void overloaded(char param_char);
4578 EXPECT_THAT(completions(Code + "fun(/*^").Completions,
4579 UnorderedElementsAre(labeled(
"foo=*/")));
4580 EXPECT_THAT(completions(Code +
"fun(1, /*^").Completions,
4581 UnorderedElementsAre(labeled(
"bar=*/")));
4582 EXPECT_THAT(completions(Code +
"/*^").Completions, IsEmpty());
4585 completions(Code +
"overloaded(/*^").Completions,
4586 UnorderedElementsAre(labeled(
"param_int=*/"), labeled(
"param_char=*/")));
4588 EXPECT_THAT(completions(Code +
"fun(/* ^").Completions,
4589 UnorderedElementsAre(labeled(
"foo=*/")));
4590 EXPECT_THAT(completions(Code +
"fun(/* f^").Completions,
4591 UnorderedElementsAre(labeled(
"foo=*/")));
4592 EXPECT_THAT(completions(Code +
"fun(/* x^").Completions, IsEmpty());
4593 EXPECT_THAT(completions(Code +
"fun(/* f ^").Completions, IsEmpty());
4597 std::string CompletionRangeTest(Code +
"fun(/*[[^]]");
4598 auto Results = completions(CompletionRangeTest);
4599 EXPECT_THAT(Results.InsertRange,
4600 llvm::ValueIs(
Annotations(CompletionRangeTest).range()));
4602 Results.Completions,
4604 AllOf(replacesRange(
Annotations(CompletionRangeTest).range()),
4608 std::string CompletionRangeTest(Code +
"fun(/*[[fo^]]");
4609 auto Results = completions(CompletionRangeTest);
4610 EXPECT_THAT(Results.InsertRange,
4611 llvm::ValueIs(
Annotations(CompletionRangeTest).range()));
4613 Results.Completions,
4615 AllOf(replacesRange(
Annotations(CompletionRangeTest).range()),
4620 clangd::CodeCompleteOptions ReplaceOpts;
4621 ReplaceOpts.EnableInsertReplace =
true;
4624 const std::string NoEquals(Code +
"fun(/*$replace[[$insert[[fo^]]o*/]])");
4625 const CodeCompleteResult Results = completions(NoEquals, {}, ReplaceOpts);
4627 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4628 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4632 const std::string WithEquals(Code +
4633 "fun(/*$replace[[$insert[[fo^]]o=*/]])");
4634 const CodeCompleteResult Results = completions(WithEquals, {}, ReplaceOpts);
4636 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4637 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4641 const std::string NoClose(Code +
"fun(/*[[fo^]]");
4642 const CodeCompleteResult Results = completions(NoClose, {}, ReplaceOpts);
4644 EXPECT_EQ(Results.InsertRange,
A.range());
4645 EXPECT_EQ(Results.ReplaceRange,
A.range());
4649 const std::string WithUTF8(Code +
"fun(/*$replace[[$insert[[ca^]]fé=*/]])");
4650 const CodeCompleteResult Results = completions(WithUTF8, {}, ReplaceOpts);
4652 EXPECT_EQ(Results.InsertRange,
A.range(
"insert"));
4653 EXPECT_EQ(Results.ReplaceRange,
A.range(
"replace"));
4657TEST(CompletionTest, Concepts) {
4660 concept A = sizeof(T) <= 8;
4662 template<$tparam^A U>
4665 template<typename T>
4666 int bar(T t) requires $expr^A<int>;
4669 concept b = $expr^A && $expr^sizeof(T) % 2 == 0 || $expr^A && sizeof(T) == 1;
4671 $toplevel^A auto i = 19;
4673 template<$toplevel^A auto i> void constrainedNTTP();
4675 // FIXME: The first parameter should be dropped in this case.
4676 void abbreviated($expr^A auto x) {}
4679 TU.Code = Code.code().str();
4680 TU.ExtraArgs = {"-std=c++20"};
4683 Sym.Signature =
"<typename Tp, typename Up>";
4684 Sym.CompletionSnippetSuffix =
"<${1:typename Tp}, ${2:typename Up}>";
4685 std::vector<Symbol> Syms = {Sym};
4686 for (
auto P : Code.points(
"tparam")) {
4688 completions(TU, P, Syms).Completions,
4689 AllOf(Contains(AllOf(named(
"A"), signature(
""), snippetSuffix(
""))),
4690 Contains(AllOf(named(
"same_as"), signature(
"<typename Up>"),
4691 snippetSuffix(
"<${2:typename Up}>"))),
4692 Contains(named(
"class")), Contains(named(
"typename"))))
4693 <<
"Completing template parameter at position " <<
P;
4696 for (
auto P : Code.points(
"toplevel")) {
4698 completions(TU, P, Syms).Completions,
4699 AllOf(Contains(AllOf(named(
"A"), signature(
""), snippetSuffix(
""))),
4700 Contains(AllOf(named(
"same_as"), signature(
"<typename Up>"),
4701 snippetSuffix(
"<${2:typename Up}>")))))
4702 <<
"Completing 'requires' expression at position " <<
P;
4705 for (
auto P : Code.points(
"expr")) {
4707 completions(TU, P, Syms).Completions,
4708 AllOf(Contains(AllOf(named(
"A"), signature(
"<class T>"),
4709 snippetSuffix(
"<${1:class T}>"))),
4711 named(
"same_as"), signature(
"<typename Tp, typename Up>"),
4712 snippetSuffix(
"<${1:typename Tp}, ${2:typename Up}>")))))
4713 <<
"Completing 'requires' expression at position " <<
P;
4719 // Comment `with` markup.
4721 void bar() { foo(^); }
4723 for (
auto DocumentationFormat :
4725 auto Sigs = signatures(Code.code(), Code.point(), {},
4726 DocumentationFormat);
4727 ASSERT_EQ(Sigs.signatures.size(), 1U);
4728 EXPECT_EQ(Sigs.signatures[0].documentation.kind, DocumentationFormat);
4733 std::string Top = R
"cpp(
4734 template <typename T, int> bool foo(char);
4735 template <int I, int> bool foo(float);
4738 auto First = signatures(Top +
"bool x = foo<^");
4741 UnorderedElementsAre(sig(
"foo<[[typename T]], [[int]]>() -> bool"),
4742 sig(
"foo<[[int I]], [[int]]>() -> bool")));
4743 EXPECT_EQ(First.activeParameter, 0);
4745 auto Second = signatures(Top +
"bool x = foo<1, ^");
4746 EXPECT_THAT(Second.signatures,
4747 ElementsAre(sig(
"foo<[[int I]], [[int]]>() -> bool")));
4748 EXPECT_EQ(Second.activeParameter, 1);
4751TEST(CompletionTest, DoNotCrash) {
4752 llvm::StringLiteral Cases[] = {
4754 template <typename = int> struct Foo {};
4755 auto a = [x(3)](Foo<^>){};
4758 for (
auto Case : Cases) {
4760 auto Completions = completions(Case);
4763TEST(CompletionTest, PreambleFromDifferentTarget) {
4764 constexpr std::string_view PreambleTarget =
"x86_64";
4765 constexpr std::string_view Contents =
4766 "int foo(int); int num; int num2 = foo(n^";
4770 TU.ExtraArgs.emplace_back(
"-target");
4771 TU.ExtraArgs.emplace_back(PreambleTarget);
4775 TU.ExtraArgs.pop_back();
4776 TU.ExtraArgs.emplace_back(
"wasm32");
4779 auto Inputs = TU.inputs(FS);
4786 EXPECT_THAT(Result.Completions, Not(testing::IsEmpty()));
4787 EXPECT_THAT(Signatures.signatures, Not(testing::IsEmpty()));
4790TEST(CompletionTest, SkipExplicitObjectParameter) {
4793 void foo(this auto&& self, int arg);
4794 void bar(this A self, int arg);
4813 TU.ExtraArgs = {
"-std=c++23"};
4821 auto Inputs = TU.inputs(FS);
4826 EXPECT_THAT(Result.Completions,
4827 UnorderedElementsAre(AllOf(named(
"foo"), signature(
"(int arg)"),
4828 snippetSuffix(
"(${1:int arg})")),
4829 AllOf(named(
"bar"), signature(
"(int arg)"),
4830 snippetSuffix(
"(${1:int arg})"))));
4837 ElementsAre(AllOf(named(
"foo"), signature(
"<class self:auto>(int arg)"),
4838 snippetSuffix(
"<${1:class self:auto}>"))));
4843 EXPECT_THAT(Result.Completions,
4844 ElementsAre(AllOf(named(
"bar"), signature(
"(int arg)"),
4845 snippetSuffix(
""))));
4849TEST(CompletionTest, MemberAccessInExplicitObjMemfn) {
4853 int memberFnA(int a);
4854 int memberFnA(this A&, float a);
4856 void foo(this A& self) {
4857 // Should not offer any members here, since
4858 // it needs to be referenced through `self`.
4860 // should offer all results
4864 // should not offer any results
4872 TU.ExtraArgs = {
"-std=c++23"};
4880 auto Inputs = TU.inputs(FS);
4886 EXPECT_THAT(Result.Completions, ElementsAre());
4894 UnorderedElementsAre(named(
"member"),
4895 AllOf(named(
"memberFnA"), signature(
"(int a)"),
4896 snippetSuffix(
"(${1:int a})")),
4897 AllOf(named(
"memberFnA"), signature(
"(float a)"),
4898 snippetSuffix(
"(${1:float a})"))));
4904 EXPECT_THAT(Result.Completions, ElementsAre());
4908TEST(CompletionTest, ListExplicitObjectOverloads) {
4912 void foo2(int a) const;
4913 void foo2(this const S& self, float a);
4914 void foo3(this const S& self, int a);
4915 void foo4(this S& self, int a);
4918 void S::foo1(int a) {
4922 void S::foo2(int a) const {
4926 void S::foo3(this const S& self, int a) {
4930 void S::foo4(this S& self, int a) {
4938 void test2(const S s) {
4944 TU.ExtraArgs = {
"-std=c++23"};
4952 auto Inputs = TU.inputs(FS);
4959 UnorderedElementsAre(AllOf(named(
"foo1"), signature(
"(int a)"),
4960 snippetSuffix(
"(${1:int a})")),
4961 AllOf(named(
"foo2"), signature(
"(int a) const"),
4962 snippetSuffix(
"(${1:int a})")),
4963 AllOf(named(
"foo2"), signature(
"(float a) const"),
4964 snippetSuffix(
"(${1:float a})")),
4965 AllOf(named(
"foo3"), signature(
"(int a) const"),
4966 snippetSuffix(
"(${1:int a})")),
4967 AllOf(named(
"foo4"), signature(
"(int a)"),
4968 snippetSuffix(
"(${1:int a})"))));
4975 UnorderedElementsAre(AllOf(named(
"foo2"), signature(
"(int a) const"),
4976 snippetSuffix(
"(${1:int a})")),
4977 AllOf(named(
"foo2"), signature(
"(float a) const"),
4978 snippetSuffix(
"(${1:float a})")),
4979 AllOf(named(
"foo3"), signature(
"(int a) const"),
4980 snippetSuffix(
"(${1:int a})"))));
4987 UnorderedElementsAre(AllOf(named(
"foo2"), signature(
"(int a) const"),
4988 snippetSuffix(
"(${1:int a})")),
4989 AllOf(named(
"foo2"), signature(
"(float a) const"),
4990 snippetSuffix(
"(${1:float a})")),
4991 AllOf(named(
"foo3"), signature(
"(int a) const"),
4992 snippetSuffix(
"(${1:int a})"))));
4999 UnorderedElementsAre(AllOf(named(
"foo1"), signature(
"(int a)"),
5000 snippetSuffix(
"(${1:int a})")),
5001 AllOf(named(
"foo2"), signature(
"(int a) const"),
5002 snippetSuffix(
"(${1:int a})")),
5003 AllOf(named(
"foo2"), signature(
"(float a) const"),
5004 snippetSuffix(
"(${1:float a})")),
5005 AllOf(named(
"foo3"), signature(
"(int a) const"),
5006 snippetSuffix(
"(${1:int a})")),
5007 AllOf(named(
"foo4"), signature(
"(int a)"),
5008 snippetSuffix(
"(${1:int a})"))));
5015 UnorderedElementsAre(AllOf(named(
"foo1"), signature(
"(int a)"),
5016 snippetSuffix(
"(${1:int a})")),
5017 AllOf(named(
"foo2"), signature(
"(int a) const"),
5018 snippetSuffix(
"(${1:int a})")),
5019 AllOf(named(
"foo2"), signature(
"(float a) const"),
5020 snippetSuffix(
"(${1:float a})")),
5021 AllOf(named(
"foo3"), signature(
"(int a) const"),
5022 snippetSuffix(
"(${1:int a})")),
5023 AllOf(named(
"foo4"), signature(
"(int a)"),
5024 snippetSuffix(
"(${1:int a})"))));
5031 UnorderedElementsAre(AllOf(named(
"foo2"), signature(
"(int a) const"),
5032 snippetSuffix(
"(${1:int a})")),
5033 AllOf(named(
"foo2"), signature(
"(float a) const"),
5034 snippetSuffix(
"(${1:float a})")),
5035 AllOf(named(
"foo3"), signature(
"(int a) const"),
5036 snippetSuffix(
"(${1:int a})"))));
5040TEST(CompletionTest, FuzzyMatchMacro) {
5043 #define _gl_foo() 42
5063 auto Results = completions(TU, Code.point(
"c1"), {}, Opts);
5065 Results.Completions,
5066 ElementsAre(named(
"gl_frob"), named(
"_gl_frob"), named(
"glfbar")));
5070 auto Results = completions(TU, Code.point(
"c2"), {}, Opts);
5071 EXPECT_THAT(Results.Completions,
5072 ElementsAre(named(
"_gl_frob"), named(
"_gl_foo")));
5083 auto Results = completions(TU, Code.point(
"c1"), {}, Opts);
5084 EXPECT_THAT(Results.Completions,
5085 ElementsAre(named(
"gl_frob"), named(
"_gl_frob"),
5086 named(
"glfbar"), named(
"gl_foo")));
5091 auto Results = completions(TU, Code.point(
"c2"), {}, Opts);
5092 EXPECT_THAT(Results.Completions,
5093 ElementsAre(named(
"_gl_frob"), named(
"_gl_foo")));
#define EXPECT_IFF(condition, value, matcher)
Same as llvm::Annotations, but adjusts functions to LSP-specific types for positions and ranges.
clangd::Range range(llvm::StringRef Name="") const
Manages a collection of source files and derived data (ASTs, indexes), and provides language-aware fe...
static Options optsForTest()
A context is an immutable container for per-request data that must be propagated through layers that ...
static std::unique_ptr< SymbolIndex > build(SymbolSlab Symbols, RefSlab Refs, RelationSlab Relations)
Builds an index from slabs. The index takes ownership of the data.
llvm::StringMap< std::string > Files
An efficient structure of storing large set of symbol references in memory.
Interface for symbol indexes that can be used for searching or matching symbols among a set of symbol...
SymbolSlab::Builder is a mutable container that can 'freeze' to SymbolSlab.
void insert(const Symbol &S)
Adds a symbol, overwriting any existing one with the same ID.
static llvm::Expected< URI > create(llvm::StringRef AbsolutePath, llvm::StringRef Scheme)
Creates a URI for a file in the given scheme.
WithContextValue extends Context::current() with a single value.
FIXME: Skip testing on windows temporarily due to the different escaping code mode.
Symbol objcProtocol(llvm::StringRef Name)
Symbol objcClass(llvm::StringRef Name)
Symbol func(llvm::StringRef Name)
Symbol cls(llvm::StringRef Name)
Symbol objcCategory(llvm::StringRef Name, llvm::StringRef CategoryName)
std::string sortText(float Score, llvm::StringRef Name)
Returns a string that sorts in the same order as (-Score, Tiebreak), for LSP.
Symbol conceptSym(llvm::StringRef Name)
IndexContents
Describes what data is covered by an index.
size_t lspLength(llvm::StringRef Code)
CompletionPrefix guessCompletionPrefix(llvm::StringRef Content, unsigned Offset)
std::unique_ptr< CompilerInvocation > buildCompilerInvocation(const ParseInputs &Inputs, clang::DiagnosticConsumer &D, std::vector< std::string > *CC1Args)
Builds compiler invocation that could be used to build AST or preamble.
Symbol sym(llvm::StringRef QName, index::SymbolKind Kind, llvm::StringRef USRFormat, llvm::StringRef Signature)
CompletionItemKind
The kind of a completion entry.
Symbol ns(llvm::StringRef Name)
bool allowImplicitCompletion(llvm::StringRef Content, unsigned Offset)
llvm::unique_function< void(llvm::Expected< T >)> Callback
A Callback<T> is a void function that accepts Expected<T>.
llvm::raw_ostream & operator<<(llvm::raw_ostream &OS, const CodeCompletion &C)
llvm::Expected< CodeCompleteResult > runCodeComplete(ClangdServer &Server, PathRef File, Position Pos, clangd::CodeCompleteOptions Opts)
llvm::Expected< SignatureHelp > runSignatureHelp(ClangdServer &Server, PathRef File, Position Pos, MarkupKind DocumentationFormat)
std::string testPath(PathRef File, llvm::sys::path::Style Style)
std::shared_ptr< const PreambleData > buildPreamble(PathRef FileName, CompilerInvocation CI, const ParseInputs &Inputs, bool StoreInMemory, PreambleParsedCallback PreambleCallback, PreambleBuildStats *Stats)
Build a preamble for the new inputs unless an old one can be reused.
TEST(BackgroundQueueTest, Priority)
void wait(std::unique_lock< std::mutex > &Lock, std::condition_variable &CV, Deadline D)
Wait once on CV for the specified duration.
llvm::Expected< size_t > positionToOffset(llvm::StringRef Code, Position P, bool AllowColumnsBeyondLineLength)
Turn a [line, column] pair into an offset in Code.
void runAddDocument(ClangdServer &Server, PathRef File, llvm::StringRef Contents, llvm::StringRef Version, WantDiagnostics WantDiags, bool ForceRebuild)
Symbol macro(llvm::StringRef Name, llvm::StringRef ArgList)
llvm::StringRef PathRef
A typedef to represent a ref to file path.
@ No
Diagnostics must be generated for this snapshot.
std::vector< std::string > lookup(const SymbolIndex &I, llvm::ArrayRef< SymbolID > IDs)
Symbol enmConstant(llvm::StringRef Name)
Deadline timeoutSeconds(std::optional< double > Seconds)
Makes a deadline from a timeout in seconds. std::nullopt means wait forever.
CodeCompleteResult codeComplete(PathRef FileName, Position Pos, const PreambleData *Preamble, const ParseInputs &ParseInput, CodeCompleteOptions Opts, SpeculativeFuzzyFind *SpecFuzzyFind)
Gets code completions at a specified Pos in FileName.
@ PlainText
The primary text to be inserted is treated as a plain string.
@ Snippet
The primary text to be inserted is treated as a snippet.
SignatureHelp signatureHelp(PathRef FileName, Position Pos, const PreambleData &Preamble, const ParseInputs &ParseInput, MarkupKind DocumentationFormat)
Get signature help at a specified Pos in FileName.
PolySubsequenceMatcher< Args... > HasSubsequence(Args &&... M)
Symbol var(llvm::StringRef Name)
cppcoreguidelines::ProBoundsAvoidUncheckedContainerAccessCheck P
===– Representation.cpp - ClangDoc Representation --------—*- C++ -*-===//
Signals derived from a valid AST of a file.
llvm::DenseMap< SymbolID, unsigned > ReferencedSymbols
Number of occurrences of each symbol present in the file.
Symbol::IncludeDirective InsertionDirective
Preferred preprocessor directive to use for inclusions by the file.
Represents a collection of completion items to be presented in the editor.
Settings that express user/project preferences and control clangd behavior.
static clangd::Key< Config > Key
Context key which can be used to set the current Config.
@ None
nothing, no argument list and also NO Delimiters "()" or "<>".
@ Delimiters
empty pair of delimiters "()" or "<>".
@ OpenDelimiter
open, only opening delimiter "(" or "<".
Same semantics as CodeComplete::Score.
std::vector< std::string > Scopes
If this is non-empty, symbols must be in at least one of the scopes (e.g.
int line
Line position in a document (zero-based).
int character
Character offset on a line in a document (zero-based).
Position end
The range's end position.
Represents the signature of a callable.
Attributes of a symbol that affect how much we like it.
Attributes of a symbol-query pair that affect how much we like it.
The class presents a C++ symbol, e.g.
@ IndexedForCodeCompletion
Whether or not this symbol is meant to be used for the code completion.
@ Include
#include "header.h"
@ Import
#import "header.h"
SymbolID ID
The ID of the symbol.
std::vector< std::string > ExtraArgs
static TestTU withHeaderCode(llvm::StringRef HeaderCode)
static TestTU withCode(llvm::StringRef Code)
llvm::StringMap< std::string > AdditionalFiles
std::unique_ptr< SymbolIndex > index() const
Range range
The range of the text document to be manipulated.