Format only the literal string arguments of textwrap.dedent() calls
Project description
format-dedent
Format multiline strings with proper indentation — A Python code formatter that formats only the literal string arguments of textwrap.dedent() calls.
✨ What does it do?
format-dedent automatically formats multiline strings inside textwrap.dedent() calls to make them visually match their runtime output. This makes your code more readable while preserving the exact behavior.
Key features:
- 🎯 Surgical precision — Only formats strings inside
dedent()calls, leaves everything else untouched - 🔄 Two modes — Format existing dedent strings OR automatically add
dedent()to strings that need it - 👀 Safe — Validates that formatting doesn't change runtime behavior
- 🎨 Smart indentation — Aligns content with the visual structure of your code
- 🧹 Clean — Removes trailing whitespace and normalizes spacing
📦 Installation
uv tool install format-dedent
🚀 Quick Start
Format strings (default mode)
Preview formatted output without modifying files:
uvx format-dedent yourfile.py
Write changes to files:
uvx format-dedent yourfile.py --write
Format multiple files or directories:
uvx format-dedent src/ tests/ --write
Add dedent() calls (--add-dedent mode)
Automatically wrap multiline strings with dedent() calls:
uvx format-dedent yourfile.py --add-dedent --write
This will:
- Find multiline strings where
dedent(str) == str(no leading indentation to remove) - Wrap them with
dedent()calls - Add
from textwrap import dedentimport if needed
📖 Usage Options
python -m format_dedent [OPTIONS] [FILES/DIRECTORIES]
Options:
-w, --write Write changes to files (default: output to stdout)
--add-dedent Add dedent() calls to multiline strings
-h, --help Show help message
Behavior:
- Default → Output formatted code to stdout (no file modification)
--write→ Modify files directly and print confirmation
🔧 Pre-commit Hook
Use format-dedent as a pre-commit hook to automatically format dedent strings before each commit.
Add this to your .pre-commit-config.yaml:
repos:
- repo: https://github.com/15r10nk/format-dedent
rev: v0.1.0 # Use the latest version
hooks:
- id: format-dedent
💡 Examples
Example 1: Formatting SQL queries
Before formatting:
import textwrap
def get_sql_query():
return textwrap.dedent("""
SELECT users.name, orders.total
FROM users
JOIN orders ON users.id = orders.user_id
WHERE orders.status = 'complete'
""")
Inconsistent indentation inside the string makes it hard to read and understand the actual SQL query structure.
After formatting:
import textwrap
def get_sql_query():
return textwrap.dedent("""
SELECT users.name, orders.total
FROM users
JOIN orders ON users.id = orders.user_id
WHERE orders.status = 'complete'
""")
The key insight: The indentation you see in the source code now matches what dedent() returns. When this code runs, dedent() strips the common leading whitespace, and you get properly formatted SQL.
Example 2: Using --add-dedent mode
Before:
def get_message():
message = """
Hello World!
This is a message.
"""
return message
After running with --add-dedent:
from textwrap import dedent
def get_message():
message = dedent("""
Hello World!
This is a message.
""")
return message
✅ What changed:
- Detected that the string has no leading whitespace (left-aligned)
- Wrapped it with
dedent()for consistency - Added the import statement automatically
- Reformatted with proper indentation matching the code structure
Why use dedent here? Even though this string doesn't need dedenting now, using dedent() consistently makes it easier to modify the string later. You can add indentation for readability without affecting the runtime output.
Example 3: HTML template formatting
Before:
from textwrap import dedent
def render_html():
return dedent("""
<div class="container">
<h1>Welcome</h1>
<p>This is a paragraph.</p>
</div>
""")
After:
from textwrap import dedent
def render_html():
return dedent("""
<div class="container">
<h1>Welcome</h1>
<p>This is a paragraph.</p>
</div>
""")
Sponsors
I would like to thank my sponsors. Without them, I would not be able to invest so much time in my projects.
Silver sponsor 🥈
🧠 How It Works
- Parse — Uses Python's AST module to analyze source code
- Find — Locates all
dedent()ortextwrap.dedent()calls with string arguments - Analyze — Determines the appropriate indentation level based on context
- Validate — Ensures
dedent(original) == dedent(formatted)(behavior unchanged) - Replace — Updates the source file with formatted strings
The key insight: Strings are formatted in the source to visually match their runtime output after dedent() processes them. This makes the code more readable without changing behavior.
🛡️ Safety & Compatibility
- Non-destructive — Always validates that
dedent(original) == dedent(formatted) - Preserves behavior — Formatted strings have identical runtime output
- Quote style aware — Maintains your choice of
"""vs''' - Escape handling — Correctly handles backslashes and escape sequences
- Python 3.8+ — Works with modern Python versions
What gets formatted:
- ✅ Literal strings inside
textwrap.dedent()calls - ✅ Literal strings inside
dedent()calls (when imported)
What doesn't get formatted:
- ❌ Regular strings (not in dedent calls)
- ❌ F-strings (can't be wrapped with dedent)
- ❌ String concatenations
- ❌ Docstrings at module level
- ❌ All other code (completely untouched)
🧪 Development
Setup
# Clone the repository
git clone https://github.com/15r10nk/format-dedent.git
cd format-dedent
# Install with development dependencies
pip install -e ".[dev]"
# Install pre-commit hooks
pre-commit install
Running Tests
# Run all tests
pytest
# Run with verbose output
pytest -v
# Run specific test file
pytest tests/test_formatter.py
Tests use inline-snapshot for snapshot testing.
📝 License
MIT License - See LICENSE file for details
🤝 Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
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 format_dedent-0.1.1.tar.gz.
File metadata
- Download URL: format_dedent-0.1.1.tar.gz
- Upload date:
- Size: 20.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7083bee13baa0218913fd47c55973b677ee58bbc56b8742be8ede4a87c64c2e0
|
|
| MD5 |
0708416e1838491246f3d8cba6a341bf
|
|
| BLAKE2b-256 |
5028a325598ee3c35d075f65166817bc0e4858705133fc6928d839bdee05521b
|
File details
Details for the file format_dedent-0.1.1-py3-none-any.whl.
File metadata
- Download URL: format_dedent-0.1.1-py3-none-any.whl
- Upload date:
- Size: 12.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
14eae90481e2c3498e9591a2f9849377f76364575d18166ca1a76f7c2eb32d37
|
|
| MD5 |
b62acd3706f891061ecece56d3109c9a
|
|
| BLAKE2b-256 |
469823da7b172727585f77caef46255d3ad19acc905c39a9137cec49b9fe89d1
|