madduck's git repository

Every one of the projects in this repository is available at the canonical URL git://git.madduck.net/madduck/pub/<projectpath> — see each project's metadata for the exact URL.

All patches and comments are welcome. Please squash your changes to logical commits before using git-format-patch and git-send-email to patches@git.madduck.net. If you'd read over the Git project's submission guidelines and adhered to them, I'd be especially grateful.

SSH access, as well as push access can be individually arranged.

If you use my repositories frequently, consider adding the following snippet to ~/.gitconfig and using the third clone URL listed for each project:

[url "git://git.madduck.net/madduck/"]
  insteadOf = madduck:

Fix crash on assert and parenthesized % format (fixes #1597, fixes #1605) (#1681)
[etc/vim.git] / docs / conf.py
1 # -*- coding: utf-8 -*-
2 #
3 # Configuration file for the Sphinx documentation builder.
4 #
5 # This file does only contain a selection of the most common options. For a
6 # full list see the documentation:
7 # http://www.sphinx-doc.org/en/stable/config
8
9 # -- Path setup --------------------------------------------------------------
10
11 # If extensions (or modules to document with autodoc) are in another directory,
12 # add these directories to sys.path here. If the directory is relative to the
13 # documentation root, use os.path.abspath to make it absolute, like shown here.
14 #
15 from pathlib import Path
16 import re
17 import string
18 from typing import Callable, Dict, List, Optional, Pattern, Tuple, Set
19 from dataclasses import dataclass
20 import logging
21
22 from pkg_resources import get_distribution
23
24 logging.basicConfig(format="%(levelname)s: %(message)s", level=logging.INFO)
25
26 LOG = logging.getLogger(__name__)
27
28 CURRENT_DIR = Path(__file__).parent
29 README = CURRENT_DIR / ".." / "README.md"
30 REFERENCE_DIR = CURRENT_DIR / "reference"
31 STATIC_DIR = CURRENT_DIR / "_static"
32
33
34 @dataclass
35 class SrcRange:
36     """Tracks which part of a file to get a section's content.
37
38     Data:
39         start_line: The line where the section starts (i.e. its sub-header) (inclusive).
40         end_line: The line where the section ends (usually next sub-header) (exclusive).
41     """
42
43     start_line: int
44     end_line: int
45
46
47 @dataclass
48 class DocSection:
49     """Tracks information about a section of documentation.
50
51     Data:
52         name: The section's name. This will used to detect duplicate sections.
53         src: The filepath to get its contents.
54         processors: The processors to run before writing the section to CURRENT_DIR.
55         out_filename: The filename to use when writing the section to CURRENT_DIR.
56         src_range: The line range of SRC to gets its contents.
57     """
58
59     name: str
60     src: Path
61     src_range: SrcRange = SrcRange(0, 1_000_000)
62     out_filename: str = ""
63     processors: Tuple[Callable, ...] = ()
64
65     def get_out_filename(self) -> str:
66         if not self.out_filename:
67             return self.name + ".md"
68         else:
69             return self.out_filename
70
71
72 def make_pypi_svg(version: str) -> None:
73     template: Path = CURRENT_DIR / "_static" / "pypi_template.svg"
74     target: Path = CURRENT_DIR / "_static" / "pypi.svg"
75     with open(str(template), "r", encoding="utf8") as f:
76         svg: str = string.Template(f.read()).substitute(version=version)
77     with open(str(target), "w", encoding="utf8") as f:
78         f.write(svg)
79
80
81 def make_filename(line: str) -> str:
82     non_letters: Pattern = re.compile(r"[^a-z]+")
83     filename: str = line[3:].rstrip().lower()
84     filename = non_letters.sub("_", filename)
85     if filename.startswith("_"):
86         filename = filename[1:]
87     if filename.endswith("_"):
88         filename = filename[:-1]
89     return filename + ".md"
90
91
92 def get_contents(section: DocSection) -> str:
93     """Gets the contents for the DocSection."""
94     contents: List[str] = []
95     src: Path = section.src
96     start_line: int = section.src_range.start_line
97     end_line: int = section.src_range.end_line
98     with open(src, "r", encoding="utf-8") as f:
99         for lineno, line in enumerate(f, start=1):
100             if lineno >= start_line and lineno < end_line:
101                 contents.append(line)
102     result = "".join(contents)
103     # Let's make Prettier happy with the amount of trailing newlines in the sections.
104     if result.endswith("\n\n"):
105         result = result[:-1]
106     if not result.endswith("\n"):
107         result = result + "\n"
108     return result
109
110
111 def get_sections_from_readme() -> List[DocSection]:
112     """Gets the sections from README so they can be processed by process_sections.
113
114     It opens README and goes down line by line looking for sub-header lines which
115     denotes a section. Once it finds a sub-header line, it will create a DocSection
116     object with all of the information currently available. Then on every line, it will
117     track the ending line index of the section. And it repeats this for every sub-header
118     line it finds.
119     """
120     sections: List[DocSection] = []
121     section: Optional[DocSection] = None
122     with open(README, "r", encoding="utf-8") as f:
123         for lineno, line in enumerate(f, start=1):
124             if line.startswith("## "):
125                 filename = make_filename(line)
126                 section_name = filename[:-3]
127                 section = DocSection(
128                     name=str(section_name),
129                     src=README,
130                     src_range=SrcRange(lineno, lineno),
131                     out_filename=filename,
132                     processors=(fix_headers,),
133                 )
134                 sections.append(section)
135             if section is not None:
136                 section.src_range.end_line += 1
137     return sections
138
139
140 def fix_headers(contents: str) -> str:
141     """Fixes the headers of sections copied from README.
142
143     Removes one octothorpe (#) from all headers since the contents are no longer nested
144     in a root document (i.e. the README).
145     """
146     lines: List[str] = contents.splitlines()
147     fixed_contents: List[str] = []
148     for line in lines:
149         if line.startswith("##"):
150             line = line[1:]
151         fixed_contents.append(line + "\n")  # splitlines strips the leading newlines
152     return "".join(fixed_contents)
153
154
155 def process_sections(
156     custom_sections: List[DocSection], readme_sections: List[DocSection]
157 ) -> None:
158     """Reads, processes, and writes sections to CURRENT_DIR.
159
160     For each section, the contents will be fetched, processed by processors
161     required by the section, and written to CURRENT_DIR. If it encounters duplicate
162     sections (i.e. shares the same name attribute), it will skip processing the
163     duplicates.
164
165     It processes custom sections before the README generated sections so sections in the
166     README can be overwritten with custom options.
167     """
168     processed_sections: Dict[str, DocSection] = {}
169     modified_files: Set[Path] = set()
170     sections: List[DocSection] = custom_sections
171     sections.extend(readme_sections)
172     for section in sections:
173         if section.name in processed_sections:
174             LOG.warning(
175                 f"Skipping '{section.name}' from '{section.src}' as it is a duplicate"
176                 f" of a custom section from '{processed_sections[section.name].src}'"
177             )
178             continue
179
180         LOG.info(f"Processing '{section.name}' from '{section.src}'")
181         target_path: Path = CURRENT_DIR / section.get_out_filename()
182         if target_path in modified_files:
183             LOG.warning(
184                 f"{target_path} has been already written to, its contents will be"
185                 " OVERWRITTEN and notices will be duplicated"
186             )
187         contents: str = get_contents(section)
188
189         # processors goes here
190         if fix_headers in section.processors:
191             contents = fix_headers(contents)
192
193         with open(target_path, "w", encoding="utf-8") as f:
194             if section.src.suffix == ".md" and section.src != target_path:
195                 rel = section.src.resolve().relative_to(CURRENT_DIR.parent)
196                 f.write(f'[//]: # "NOTE: THIS FILE WAS AUTOGENERATED FROM {rel}"\n\n')
197             f.write(contents)
198         processed_sections[section.name] = section
199         modified_files.add(target_path)
200
201
202 # -- Project information -----------------------------------------------------
203
204 project = "Black"
205 copyright = "2020, Łukasz Langa and contributors to Black"
206 author = "Łukasz Langa and contributors to Black"
207
208 # Autopopulate version
209 # The version, including alpha/beta/rc tags, but not commit hash and datestamps
210 release = get_distribution("black").version.split("+")[0]
211 # The short X.Y version.
212 version = release
213 for sp in "abcfr":
214     version = version.split(sp)[0]
215
216 custom_sections = [
217     DocSection("the_black_code_style", CURRENT_DIR / "the_black_code_style.md"),
218     DocSection("editor_integration", CURRENT_DIR / "editor_integration.md"),
219     DocSection("blackd", CURRENT_DIR / "blackd.md"),
220     DocSection("black_primer", CURRENT_DIR / "black_primer.md"),
221     DocSection("contributing_to_black", CURRENT_DIR / ".." / "CONTRIBUTING.md"),
222     DocSection("change_log", CURRENT_DIR / ".." / "CHANGES.md"),
223 ]
224
225 # Sphinx complains when there is a source file that isn't referenced in any of the docs.
226 # Since some sections autogenerated from the README are unused warnings will appear.
227 #
228 # Sections must be listed to what their name is when passed through make_filename().
229 blocklisted_sections_from_readme = {
230     "license",
231     "pragmatism",
232     "testimonials",
233     "used_by",
234 }
235
236 make_pypi_svg(release)
237 readme_sections = get_sections_from_readme()
238 readme_sections = [
239     x for x in readme_sections if x.name not in blocklisted_sections_from_readme
240 ]
241
242 process_sections(custom_sections, readme_sections)
243
244
245 # -- General configuration ---------------------------------------------------
246
247 # If your documentation needs a minimal Sphinx version, state it here.
248 needs_sphinx = "3.0"
249
250 # Add any Sphinx extension module names here, as strings. They can be
251 # extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
252 # ones.
253 extensions = [
254     "sphinx.ext.autodoc",
255     "sphinx.ext.intersphinx",
256     "sphinx.ext.napoleon",
257     "recommonmark",
258 ]
259
260 # If you need extensions of a certain version or higher, list them here.
261 needs_extensions = {"recommonmark": "0.5"}
262
263 # Add any paths that contain templates here, relative to this directory.
264 templates_path = ["_templates"]
265
266 # The suffix(es) of source filenames.
267 # You can specify multiple suffix as a list of string:
268 source_suffix = [".rst", ".md"]
269
270 # The master toctree document.
271 master_doc = "index"
272
273 # The language for content autogenerated by Sphinx. Refer to documentation
274 # for a list of supported languages.
275 #
276 # This is also used if you do content translation via gettext catalogs.
277 # Usually you set "language" from the command line for these cases.
278 language = None
279
280 # List of patterns, relative to source directory, that match files and
281 # directories to ignore when looking for source files.
282 # This pattern also affects html_static_path and html_extra_path .
283
284 exclude_patterns = ["_build", "Thumbs.db", ".DS_Store"]
285
286 # The name of the Pygments (syntax highlighting) style to use.
287 pygments_style = "sphinx"
288
289
290 # -- Options for HTML output -------------------------------------------------
291
292 # The theme to use for HTML and HTML Help pages.  See the documentation for
293 # a list of builtin themes.
294 #
295 html_theme = "alabaster"
296
297 html_sidebars = {
298     "**": [
299         "about.html",
300         "navigation.html",
301         "relations.html",
302         "sourcelink.html",
303         "searchbox.html",
304     ]
305 }
306
307 html_theme_options = {
308     "show_related": False,
309     "description": "“Any color you like.”",
310     "github_button": True,
311     "github_user": "psf",
312     "github_repo": "black",
313     "github_type": "star",
314     "show_powered_by": True,
315     "fixed_sidebar": True,
316     "logo": "logo2.png",
317     "travis_button": True,
318 }
319
320
321 # Add any paths that contain custom static files (such as style sheets) here,
322 # relative to this directory. They are copied after the builtin static files,
323 # so a file named "default.css" will overwrite the builtin "default.css".
324 html_static_path = ["_static"]
325
326 # Custom sidebar templates, must be a dictionary that maps document names
327 # to template names.
328 #
329 # The default sidebars (for documents that don't match any pattern) are
330 # defined by theme itself.  Builtin themes are using these templates by
331 # default: ``['localtoc.html', 'relations.html', 'sourcelink.html',
332 # 'searchbox.html']``.
333 #
334 # html_sidebars = {}
335
336
337 # -- Options for HTMLHelp output ---------------------------------------------
338
339 # Output file base name for HTML help builder.
340 htmlhelp_basename = "blackdoc"
341
342
343 # -- Options for LaTeX output ------------------------------------------------
344
345 latex_elements = {
346     # The paper size ('letterpaper' or 'a4paper').
347     #
348     # 'papersize': 'letterpaper',
349     # The font size ('10pt', '11pt' or '12pt').
350     #
351     # 'pointsize': '10pt',
352     # Additional stuff for the LaTeX preamble.
353     #
354     # 'preamble': '',
355     # Latex figure (float) alignment
356     #
357     # 'figure_align': 'htbp',
358 }
359
360 # Grouping the document tree into LaTeX files. List of tuples
361 # (source start file, target name, title,
362 #  author, documentclass [howto, manual, or own class]).
363 latex_documents = [
364     (
365         master_doc,
366         "black.tex",
367         "Documentation for Black",
368         "Łukasz Langa and contributors to Black",
369         "manual",
370     )
371 ]
372
373
374 # -- Options for manual page output ------------------------------------------
375
376 # One entry per manual page. List of tuples
377 # (source start file, name, description, authors, manual section).
378 man_pages = [(master_doc, "black", "Documentation for Black", [author], 1)]
379
380
381 # -- Options for Texinfo output ----------------------------------------------
382
383 # Grouping the document tree into Texinfo files. List of tuples
384 # (source start file, target name, title, author,
385 #  dir menu entry, description, category)
386 texinfo_documents = [
387     (
388         master_doc,
389         "Black",
390         "Documentation for Black",
391         author,
392         "Black",
393         "The uncompromising Python code formatter",
394         "Miscellaneous",
395     )
396 ]
397
398
399 # -- Options for Epub output -------------------------------------------------
400
401 # Bibliographic Dublin Core info.
402 epub_title = project
403 epub_author = author
404 epub_publisher = author
405 epub_copyright = copyright
406
407 # The unique identifier of the text. This can be a ISBN number
408 # or the project homepage.
409 #
410 # epub_identifier = ''
411
412 # A unique identification for the text.
413 #
414 # epub_uid = ''
415
416 # A list of files that should not be packed into the epub file.
417 epub_exclude_files = ["search.html"]
418
419
420 # -- Extension configuration -------------------------------------------------
421
422 autodoc_member_order = "bysource"
423
424 # -- Options for intersphinx extension ---------------------------------------
425
426 # Example configuration for intersphinx: refer to the Python standard library.
427 intersphinx_mapping = {"https://docs.python.org/3/": None}