A reStructuredText formatter
Project description
🧾 Restitutor
An opinionated formatter for reStructuredText (reST) files.
Restitutor parses .rst files with docutils, reconstructs them from the AST, and writes them back in a normalized, consistent style. It aims to be safe, predictable, and friendly to tooling and code review.
✨ Features
- Normalizes general reStructuredText structure:
- Headings with consistent adornments
- Paragraph spacing
- Bullet and enumerated lists
- Preserves and re-emits several Sphinx-style constructs:
.. toctree::.. currentmodule::- Cross-reference roles like
:func:,:class:,:mod:,:ref:,:term:, etc. - Breathe directives such as
.. doxygenclass::,.. doxygenfunction::, etc.
- Enhanced table handling:
- Grid tables are reflowed with consistent borders and spacing
.. list-table::tables are preserved as list-table directives- Optional preservation of blank-line spacing between list-table rows
- Handles inline markup:
*emphasis*,**strong**,inline literals- Hyperlinks and cross-references
- Renders:
- Literal/code blocks (
::and.. code:: lang) - Images (
.. image::) - Generic admonitions (
.. admonition:: Title) - Line blocks (
| line)
- Literal/code blocks (
The formatter currently targets a subset of docutils/Sphinx nodes. If it encounters unknown or unsupported nodes, it fails loudly rather than silently corrupting the output.
📦 Installation
Restitutor requires Python 3.12+ and is available on PyPI:
pip install restitutor
🧑💻 Usage
The package installs a console script restitute with the following interface:
restitute [-i|--in-place] [-c|--clean] RST [RST ...]
-
RST
One or more paths to.rstfiles to process. -
-i, --in-place
Rewrite the given.rstfiles in-place instead of printing tostdout. -
-c, --clean
Normalize newlines.By default, Restitutor preserves blank-line counts between rows in
.. list-table::tables. With--clean, extra blank lines are removed for a cleaner, but less faithful, representation.
📚 Examples
# Format one or more `.rst` files and print to stdout
restitute docs/index.rst
restitute docs/index.rst docs/usage.rst
# Rewrite files in place
restitute -i docs/index.rst
restitute --in-place docs/*.rst
# Normalize newlines between `list-table` rows
restitute -i --clean docs/*.rst
🧠 How it works
At a high level:
docutils.core.publish_doctree()parses the input.rstinto anodes.documentAST.- Restitutor registers a set of custom directives and roles so that Sphinx-like constructs are captured as specific node types:
TocTreeNodefor.. toctree::CurrentModuleNodefor.. currentmodule::XRefNodefor cross-reference rolesMarkingListTablewraps thelist-tabledirective and annotates the resulting table with metadata to reconstruct it.- Several
Doxy*Nodetypes for Breathe directives
ast_to_rst()recursively walks the doctree and emits reStructuredText, guided by a formatting context (FmtCtx) that tracks indentation and list prefixes.- The resulting text is either printed or written back to the original file.
Headings
Headings use a fixed adornment sequence:
ADORNMENTS = ["#", "*", "=", "-", "^"]
Level 0 and 1 use overline/underline; deeper levels use underline only:
#############
Top Heading
#############
************
Subheading
************
Title
=====
Subtitle
--------
Minor
^^^^^
Lists
Bullet and enumerated lists are normalized to a consistent style:
- Bullet lists use
- - Enumerations preserve style where possible (
1.,a.,A., Roman numerals, etc.)
Tables
- Grid tables are rendered with full borders and aligned columns.
.. list-table::tables are preserved as.. list-table::; options like:header-rows:and:widths:are preserved.- Without
--clean, Restitutor preserves the number of blank lines between top-level list-table rows.
⚠️ Caveats and Limitations
- Only a subset of docutils/Sphinx nodes is supported. Unsupported nodes raise
RuntimeErrorduring formatting. - There is no test suite yet.
🧩 API (Internal)
The project is currently focused on the CLI. The main internal entry points are:
-
ast_to_rst(node: nodes.Node, ctx: FmtCtx) -> str
Convert a docutils node (usually a wholedocument) back into reStructuredText. -
FmtCtx
A dataclass controlling indentation and list-formatting context.
These APIs are internal and may change without notice.
🤝 Contributing
Issues and pull requests are welcome at:
- Issues: https://github.com/KurtBoehm/restitutor/issues
- Repository: https://github.com/KurtBoehm/restitutor
If you encounter
- crashes on particular
.rstinput, - incorrect rendering of supported constructs,
- or want support for additional Sphinx/directive features,
please open an issue with a minimal reproducer (input .rst and observed vs expected output).
📜 Licence
Restitutor is licensed under the Mozilla Public Licence 2.0, provided in License.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file restitutor-0.1.0.tar.gz.
File metadata
- Download URL: restitutor-0.1.0.tar.gz
- Upload date:
- Size: 19.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b477966a32c8ecd7412a86ffbdaff563ad5bf64371c74fc4410fd623b2727665
|
|
| MD5 |
129d3b556c386ca87fead6d26f1f8f18
|
|
| BLAKE2b-256 |
398579257b6372af3a0e4ef7dc72adae746bcba6e396ad6b3aa364dee091bb98
|
File details
Details for the file restitutor-0.1.0-py3-none-any.whl.
File metadata
- Download URL: restitutor-0.1.0-py3-none-any.whl
- Upload date:
- Size: 18.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6807fca3b5bb2aa51329c1a5cd6d463cd581457f24bfa1d6483ac0d9c59eb766
|
|
| MD5 |
71a10a2be9de91f27a81f4ce8a3420ab
|
|
| BLAKE2b-256 |
dd7dc172550c4a0529b10aa8cfca275715c237b483b5d1d538b86ff9ee0cd930
|