Source code for abjad.ext.sphinx

import copy
import enum
import hashlib
import os
import pathlib
import shutil
import subprocess

import sphinx
from docutils.nodes import (
    Element,
    FixedTextElement,
    General,
    SkipNode,
    image,
    literal_block,
)
from docutils.parsers.rst import Directive, directives
from sphinx.util import logging
from sphinx.util.console import brown  # type: ignore
from sphinx.util.nodes import set_source_info
from sphinx.util.osutil import copyfile, ensuredir
from uqbar.book.extensions import Extension
from uqbar.strings import normalize

from .. import format as _format
from .. import lilypondfile as _lilypondfile
from .. import tag as _tag
from ..configuration import Configuration
from ..contextmanagers import TemporaryDirectoryChange
from ..illustrators import illustrate
from ..io import Illustrator, LilyPondIO, Player

configuration = Configuration()
logger = logging.getLogger(__name__)


[docs] class HiddenDoctestDirective(Directive): """ A hidden doctest directive. Contributes no formatting to documents built by Sphinx. """ ### CLASS VARIABLES ### __documentation_ignore_inherited__ = True has_content = True required_arguments = 0 optional_arguments = 0 final_argument_whitespace = True option_spec: dict[str, str] = {} ### PUBLIC METHODS ###
[docs] def run(self): """Executes the directive.""" self.assert_has_content() return []
[docs] class ShellDirective(Directive): """ A shell directive. Represents a shell session. Generates a docutils ``literal_block`` node. """ ### CLASS VARIABLES ### __documentation_ignore_inherited__ = True has_content = True required_arguments = 0 optional_arguments = 0 final_argument_whitespace = False option_spec: dict[str, str] = {} ### PUBLIC METHODS ###
[docs] def run(self): self.assert_has_content() result = [] with TemporaryDirectoryChange(configuration.abjad_directory): cwd = pathlib.Path.cwd() for line in self.content: result.append(f"{cwd.name}$ {line}") completed_process = subprocess.run( line, shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, text=True, ) result.append(completed_process.stdout) code = "\n".join(result) literal = literal_block(code, code) literal["language"] = "console" set_source_info(self, literal) return [literal]
[docs] class ThumbnailDirective(Directive): """ A thumbnail directive. """ ### CLASS VARIABLES ### __documentation_ignore_inherited__ = True final_argument_whitespace = True has_content = False option_spec = { "class": directives.class_option, "group": directives.unchanged, "title": directives.unchanged, } optional_arguments = 0 required_arguments = 1 ### PUBLIC METHODS ###
[docs] def run(self): """Executes the directive.""" node = thumbnail_block() node["classes"] += self.options.get("class", "") node["group"] = self.options.get("group", "") node["title"] = self.options.get("title", "") node["uri"] = self.arguments[0] environment = self.state.document.settings.env name, suffix = os.path.splitext(node["uri"]) environment.images.add_file(environment.docname, node["uri"]) thumbnail_uri = name + "-thumbnail" + suffix environment.thumbnails[thumbnail_uri] = environment.docname # this may also work: # environment.thumbnails[environment.docname] = thumbnail_uri return [node]
[docs] class thumbnail_block(image, General, Element): __documentation_ignore_inherited__ = True
[docs] def visit_thumbnail_block_html(self, node): template = normalize( """ <a data-lightbox="{group}" href="{target_path}" title="{title}" data-title="{title}" class="{cls}"> <img src="{thumbnail_path}" alt="{alt}"/> </a> """ ) title = node["title"] classes = " ".join(node["classes"]) group = "group-{}".format(node["group"] if node["group"] else node["uri"]) if node["uri"] in self.builder.images: node["uri"] = os.path.join( self.builder.imgpath, self.builder.images[node["uri"]] ) target_path = node["uri"] prefix, suffix = os.path.splitext(target_path) if suffix == ".svg": thumbnail_path = target_path else: thumbnail_path = "{}-thumbnail{}".format(prefix, suffix) output = template.format( alt=title, group=group, target_path=target_path, cls=classes, thumbnail_path=thumbnail_path, title=title, ) self.body.append(output) raise SkipNode
[docs] def visit_thumbnail_block_latex(self, node): raise SkipNode
[docs] def on_builder_inited(app): app.env.thumbnails = {} # separate so Sphinx doesn't purge it install_lightbox_static_files(app) (pathlib.Path(app.builder.outdir) / "_images").mkdir(exist_ok=True)
[docs] class LilyPondExtension(Extension): class Kind(enum.Enum): IMAGE = 1 AUDIO = 2 class lilypond_block(General, FixedTextElement): pass
[docs] @classmethod def setup_console(cls, console, monkeypatch): monkeypatch.setattr( Illustrator, "__call__", lambda self: console.push_proxy( cls( self.illustrable, cls.Kind.IMAGE, **{ key.replace("lilypond/", "").replace("-", "_"): value for key, value in console.proxy_options.items() if key.startswith("lilypond/") }, ), ), ) monkeypatch.setattr( Player, "__call__", lambda self: console.push_proxy( cls( self.illustrable, cls.Kind.AUDIO, **{ key.replace("lilypond/", "").replace("-", "_"): value for key, value in console.proxy_options.items() if key.startswith("lilypond/") }, ), ), )
[docs] @classmethod def setup_sphinx(cls, app): app.add_node( cls.lilypond_block, html=[cls.visit_block_html, None], latex=[cls.visit_block_latex, None], text=[cls.visit_block_text, cls.depart_block_text], ) cls.add_option("lilypond/no-trim", directives.flag) cls.add_option("lilypond/pages", directives.unchanged) cls.add_option("lilypond/with-columns", int)
def __init__( self, illustrable, kind, no_trim=None, pages=None, with_columns=None, **keywords, ): self.illustrable = copy.deepcopy(illustrable) self.keywords = keywords self.kind = kind self.no_trim = no_trim self.pages = pages self.with_columns = with_columns
[docs] def to_docutils(self): if isinstance(self.illustrable, _lilypondfile.LilyPondFile): illustration = self.illustrable else: illustration = illustrate(self.illustrable, **self.keywords) if self.kind == self.Kind.AUDIO: block = _lilypondfile.Block("midi") illustration["score"].items.append(block) illustration.lilypond_version_token = r'\version "2.19.83"' code = illustration._get_lilypond_format() code = _format.remove_site_comments(code) code = _tag.remove_tags(code) node = self.lilypond_block(code, code) node["kind"] = self.kind.name.lower() node["no-trim"] = self.no_trim node["pages"] = self.pages node["with-columns"] = self.with_columns return [node]
[docs] @staticmethod def visit_block_html(self, node): output_directory = pathlib.Path(self.builder.outdir) / "_images" render_prefix = "lilypond-{}".format( hashlib.sha256(node[0].encode()).hexdigest() ) if node["kind"] == "audio": flags = [] glob = f"{render_prefix}.mid*" else: flags = ["-dcrop", "-dbackend=svg"] glob = f"{render_prefix}*.svg" lilypond_io = LilyPondIO( None, flags=flags, output_directory=output_directory, render_prefix=render_prefix, should_copy_stylesheets=True, should_open=False, should_persist_log=False, string=node[0], ) if not list(output_directory.glob(glob)): _, _, _, success, log = lilypond_io() if not success: logger.warning(f"LilyPond render failed\n{log}", location=node) source_path = (pathlib.Path(self.builder.imgpath) / render_prefix).with_suffix( ".ly" ) if node["kind"] == "audio": pass else: embed_images(self, node, output_directory, render_prefix, source_path) raise SkipNode
table_row_open_template = '<div class="table-row">' table_row_close_template = "</div>" basic_image_template = normalize( """ <div class="uqbar-book"> <a href="{source_path}"><img src="{relative_path}"/></a> </div> """ ) thumbnail_template = normalize( """ <a data-lightbox="{group}" href="{fullsize_path}" title="{title}" data-title="{title}" class="{cls}"> <img src="{thumbnail_path}" alt="{alt}"/> </a> """ )
[docs] def embed_images(self, node, output_directory, render_prefix, source_path): paths_to_embed = [] if node.get("pages"): for page_spec in node["pages"].split(","): page_spec = page_spec.strip() if "-" in page_spec: start_spec, _, stop_spec = page_spec.partition("-") start_page, stop_page = int(start_spec), int(stop_spec) + 1 else: start_page, stop_page = int(page_spec), int(page_spec) + 1 for page_number in range(start_page, stop_page): for path in output_directory.glob(f"{render_prefix}-{page_number}.svg"): paths_to_embed.append(path) elif node.get("no-trim"): for path in output_directory.glob(f"{render_prefix}.svg"): paths_to_embed.append(path) if not paths_to_embed: page_count = len(list(output_directory.glob(f"{render_prefix}-*.svg"))) for page_number in range(1, page_count + 1): for path in output_directory.glob(f"{render_prefix}-{page_number}.svg"): paths_to_embed.append(path) else: for path in output_directory.glob(f"{render_prefix}.cropped.svg"): paths_to_embed.append(path) with_columns = node.get("with-columns") if with_columns: for i in range(0, len(paths_to_embed), with_columns): self.body.append(table_row_open_template) for path in paths_to_embed[i : i + with_columns]: relative_path = pathlib.Path(self.builder.imgpath) / path.name self.body.append( thumbnail_template.format( alt="", cls="table-cell thumbnail", group=f"group-{render_prefix}", fullsize_path=relative_path, thumbnail_path=relative_path, title="", ) ) self.body.append(table_row_close_template) else: for path in paths_to_embed: relative_path = pathlib.Path(self.builder.imgpath) / path.name self.body.append( basic_image_template.format( relative_path=relative_path, source_path=source_path ) )
[docs] def install_lightbox_static_files(app): source_static_path = os.path.join(app.builder.srcdir, "_static") target_static_path = os.path.join(app.builder.outdir, "_static") source_lightbox_path = os.path.join(source_static_path, "lightbox2") target_lightbox_path = os.path.join(target_static_path, "lightbox2") relative_file_paths = [] for root, _, file_names in os.walk(source_lightbox_path): for file_name in file_names: absolute_file_path = os.path.join(root, file_name) relative_file_path = os.path.relpath(absolute_file_path, source_static_path) relative_file_paths.append(relative_file_path) if os.path.exists(target_lightbox_path): shutil.rmtree(target_lightbox_path) for relative_file_path in sphinx.util.display.status_iterator( relative_file_paths, "installing lightbox files... ", brown, len(relative_file_paths), ): source_path = os.path.join(source_static_path, relative_file_path) target_path = os.path.join(target_static_path, relative_file_path) target_directory = os.path.dirname(target_path) if not os.path.exists(target_directory): ensuredir(target_directory) copyfile(source_path, target_path) if relative_file_path.endswith(".js"): app.add_js_file(relative_file_path, defer="defer") elif relative_file_path.endswith(".css"): app.add_css_file(relative_file_path)
[docs] def on_html_collect_pages(app): for path in sphinx.util.display.status_iterator( app.env.thumbnails, "copying gallery thumbnails...", "brown", len(app.env.thumbnails), ): source_path = pathlib.Path(app.srcdir) / path target_path = pathlib.Path(app.outdir) / "_images" / source_path.name try: shutil.copy(source_path, target_path) except Exception: logger.warning(f"Could not copy {source_path}") return []
[docs] def setup(app): app.connect("builder-inited", on_builder_inited) app.connect("html-collect-pages", on_html_collect_pages) app.add_css_file("abjad.css") app.add_directive("docs", HiddenDoctestDirective) app.add_directive("shell", ShellDirective) app.add_directive("thumbnail", ThumbnailDirective) app.add_js_file("copybutton.js", defer="defer") app.add_js_file("ga.js") app.add_node( thumbnail_block, html=[visit_thumbnail_block_html, None], latex=[visit_thumbnail_block_latex, None], )