usse/scrape/venv/lib/python3.10/site-packages/esbonio/lsp/rst/io.py

223 lines
6.7 KiB
Python
Raw Normal View History

2023-12-22 14:26:01 +00:00
import logging
import typing
from typing import IO
from typing import Any
from typing import Callable
from typing import Optional
from typing import Type
import pygls.uris as uri
from docutils import nodes
from docutils.core import Publisher
from docutils.io import NullOutput
from docutils.io import StringInput
from docutils.parsers.rst import Directive
from docutils.parsers.rst import Parser
from docutils.parsers.rst import directives
from docutils.parsers.rst import roles
from docutils.readers.standalone import Reader
from docutils.utils import Reporter
from docutils.writers import Writer
from pygls.workspace import Document
from sphinx.environment import default_settings
from esbonio.lsp.util.patterns import DIRECTIVE
from esbonio.lsp.util.patterns import ROLE
class a_directive(nodes.Element, nodes.Inline):
"""Represents a directive."""
class a_role(nodes.Element):
"""Represents a role."""
def dummy_role(name, rawtext, text, lineno, inliner, options={}, content=[]):
node = a_role()
node.line = lineno
match = ROLE.match(rawtext)
if match is None:
node.attributes["text"] = rawtext
else:
node.attributes.update(match.groupdict())
node.attributes["text"] = match.group(0)
return [node], []
class DummyDirective(Directive):
has_content = True
def run(self):
node = a_directive()
node.line = self.lineno
parent = self.state.parent
lines = self.block_text
# substitution definitions require special handling
if isinstance(parent, nodes.substitution_definition):
lines = parent.rawsource
text = lines.split("\n")[0]
match = DIRECTIVE.match(text)
if match:
node.attributes.update(match.groupdict())
node.attributes["text"] = match.group(0)
else:
self.state.reporter.warning(f"Unable to parse directive: '{text}'")
node.attributes["text"] = text
if self.content:
# This is essentially what `nested_parse_with_titles` does in Sphinx.
# But by passing the content_offset to state.nested_parse we ensure any line
# numbers remain relative to the start of the current file.
current_titles = self.state.memo.title_styles
current_sections = self.state.memo.section_level
self.state.memo.title_styles = []
self.state.memo.section_level = 0
try:
self.state.nested_parse(
self.content, self.content_offset, node, match_titles=1
)
finally:
self.state.memo.title_styles = current_titles
self.state.memo.section_level = current_sections
return [node]
class disable_roles_and_directives:
"""Disables all roles and directives from being expanded.
The ``CustomReSTDispactcher`` from Sphinx is *very* cool.
It provides a way to override the mechanism used to lookup roles and directives
during parsing!
It's perfect for temporarily replacing all role and directive implementations with
dummy ones. Parsing an rst document with these dummy implementations effectively
gives us something that could be called an abstract syntax tree.
Unfortunately, it's only available in a relatively recent version of Sphinx (4.4)
so we have to implement the same idea ourselves for now.
"""
def __init__(self) -> None:
self.directive_backup: Optional[Callable] = None
self.role_backup: Optional[Callable] = None
def __enter__(self) -> None:
self.directive_backup = directives.directive
self.role_backup = roles.role
directives.directive = self.directive
roles.role = self.role
def __exit__(self, exc_type: Type[Exception], exc_value: Exception, traceback: Any):
directives.directive = self.directive_backup # type: ignore
roles.role = self.role_backup # type: ignore
self.directive_backup = None
self.role_backup = None
def directive(self, directive_name, language_module, document):
return DummyDirective, []
def role(self, role_name, language_module, lineno, reporter):
return dummy_role, []
class DummyWriter(Writer):
"""A writer that doesn't do anything."""
def translate(self):
pass
class LogStream:
def __init__(self, logger: logging.Logger):
self.logger = logger
def write(self, text: str):
self.logger.debug(text)
class LogReporter(Reporter):
"""A docutils reporter that writes to the given logger."""
def __init__(
self,
logger: logging.Logger,
source: str,
report_level: int,
halt_level: int,
debug: bool,
error_handler: str,
) -> None:
stream = typing.cast(IO, LogStream(logger))
super().__init__(
source, report_level, halt_level, stream, debug, error_handler=error_handler
) # type: ignore
class InitialDoctreeReader(Reader):
"""A reader that replaces the default reporter with one compatible with esbonio's
logging setup."""
def __init__(self, logger: logging.Logger, *args, **kwargs):
self.logger = logger
super().__init__(*args, **kwargs)
def new_document(self) -> nodes.document:
document = super().new_document()
reporter = document.reporter
document.reporter = LogReporter(
self.logger,
reporter.source,
reporter.report_level,
reporter.halt_level,
reporter.debug_flag,
reporter.error_handler,
)
return document
def read_initial_doctree(
document: Document, logger: logging.Logger
) -> Optional[nodes.document]:
"""Parse the given reStructuredText document into its "initial" doctree.
An "initial" doctree can be thought of as the abstract syntax tree of a
reStructuredText document. This method disables all role and directives
from being executed, instead they are replaced with nodes that simply
represent that they exist.
Parameters
----------
document
The document containing the reStructuredText source.
logger
Logger to log debug info to.
"""
parser = Parser()
with disable_roles_and_directives():
publisher = Publisher(
reader=InitialDoctreeReader(logger),
parser=parser,
writer=DummyWriter(),
source_class=StringInput,
destination=NullOutput(),
)
publisher.process_programmatic_settings(None, default_settings, None)
publisher.set_source(
source=document.source, source_path=uri.to_fs_path(document.uri)
)
publisher.publish()
return publisher.document