usse/scrape/venv/lib/python3.10/site-packages/esbonio/lsp/sphinx/includes.py

107 lines
3.2 KiB
Python
Raw Normal View History

2023-12-22 14:26:01 +00:00
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)