PrePrintExec - A lightweight, non-intrusive debugging tool based on compiler AST transformation
Project description
PPE - PrePrintExec
A lightweight, non-intrusive debugging tool that uses AST transformation to print debug information before executing your Python code.
Why PPE?
In real-world development, inserting print statements is one of the most common debugging techniques. But doing so often clutters the codebase, requires cleanup before commits, and introduces friction in the development cycle. PPE (PrePrintExec) offers a clean, elegant alternative: reuse your existing inline comments as actionable debug logs.
With just a decorator, PPE automatically prints your comment string before executing each line—turning what used to be dead documentation into live, contextual insight. This helps you:
- 🧠 Understand complex workflows step by step
- 🔎 Instantly locate which operation failed during runtime
- 🧼 Avoid littering your code with temporary print() statements
- ⚙️ Toggle debugging on/off cleanly with a single annotation
Unlike traditional logging tools, PPE does not require you to write explicit logging code. It uses your comments—already written for humans—and makes them available at runtime when you need them most.
Installation
pip install ppe-debug
Quick Start
from ppe import ppe_debug
@ppe_debug
def my_function():
x = 10 ## Setting x value
y = x * 2 ## -
return y
result = my_function()
# Output:
# PPE: Setting x value
# PPE: y = x * 2
Features
- 🎯 Non-intrusive: Just add a decorator
- 🔧 Multiple debug modes: Custom messages, statement echoing, or variable inspection
- 🚀 Zero dependencies: Uses only Python standard library
- 🧹 Clean: Remove decorator to disable debugging
- 🔍 Comprehensive: Works with all Python statements
Usage
1. 🗒️ Custom Debug Messages
Use ## comments to print your custom debug messages.
@ppe_debug
def calculate():
result = 10 + 5 ## Adding two numbers
return result
# Output:
# PPE: Adding two numbers
2. 🔁 Statement Echoing
Use ## - to echo the actual statement being executed.
@ppe_debug
def calculate():
result = 10 + 5 ## -
return result
# Output:
# PPE: result = 10 + 5
3. 🧪 Variable Inspection
Use ## @var1,var2 to inspect variable values (prints after execution)
Use ## @before:var1,var2 to inspect variables before execution
Use ## @after:var1,var2 to explicitly inspect variables after execution
Note that if some variables are not defined at the time of inspection, they will trigger an error.
@ppe_debug
def calculate():
a = 1 ## @a
b = 2 ## @after:b
c = a + b ## @before:a,b
d = 10 ## @before:d
return c
# Output:
# PPE: [After] a=1
# PPE: [After] b=2
# PPE: [Before] a=1, b=2
# PPE: Variable inspection failed
4. ⚠️ Try-Wrapping
Use ## try: message to wrap the statement in a try-except block and catch any exception gracefully.
If no message is provided, the default message echoes the statement.
@ppe_debug
def run():
risky = 10 / 0 ## try: Attempting risky division
# Output:
# PPE: Attempting risky division
# PPE: Try-wrapped statement failed: division by zero
5. 🚧 Checkpoints
Use ## checkpoint: description to insert a clear visual marker at runtime.
If no message is provided, the line number will be used automatically.
@ppe_debug
def setup():
config = load_config() ## checkpoint: Loaded config
preprocess_data(config) ## checkpoint:
# Output:
# PPE: ===== Checkpoint: Loaded config =====
# PPE: ===== Checkpoint: Line 3 =====
For Researchers & Advanced Developers
At a glance, PPE might appear to be a simple utility that converts inline comments into print() statements. However, due to how the Python compiler works, this functionality requires much deeper intervention.
In Python, comments are discarded during the parsing stage and are not included in the Abstract Syntax Tree (AST), which is what most static analysis or transformation tools operate on. To retain access to the comment strings, PPE synchronizes the parsed AST (ast module from Python’s standard library) with the original source lines by matching line numbers, effectively reattaching comments to the corresponding AST nodes.
Rather than modifying source code textually or relying on deprecated tools like lib2to3, PPE implements a custom ast.NodeTransformer that walks and transforms the AST in a semantically-aware and structurally sound manner. The transformed tree is then dynamically compiled and executed, preserving all scope, bindings, and runtime behavior.
To make this accessible, we wrap the entire transformation and execution logic inside a lightweight decorator (@ppe_debug), so developers can instrument their functions non-invasively—without code duplication or string parsing hacks.
This project serves not just as a debugging tool, but as a practical demonstration of compiler-inspired instrumentation using Python's AST. If you're interested in programming languages, interpreters, or tooling, PPE is a compact but meaningful example of source-level augmentation with runtime effects.
See my blog post for a deeper dive into the details and design choices behind PPE.
License
MIT License - see LICENSE file for details.
Project details
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 ppe_debug-1.1.0.tar.gz.
File metadata
- Download URL: ppe_debug-1.1.0.tar.gz
- Upload date:
- Size: 8.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.11.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d355b082b65fc03fa49bfd5bcadef31daad1599e1fdaa6440218dc06ac0ab84b
|
|
| MD5 |
f13a6657115629df0bb5493925113549
|
|
| BLAKE2b-256 |
ddc0dede707c3ae48952081ced9855c794960c8093d81817fd82f8d2e873bd4a
|
File details
Details for the file ppe_debug-1.1.0-py3-none-any.whl.
File metadata
- Download URL: ppe_debug-1.1.0-py3-none-any.whl
- Upload date:
- Size: 8.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.11.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d254f65c8d2e526a9b16447ae8bf43680696977c989ac4bf92969f3a2e3f8c49
|
|
| MD5 |
94d11642362608a7fbb9034c656be06c
|
|
| BLAKE2b-256 |
20218902d09377e0d79cdc46e4b78a1f59f9ceea7ed666c88115b88d1bdea546
|