Skip to content

15r10nk/format-dedent

Repository files navigation

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?

ci Docs pypi version Python Versions PyPI - Downloads coverage GitHub Sponsors

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 dedent import 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:

  1. Detected that the string has no leading whitespace (left-aligned)
  2. Wrapped it with dedent() for consistency
  3. Added the import statement automatically
  4. 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 πŸ₯ˆ

logfire


🧠 How It Works

  1. Parse β€” Uses Python's AST module to analyze source code
  2. Find β€” Locates all dedent() or textwrap.dedent() calls with string arguments
  3. Analyze β€” Determines the appropriate indentation level based on context
  4. Validate β€” Ensures dedent(original) == dedent(formatted) (behavior unchanged)
  5. 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.

About

formats textwrap.dedent("""multiline string ...""")

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

 
 
 

Contributors

Languages