215 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			215 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===-- CodeCompletionStringsTests.cpp --------------------------*- C++ -*-===//
 | |
| //
 | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | |
| // See https://llvm.org/LICENSE.txt for license information.
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "CodeCompletionStrings.h"
 | |
| #include "TestTU.h"
 | |
| #include "clang/Sema/CodeCompleteConsumer.h"
 | |
| #include "gmock/gmock.h"
 | |
| #include "gtest/gtest.h"
 | |
| 
 | |
| namespace clang {
 | |
| namespace clangd {
 | |
| namespace {
 | |
| 
 | |
| class CompletionStringTest : public ::testing::Test {
 | |
| public:
 | |
|   CompletionStringTest()
 | |
|       : Allocator(std::make_shared<clang::GlobalCodeCompletionAllocator>()),
 | |
|         CCTUInfo(Allocator), Builder(*Allocator, CCTUInfo) {}
 | |
| 
 | |
| protected:
 | |
|   void computeSignature(const CodeCompletionString &CCS,
 | |
|                         bool CompletingPattern = false) {
 | |
|     Signature.clear();
 | |
|     Snippet.clear();
 | |
|     getSignature(CCS, &Signature, &Snippet, /*RequiredQualifier=*/nullptr,
 | |
|                  CompletingPattern);
 | |
|   }
 | |
| 
 | |
|   std::shared_ptr<clang::GlobalCodeCompletionAllocator> Allocator;
 | |
|   CodeCompletionTUInfo CCTUInfo;
 | |
|   CodeCompletionBuilder Builder;
 | |
|   std::string Signature;
 | |
|   std::string Snippet;
 | |
| };
 | |
| 
 | |
| TEST_F(CompletionStringTest, ReturnType) {
 | |
|   Builder.AddResultTypeChunk("result");
 | |
|   Builder.AddResultTypeChunk("redundant result no no");
 | |
|   EXPECT_EQ(getReturnType(*Builder.TakeString()), "result");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, Documentation) {
 | |
|   Builder.addBriefComment("This is ignored");
 | |
|   EXPECT_EQ(formatDocumentation(*Builder.TakeString(), "Is this brief?"),
 | |
|             "Is this brief?");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, DocumentationWithAnnotation) {
 | |
|   Builder.addBriefComment("This is ignored");
 | |
|   Builder.AddAnnotation("Ano");
 | |
|   EXPECT_EQ(formatDocumentation(*Builder.TakeString(), "Is this brief?"),
 | |
|             "Annotation: Ano\n\nIs this brief?");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, GetDeclCommentBadUTF8) {
 | |
|   // <ff> is not a valid byte here, should be replaced by encoded <U+FFFD>.
 | |
|   auto TU = TestTU::withCode("/*x\xffy*/ struct X;");
 | |
|   auto AST = TU.build();
 | |
|   EXPECT_EQ("x\xef\xbf\xbdy",
 | |
|             getDeclComment(AST.getASTContext(), findDecl(AST, "X")));
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, MultipleAnnotations) {
 | |
|   Builder.AddAnnotation("Ano1");
 | |
|   Builder.AddAnnotation("Ano2");
 | |
|   Builder.AddAnnotation("Ano3");
 | |
| 
 | |
|   EXPECT_EQ(formatDocumentation(*Builder.TakeString(), ""),
 | |
|             "Annotations: Ano1 Ano2 Ano3\n");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, EmptySignature) {
 | |
|   Builder.AddTypedTextChunk("X");
 | |
|   Builder.AddResultTypeChunk("result no no");
 | |
|   computeSignature(*Builder.TakeString());
 | |
|   EXPECT_EQ(Signature, "");
 | |
|   EXPECT_EQ(Snippet, "");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, Function) {
 | |
|   Builder.AddResultTypeChunk("result no no");
 | |
|   Builder.addBriefComment("This comment is ignored");
 | |
|   Builder.AddTypedTextChunk("Foo");
 | |
|   Builder.AddChunk(CodeCompletionString::CK_LeftParen);
 | |
|   Builder.AddPlaceholderChunk("p1");
 | |
|   Builder.AddChunk(CodeCompletionString::CK_Comma);
 | |
|   Builder.AddPlaceholderChunk("p2");
 | |
|   Builder.AddChunk(CodeCompletionString::CK_RightParen);
 | |
| 
 | |
|   auto *CCS = Builder.TakeString();
 | |
|   computeSignature(*CCS);
 | |
|   EXPECT_EQ(Signature, "(p1, p2)");
 | |
|   EXPECT_EQ(Snippet, "(${1:p1}, ${2:p2})");
 | |
|   EXPECT_EQ(formatDocumentation(*CCS, "Foo's comment"), "Foo's comment");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, FunctionWithDefaultParams) {
 | |
|   // return_type foo(p1, p2 = 0, p3 = 0)
 | |
|   Builder.AddChunk(CodeCompletionString::CK_Comma);
 | |
|   Builder.AddTypedTextChunk("p3 = 0");
 | |
|   auto *DefaultParam2 = Builder.TakeString();
 | |
| 
 | |
|   Builder.AddChunk(CodeCompletionString::CK_Comma);
 | |
|   Builder.AddTypedTextChunk("p2 = 0");
 | |
|   Builder.AddOptionalChunk(DefaultParam2);
 | |
|   auto *DefaultParam1 = Builder.TakeString();
 | |
| 
 | |
|   Builder.AddResultTypeChunk("return_type");
 | |
|   Builder.AddTypedTextChunk("Foo");
 | |
|   Builder.AddChunk(CodeCompletionString::CK_LeftParen);
 | |
|   Builder.AddPlaceholderChunk("p1");
 | |
|   Builder.AddOptionalChunk(DefaultParam1);
 | |
|   Builder.AddChunk(CodeCompletionString::CK_RightParen);
 | |
| 
 | |
|   auto *CCS = Builder.TakeString();
 | |
|   computeSignature(*CCS);
 | |
|   EXPECT_EQ(Signature, "(p1, p2 = 0, p3 = 0)");
 | |
|   EXPECT_EQ(Snippet, "(${1:p1})");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, EscapeSnippet) {
 | |
|   Builder.AddTypedTextChunk("Foo");
 | |
|   Builder.AddChunk(CodeCompletionString::CK_LeftParen);
 | |
|   Builder.AddPlaceholderChunk("$p}1\\");
 | |
|   Builder.AddChunk(CodeCompletionString::CK_RightParen);
 | |
| 
 | |
|   computeSignature(*Builder.TakeString());
 | |
|   EXPECT_EQ(Signature, "($p}1\\)");
 | |
|   EXPECT_EQ(Snippet, "(${1:\\$p\\}1\\\\})");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, SnippetsInPatterns) {
 | |
|   auto MakeCCS = [this]() -> const CodeCompletionString & {
 | |
|     CodeCompletionBuilder Builder(*Allocator, CCTUInfo);
 | |
|     Builder.AddTypedTextChunk("namespace");
 | |
|     Builder.AddChunk(CodeCompletionString::CK_HorizontalSpace);
 | |
|     Builder.AddPlaceholderChunk("name");
 | |
|     Builder.AddChunk(CodeCompletionString::CK_Equal);
 | |
|     Builder.AddPlaceholderChunk("target");
 | |
|     Builder.AddChunk(CodeCompletionString::CK_SemiColon);
 | |
|     return *Builder.TakeString();
 | |
|   };
 | |
|   computeSignature(MakeCCS(), /*CompletingPattern=*/false);
 | |
|   EXPECT_EQ(Snippet, " ${1:name} = ${2:target};");
 | |
| 
 | |
|   // When completing a pattern, the last placeholder holds the cursor position.
 | |
|   computeSignature(MakeCCS(), /*CompletingPattern=*/true);
 | |
|   EXPECT_EQ(Snippet, " ${1:name} = ${0:target};");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, IgnoreInformativeQualifier) {
 | |
|   Builder.AddTypedTextChunk("X");
 | |
|   Builder.AddInformativeChunk("info ok");
 | |
|   Builder.AddInformativeChunk("info no no::");
 | |
|   computeSignature(*Builder.TakeString());
 | |
|   EXPECT_EQ(Signature, "info ok");
 | |
|   EXPECT_EQ(Snippet, "");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, ObjectiveCMethodNoArguments) {
 | |
|   Builder.AddResultTypeChunk("void");
 | |
|   Builder.AddTypedTextChunk("methodName");
 | |
| 
 | |
|   auto *CCS = Builder.TakeString();
 | |
|   computeSignature(*CCS);
 | |
|   EXPECT_EQ(Signature, "");
 | |
|   EXPECT_EQ(Snippet, "");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, ObjectiveCMethodOneArgument) {
 | |
|   Builder.AddResultTypeChunk("void");
 | |
|   Builder.AddTypedTextChunk("methodWithArg:");
 | |
|   Builder.AddPlaceholderChunk("(type)");
 | |
| 
 | |
|   auto *CCS = Builder.TakeString();
 | |
|   computeSignature(*CCS);
 | |
|   EXPECT_EQ(Signature, "(type)");
 | |
|   EXPECT_EQ(Snippet, "${1:(type)}");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, ObjectiveCMethodTwoArgumentsFromBeginning) {
 | |
|   Builder.AddResultTypeChunk("int");
 | |
|   Builder.AddTypedTextChunk("withFoo:");
 | |
|   Builder.AddPlaceholderChunk("(type)");
 | |
|   Builder.AddChunk(CodeCompletionString::CK_HorizontalSpace);
 | |
|   Builder.AddTypedTextChunk("bar:");
 | |
|   Builder.AddPlaceholderChunk("(type2)");
 | |
| 
 | |
|   auto *CCS = Builder.TakeString();
 | |
|   computeSignature(*CCS);
 | |
|   EXPECT_EQ(Signature, "(type) bar:(type2)");
 | |
|   EXPECT_EQ(Snippet, "${1:(type)} bar:${2:(type2)}");
 | |
| }
 | |
| 
 | |
| TEST_F(CompletionStringTest, ObjectiveCMethodTwoArgumentsFromMiddle) {
 | |
|   Builder.AddResultTypeChunk("int");
 | |
|   Builder.AddInformativeChunk("withFoo:");
 | |
|   Builder.AddTypedTextChunk("bar:");
 | |
|   Builder.AddPlaceholderChunk("(type2)");
 | |
| 
 | |
|   auto *CCS = Builder.TakeString();
 | |
|   computeSignature(*CCS);
 | |
|   EXPECT_EQ(Signature, "(type2)");
 | |
|   EXPECT_EQ(Snippet, "${1:(type2)}");
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| } // namespace clangd
 | |
| } // namespace clang
 |