from concurrent import futures from pathlib import Path from typing import Optional, Tuple from cmake_language_server.server import CMakeLanguageServer from pygls.features import ( COMPLETION, FORMATTING, HOVER, INITIALIZE, TEXT_DOCUMENT_DID_OPEN, ) from pygls.server import LanguageServer from pygls.types import ( CompletionContext, CompletionList, CompletionParams, CompletionTriggerKind, DidOpenTextDocumentParams, DocumentFormattingParams, FormattingOptions, InitializeParams, Position, TextDocumentIdentifier, TextDocumentItem, TextDocumentPositionParams, ) CALL_TIMEOUT = 2 def _init(client: LanguageServer, root: Path) -> None: retry = 3 while retry > 0: try: client.lsp.send_request( INITIALIZE, InitializeParams( process_id=1234, root_uri=root.as_uri(), capabilities=None ), ).result(timeout=CALL_TIMEOUT) except futures.TimeoutError: retry -= 1 else: break def _open(client: LanguageServer, path: Path, text: Optional[str] = None) -> None: if text is None: with open(path) as fp: text = fp.read() client.lsp.notify( TEXT_DOCUMENT_DID_OPEN, DidOpenTextDocumentParams(TextDocumentItem(path.as_uri(), "cmake", 1, text)), ) def _test_completion( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path, content: str, context: Optional[CompletionContext], ) -> CompletionList: client, server = client_server _init(client, datadir) path = datadir / "CMakeLists.txt" _open(client, path, content) params = CompletionParams( TextDocumentIdentifier(path.as_uri()), Position(0, len(content)), context ) if context is None: # some clients do not send context del params.context return client.lsp.send_request(COMPLETION, params).result(timeout=CALL_TIMEOUT) def test_initialize( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: client, server = client_server assert server._api is None _init(client, datadir) assert server._api is not None def test_completions_invoked( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: response = _test_completion( client_server, datadir, "projec", CompletionContext(CompletionTriggerKind.Invoked), ) item = next(filter(lambda x: x.label == "project", response.items), None) assert item is not None assert "" in item.documentation def test_completions_nocontext( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: response = _test_completion(client_server, datadir, "projec", None) item = next(filter(lambda x: x.label == "project", response.items), None) assert item is not None assert "" in item.documentation def test_completions_triggercharacter_variable( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: response = _test_completion( client_server, datadir, "${", CompletionContext(CompletionTriggerKind.TriggerCharacter, "{"), ) assert "PROJECT_VERSION" in [x.label for x in response.items] response_nocontext = _test_completion(client_server, datadir, "${", None) assert response == response_nocontext def test_completions_triggercharacter_module( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: response = _test_completion( client_server, datadir, "include(", CompletionContext(CompletionTriggerKind.TriggerCharacter, "("), ) assert "GoogleTest" in [x.label for x in response.items] response_nocontext = _test_completion(client_server, datadir, "include(", None) assert response == response_nocontext def test_completions_triggercharacter_package( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: response = _test_completion( client_server, datadir, "find_package(", CompletionContext(CompletionTriggerKind.TriggerCharacter, "("), ) assert "Boost" in [x.label for x in response.items] response_nocontext = _test_completion(client_server, datadir, "find_package(", None) assert response == response_nocontext def test_formatting( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: client, server = client_server _init(client, datadir) path = datadir / "CMakeLists.txt" _open(client, path, "a ( b c ) ") response = client.lsp.send_request( FORMATTING, DocumentFormattingParams( TextDocumentIdentifier(path.as_uri()), FormattingOptions(2, True) ), ).result(timeout=CALL_TIMEOUT) assert response[0].newText == "a(b c)\n" def test_hover( client_server: Tuple[LanguageServer, CMakeLanguageServer], datadir: Path ) -> None: client, server = client_server _init(client, datadir) path = datadir / "CMakeLists.txt" _open(client, path, "project()") response = client.lsp.send_request( HOVER, TextDocumentPositionParams(TextDocumentIdentifier(path.as_uri()), Position()), ).result(timeout=CALL_TIMEOUT) assert "" in response.contents.value