974 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			974 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			PHP
		
	
	
| <?php
 | |
| declare(strict_types = 1);
 | |
| 
 | |
| namespace LanguageServer\Tests\Server\TextDocument;
 | |
| 
 | |
| use PHPUnit\Framework\TestCase;
 | |
| use LanguageServer\Tests\MockProtocolStream;
 | |
| use LanguageServer\{
 | |
|     Server, LanguageClient, PhpDocumentLoader, DefinitionResolver
 | |
| };
 | |
| use LanguageServer\Index\{Index, ProjectIndex, DependenciesIndex};
 | |
| use LanguageServer\ContentRetriever\FileSystemContentRetriever;
 | |
| use LanguageServerProtocol\{
 | |
|     TextDocumentIdentifier,
 | |
|     TextEdit,
 | |
|     Range,
 | |
|     Position,
 | |
|     CompletionList,
 | |
|     CompletionItem,
 | |
|     CompletionItemKind,
 | |
|     CompletionContext,
 | |
|     CompletionTriggerKind
 | |
| };
 | |
| use function LanguageServer\pathToUri;
 | |
| 
 | |
| class CompletionTest extends TestCase
 | |
| {
 | |
|     /**
 | |
|      * @var Server\TextDocument
 | |
|      */
 | |
|     private $textDocument;
 | |
| 
 | |
|     /**
 | |
|      * @var PhpDocumentLoader
 | |
|      */
 | |
|     private $loader;
 | |
| 
 | |
|     public function setUp()
 | |
|     {
 | |
|         $client = new LanguageClient(new MockProtocolStream, new MockProtocolStream);
 | |
|         $projectIndex = new ProjectIndex(new Index, new DependenciesIndex);
 | |
|         $definitionResolver = new DefinitionResolver($projectIndex);
 | |
|         $contentRetriever = new FileSystemContentRetriever;
 | |
|         $this->loader = new PhpDocumentLoader($contentRetriever, $projectIndex, $definitionResolver);
 | |
|         $this->loader->load(pathToUri(__DIR__ . '/../../../fixtures/global_symbols.php'))->wait();
 | |
|         $this->loader->load(pathToUri(__DIR__ . '/../../../fixtures/symbols.php'))->wait();
 | |
|         $this->textDocument = new Server\TextDocument($this->loader, $definitionResolver, $client, $projectIndex);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `$obj->t|`
 | |
|      */
 | |
|     public function testPropertyAndMethodWithPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/property_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(3, 7)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'testProperty',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 '\TestClass', // Type of the property
 | |
|                 'Reprehenderit magna velit mollit ipsum do.'
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'testMethod',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 '\TestClass', // Return type of the method
 | |
|                 'Non culpa nostrud mollit esse sunt laboris in irure ullamco cupidatat amet.'
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `public function a() { tes| }`
 | |
|      */
 | |
|     public function testGlobalFunctionInsideNamespaceAndClass()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/inside_namespace_and_method.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(8, 11)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'test_function',
 | |
|                 CompletionItemKind::FUNCTION,
 | |
|                 'void', // Return type
 | |
|                 'Officia aliquip adipisicing et nulla et laboris dolore labore.',
 | |
|                 null,
 | |
|                 null,
 | |
|                 '\test_function'
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `$obj->|`
 | |
|      */
 | |
|     public function testPropertyAndMethodWithoutPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/property.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(3, 6)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'testProperty',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 '\TestClass', // Type of the property
 | |
|                 'Reprehenderit magna velit mollit ipsum do.'
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'testMethod',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 '\TestClass', // Return type of the method
 | |
|                 'Non culpa nostrud mollit esse sunt laboris in irure ullamco cupidatat amet.'
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `$|` when variables are defined
 | |
|      */
 | |
|     public function testVariable()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/variable.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(8, 5)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 '$var',
 | |
|                 CompletionItemKind::VARIABLE,
 | |
|                 'int',
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(8, 5), new Position(8, 5)), 'var')
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 '$param',
 | |
|                 CompletionItemKind::VARIABLE,
 | |
|                 'string|null',
 | |
|                 'A parameter',
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(8, 5), new Position(8, 5)), 'param')
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `$p|` when variables are defined
 | |
|      */
 | |
|     public function testVariableWithPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/variable_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(8, 6)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 '$param',
 | |
|                 CompletionItemKind::VARIABLE,
 | |
|                 'string|null',
 | |
|                 'A parameter',
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(8, 6), new Position(8, 6)), 'aram')
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `new|` when in a namespace and have used variables.
 | |
|      */
 | |
|     public function testNewInNamespace()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/used_new.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(6, 10)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             // Global TestClass definition (inserted as \TestClass)
 | |
|             new CompletionItem(
 | |
|                 'TestClass',
 | |
|                 CompletionItemKind::CLASS_,
 | |
|                 null,
 | |
|                 'Pariatur ut laborum tempor voluptate consequat ea deserunt.' . "\n\n" .
 | |
|                 'Deserunt enim minim sunt sint ea nisi. Deserunt excepteur tempor id nostrud' . "\n" .
 | |
|                 'laboris commodo ad commodo velit mollit qui non officia id. Nulla duis veniam' . "\n" .
 | |
|                 'veniam officia deserunt et non dolore mollit ea quis eiusmod sit non. Occaecat' . "\n" .
 | |
|                 'consequat sunt culpa exercitation pariatur id reprehenderit nisi incididunt Lorem' . "\n" .
 | |
|                 'sint. Officia culpa pariatur laborum nostrud cupidatat consequat mollit.',
 | |
|                 null,
 | |
|                 null,
 | |
|                 '\TestClass'
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'ChildClass',
 | |
|                 CompletionItemKind::CLASS_,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 '\ChildClass'
 | |
|             ),
 | |
|             // Namespaced, `use`d TestClass definition (inserted as TestClass)
 | |
|             new CompletionItem(
 | |
|                 'TestClass',
 | |
|                 CompletionItemKind::CLASS_,
 | |
|                 'TestNamespace',
 | |
|                 'Pariatur ut laborum tempor voluptate consequat ea deserunt.' . "\n\n" .
 | |
|                 'Deserunt enim minim sunt sint ea nisi. Deserunt excepteur tempor id nostrud' . "\n" .
 | |
|                 'laboris commodo ad commodo velit mollit qui non officia id. Nulla duis veniam' . "\n" .
 | |
|                 'veniam officia deserunt et non dolore mollit ea quis eiusmod sit non. Occaecat' . "\n" .
 | |
|                 'consequat sunt culpa exercitation pariatur id reprehenderit nisi incididunt Lorem' . "\n" .
 | |
|                 'sint. Officia culpa pariatur laborum nostrud cupidatat consequat mollit.',
 | |
|                 null,
 | |
|                 null,
 | |
|                 'TestClass'
 | |
|             ),
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `TestC|` with `use TestNamespace\TestClass`
 | |
|      */
 | |
|     public function testUsedClass()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/used_class.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(6, 5)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'TestClass',
 | |
|                 CompletionItemKind::CLASS_,
 | |
|                 'TestNamespace',
 | |
|                 'Pariatur ut laborum tempor voluptate consequat ea deserunt.' . "\n\n" .
 | |
|                     'Deserunt enim minim sunt sint ea nisi. Deserunt excepteur tempor id nostrud' . "\n" .
 | |
|                     'laboris commodo ad commodo velit mollit qui non officia id. Nulla duis veniam' . "\n" .
 | |
|                     'veniam officia deserunt et non dolore mollit ea quis eiusmod sit non. Occaecat' . "\n" .
 | |
|                     'consequat sunt culpa exercitation pariatur id reprehenderit nisi incididunt Lorem' . "\n" .
 | |
|                     'sint. Officia culpa pariatur laborum nostrud cupidatat consequat mollit.',
 | |
|                 null,
 | |
|                 null,
 | |
|                 'TestClass'
 | |
|             )
 | |
|         ], true), $items);
 | |
| 
 | |
|         $this->assertCompletionsListDoesNotContainLabel('OtherClass', $items);
 | |
|         $this->assertCompletionsListDoesNotContainLabel('TestInterface', $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `AliasNamespace\I|` with `use TestNamespace\InnerNamespace as AliasNamespace`
 | |
|      */
 | |
|     public function testUsedNamespaceWithPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/used_namespace.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(8, 16)
 | |
|         )->wait();
 | |
|         $this->assertEquals(
 | |
|             new CompletionList([
 | |
|                 new CompletionItem(
 | |
|                     'InnerClass',
 | |
|                     CompletionItemKind::CLASS_,
 | |
|                     'TestNamespace\\InnerNamespace',
 | |
|                     null,
 | |
|                     null,
 | |
|                     null,
 | |
|                     'AliasNamespace\\InnerClass'
 | |
|                 )
 | |
|             ], true),
 | |
|             $items
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `AliasNamespace\|` with `use TestNamespace\InnerNamespace as AliasNamespace`
 | |
|      */
 | |
|     public function testUsedNamespaceWithoutPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/used_namespace.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(9, 15)
 | |
|         )->wait();
 | |
|         $this->assertEquals(
 | |
|             new CompletionList([
 | |
|                 new CompletionItem(
 | |
|                     'InnerClass',
 | |
|                     CompletionItemKind::CLASS_,
 | |
|                     'TestNamespace\InnerNamespace',
 | |
|                     null,
 | |
|                     null,
 | |
|                     null,
 | |
|                     'AliasNamespace\InnerClass'
 | |
|                 ),
 | |
|             ], true),
 | |
|             $items
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `TestClass::$st|`
 | |
|      */
 | |
|     public function testStaticPropertyWithPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/static_property_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(2, 14)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'staticTestProperty',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 '\TestClass[]',
 | |
|                 'Lorem excepteur officia sit anim velit veniam enim.',
 | |
|                 null,
 | |
|                 null,
 | |
|                 '$staticTestProperty'
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `TestClass::|`
 | |
|      */
 | |
|     public function testStaticWithoutPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/static.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(2, 11)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'TEST_CLASS_CONST',
 | |
|                 CompletionItemKind::VARIABLE,
 | |
|                 'int',
 | |
|                 'Anim labore veniam consectetur laboris minim quis aute aute esse nulla ad.'
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'staticTestProperty',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 '\TestClass[]',
 | |
|                 'Lorem excepteur officia sit anim velit veniam enim.',
 | |
|                 null,
 | |
|                 null,
 | |
|                 '$staticTestProperty'
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'staticTestMethod',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed', // Method return type
 | |
|                 'Do magna consequat veniam minim proident eiusmod incididunt aute proident.'
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `TestClass::st|`
 | |
|      */
 | |
|     public function testStaticMethodWithPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/static_method_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(2, 13)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'staticTestMethod',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed',
 | |
|                 'Do magna consequat veniam minim proident eiusmod incididunt aute proident.'
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `TestClass::TE` at the root level.
 | |
|      */
 | |
|     public function testClassConstWithPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/class_const_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(2, 13)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'TEST_CLASS_CONST',
 | |
|                 CompletionItemKind::VARIABLE,
 | |
|                 'int',
 | |
|                 'Anim labore veniam consectetur laboris minim quis aute aute esse nulla ad.'
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Test completion at `\TestC|` in a namespace
 | |
|      */
 | |
|     public function testFullyQualifiedClass()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/fully_qualified_class.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(6, 6)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'TestClass',
 | |
|                 CompletionItemKind::CLASS_,
 | |
|                 null,
 | |
|                 'Pariatur ut laborum tempor voluptate consequat ea deserunt.' . "\n\n" .
 | |
|                 'Deserunt enim minim sunt sint ea nisi. Deserunt excepteur tempor id nostrud' . "\n" .
 | |
|                 'laboris commodo ad commodo velit mollit qui non officia id. Nulla duis veniam' . "\n" .
 | |
|                 'veniam officia deserunt et non dolore mollit ea quis eiusmod sit non. Occaecat' . "\n" .
 | |
|                 'consequat sunt culpa exercitation pariatur id reprehenderit nisi incididunt Lorem' . "\n" .
 | |
|                 'sint. Officia culpa pariatur laborum nostrud cupidatat consequat mollit.'
 | |
|             )
 | |
|         ], true), $items);
 | |
|         // Assert that all results are non-namespaced.
 | |
|         foreach ($items->items as $item) {
 | |
|             $this->assertSame($item->detail, null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `cl|` at root level
 | |
|      */
 | |
|     public function testKeywords()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/keywords.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(2, 1)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem('class', CompletionItemKind::KEYWORD, null, null, null, null, 'class'),
 | |
|             new CompletionItem('clone', CompletionItemKind::KEYWORD, null, null, null, null, 'clone')
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion in an empty file
 | |
|      */
 | |
|     public function testHtmlWithoutPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/html.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(0, 0)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 '<?php',
 | |
|                 CompletionItemKind::KEYWORD,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(0, 0), new Position(0, 0)), '<?php')
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion in `<|` when not within `<?php` tags
 | |
|      */
 | |
|     public function testHtmlWontBeProposedWithoutCompletionContext()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/html_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(0, 1)
 | |
|         )->wait();
 | |
| 
 | |
|         $this->assertEquals(new CompletionList([], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion in `<|` when not within `<?php` tags
 | |
|      */
 | |
|     public function testHtmlWontBeProposedWithPrefixWithCompletionContext()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/html_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(0, 1),
 | |
|             new CompletionContext(CompletionTriggerKind::TRIGGER_CHARACTER, '<')
 | |
|         )->wait();
 | |
| 
 | |
|         $this->assertEquals(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 '<?php',
 | |
|                 CompletionItemKind::KEYWORD,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(0, 1), new Position(0, 1)), '?php')
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `<|` when not within `<?php` tags when triggered by trigger character.
 | |
|      */
 | |
|     public function testHtmlPrefixShouldNotTriggerCompletion()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/html_no_completion.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(0, 1),
 | |
|             new CompletionContext(CompletionTriggerKind::TRIGGER_CHARACTER, '>')
 | |
|         )->wait();
 | |
|         $this->assertEquals(new CompletionList([], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `<|` when not within `<?php` tags when triggered by user input.
 | |
|      */
 | |
|     public function testHtmlPrefixShouldTriggerCompletionIfManuallyInvoked()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/html_no_completion.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(0, 1),
 | |
|             new CompletionContext(CompletionTriggerKind::INVOKED)
 | |
|         )->wait();
 | |
|         $this->assertEquals(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 '<?php',
 | |
|                 CompletionItemKind::KEYWORD,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(0, 1), new Position(0, 1)), '?php')
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `SomeNa|` when namespace `SomeNamespace` is defined
 | |
|      */
 | |
|     public function testNamespace()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/namespace.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(4, 6)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'SomeNamespace',
 | |
|                 CompletionItemKind::MODULE
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `echo $ab|` at the root level.
 | |
|      */
 | |
|     public function testBarePhpVariable()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/bare_php.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(4, 8)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 '$abc2',
 | |
|                 CompletionItemKind::VARIABLE,
 | |
|                 'int',
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(4, 8), new Position(4, 8)), 'c2')
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 '$abc',
 | |
|                 CompletionItemKind::VARIABLE,
 | |
|                 'int',
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 new TextEdit(new Range(new Position(4, 8), new Position(4, 8)), 'c')
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider foreachProvider
 | |
|      */
 | |
|     public function testForeach(Position $position, array $expectedItems)
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/foreach.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             $position
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList($expectedItems, true), $items);
 | |
|     }
 | |
| 
 | |
|     public function foreachProvider(): array
 | |
|     {
 | |
|         return [
 | |
|             'foreach value' => [
 | |
|                 new Position(18, 6),
 | |
|                 [
 | |
|                     new CompletionItem(
 | |
|                         '$value',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         '\\Foo\\Bar',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(18, 6), new Position(18, 6)), 'alue')
 | |
|                     ),
 | |
|                 ]
 | |
|             ],
 | |
|             'foreach value resolved' => [
 | |
|                 new Position(19, 12),
 | |
|                 [
 | |
|                     new CompletionItem(
 | |
|                         'foo',
 | |
|                         CompletionItemKind::PROPERTY,
 | |
|                         'mixed'
 | |
|                     ),
 | |
|                     new CompletionItem(
 | |
|                         'test',
 | |
|                         CompletionItemKind::METHOD,
 | |
|                         '\\Foo\\Bar[]'
 | |
|                     ),
 | |
|                 ]
 | |
|             ],
 | |
|             'array creation with multiple objects' => [
 | |
|                 new Position(23, 5),
 | |
|                 [
 | |
|                     new CompletionItem(
 | |
|                         '$value',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         '\\Foo\\Bar|\\stdClass',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(23, 5), new Position(23, 5)), 'value')
 | |
|                     ),
 | |
|                     new CompletionItem(
 | |
|                         '$key',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         'int',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(23, 5), new Position(23, 5)), 'key')
 | |
|                     ),
 | |
|                 ]
 | |
|             ],
 | |
|             'array creation with string/int keys and object values' => [
 | |
|                 new Position(27, 5),
 | |
|                 [
 | |
|                     new CompletionItem(
 | |
|                         '$value',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         '\\Foo\\Bar',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(27, 5), new Position(27, 5)), 'value')
 | |
|                     ),
 | |
|                     new CompletionItem(
 | |
|                         '$key',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         'string|int',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(27, 5), new Position(27, 5)), 'key')
 | |
|                     ),
 | |
|                 ]
 | |
|             ],
 | |
|             'array creation with only string keys' => [
 | |
|                 new Position(31, 5),
 | |
|                 [
 | |
|                     new CompletionItem(
 | |
|                         '$value',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         '\\Foo\\Bar',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(31, 5), new Position(31, 5)), 'value')
 | |
|                     ),
 | |
|                     new CompletionItem(
 | |
|                         '$key',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         'string',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(31, 5), new Position(31, 5)), 'key')
 | |
|                     ),
 | |
|                 ]
 | |
|             ],
 | |
|             'foreach function call' => [
 | |
|                 new Position(35, 5),
 | |
|                 [
 | |
|                     new CompletionItem(
 | |
|                         '$value',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         '\\Foo\\Bar',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(35, 5), new Position(35, 5)), 'value')
 | |
|                     ),
 | |
|                 ]
 | |
|             ],
 | |
|             'foreach unknown type' => [
 | |
|                 new Position(39, 10),
 | |
|                 [
 | |
|                     new CompletionItem(
 | |
|                         '$unknown',
 | |
|                         CompletionItemKind::VARIABLE,
 | |
|                         'mixed',
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         null,
 | |
|                         new TextEdit(new Range(new Position(39, 10), new Position(39, 10)), 'wn')
 | |
|                     ),
 | |
|                 ]
 | |
|             ],
 | |
|         ];
 | |
|     }
 | |
| 
 | |
|     public function testMethodReturnType()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/method_return_type.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(10, 6)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'foo',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 '\FooClass',
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     public function testStaticMethodReturnType()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/static_method_return_type.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(11, 6)
 | |
|         )->wait();
 | |
|         $this->assertCompletionsListSubset(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'bar',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed',
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null,
 | |
|                 null
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     private function assertCompletionsListSubset(CompletionList $subsetList, CompletionList $list)
 | |
|     {
 | |
|         foreach ($subsetList->items as $expectedItem) {
 | |
|             $this->assertContains($expectedItem, $list->items, null, null, false);
 | |
|         }
 | |
| 
 | |
|         $this->assertEquals($subsetList->isIncomplete, $list->isIncomplete);
 | |
|     }
 | |
| 
 | |
|     private function assertCompletionsListDoesNotContainLabel(string $label, CompletionList $list)
 | |
|     {
 | |
|         foreach ($list->items as $item) {
 | |
|             $this->assertNotSame($label, $item->label, "Completion list should not contain $label.");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion for `$this->|`
 | |
|      */
 | |
|     public function testThisWithoutPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/this.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(12, 15)
 | |
|         )->wait();
 | |
|         $this->assertEquals(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'foo',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 'mixed', // Type of the property
 | |
|                 null
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'bar',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 'mixed', // Type of the property
 | |
|                 null
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'method',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed', // Return type of the method
 | |
|                 null
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'test',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed', // Return type of the method
 | |
|                 null
 | |
|             )
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `$this->m|`
 | |
|      */
 | |
|     public function testThisWithPrefix()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/this_with_prefix.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(12, 16)
 | |
|         )->wait();
 | |
|         $this->assertEquals(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'foo',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 'mixed', // Type of the property
 | |
|                 null
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'bar',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 'mixed', // Type of the property
 | |
|                 null
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'method',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed', // Return type of the method
 | |
|                 null
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'test',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed', // Return type of the method
 | |
|                 null
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'testProperty',
 | |
|                 CompletionItemKind::PROPERTY,
 | |
|                 '\TestClass', // Type of the property
 | |
|                 'Reprehenderit magna velit mollit ipsum do.'
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'testMethod',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 '\TestClass', // Return type of the method
 | |
|                 'Non culpa nostrud mollit esse sunt laboris in irure ullamco cupidatat amet.'
 | |
|             ),
 | |
|         ], true), $items);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Tests completion at `$this->foo()->q|`
 | |
|      */
 | |
|     public function testThisReturnValue()
 | |
|     {
 | |
|         $completionUri = pathToUri(__DIR__ . '/../../../fixtures/completion/this_return_value.php');
 | |
|         $this->loader->open($completionUri, file_get_contents($completionUri));
 | |
|         $items = $this->textDocument->completion(
 | |
|             new TextDocumentIdentifier($completionUri),
 | |
|             new Position(17, 23)
 | |
|         )->wait();
 | |
|         $this->assertEquals(new CompletionList([
 | |
|             new CompletionItem(
 | |
|                 'bar',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed' // Return type of the method
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'qux',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 'mixed' // Return type of the method
 | |
|             ),
 | |
|             new CompletionItem(
 | |
|                 'foo',
 | |
|                 CompletionItemKind::METHOD,
 | |
|                 '$this' // Return type of the method
 | |
|             ),
 | |
|         ], true), $items);
 | |
|     }
 | |
| }
 |