import os.path import pathlib from typing import List from typing import Optional from typing import Tuple import pygls.uris as Uri from lsprotocol.types import CompletionItem from lsprotocol.types import Location from lsprotocol.types import Position from lsprotocol.types import Range from pygls.workspace import Document from esbonio.lsp.directives import Directives from esbonio.lsp.rst import CompletionContext from esbonio.lsp.rst import DefinitionContext from esbonio.lsp.rst import DocumentLinkContext from esbonio.lsp.sphinx import SphinxLanguageServer from esbonio.lsp.util.filepaths import complete_sphinx_filepaths from esbonio.lsp.util.filepaths import path_to_completion_item class Includes: def __init__(self, rst: SphinxLanguageServer): self.rst = rst self.logger = rst.logger.getChild(self.__class__.__name__) def complete_arguments( self, context: CompletionContext, domain: str, name: str ) -> List[CompletionItem]: if domain or name not in {"include", "literalinclude"}: return [] if not self.rst.app: return [] srcdir = self.rst.app.srcdir partial = context.match.group("argument") base = os.path.dirname(Uri.to_fs_path(context.doc.uri)) items = complete_sphinx_filepaths(srcdir, base, partial) # type: ignore[arg-type] return [path_to_completion_item(context, p) for p in items] def find_definitions( self, context: DefinitionContext, directive: str, domain: Optional[str], argument: str, ) -> List[Location]: if domain or directive not in {"literalinclude", "include"}: return [] uri = self.resolve_path(context.doc, argument) if not uri: return [] return [ Location( uri=uri, range=Range( start=Position(line=0, character=0), end=Position(line=1, character=0), ), ) ] def resolve_link( self, context: DocumentLinkContext, directive: str, domain: Optional[str], argument: str, ) -> Tuple[Optional[str], Optional[str]]: if domain or directive not in {"literalinclude", "include"}: return None, None return self.resolve_path(context.doc, argument), None def resolve_path(self, doc: Document, argument: str) -> Optional[str]: if argument.startswith("/"): if not self.rst.app: return None basedir = pathlib.Path(self.rst.app.srcdir) # Remove the leading '/' otherwise is will wipe out the basedir when # concatenated argument = argument[1:] else: basedir = pathlib.Path(Uri.to_fs_path(doc.uri)).parent fpath = (basedir / argument).resolve() if not fpath.exists(): return None return Uri.from_fs_path(str(fpath)) def esbonio_setup(rst: SphinxLanguageServer, directives: Directives): includes = Includes(rst) directives.add_argument_definition_provider(includes) directives.add_argument_completion_provider(includes) directives.add_argument_link_provider(includes)