A Python debugging and analysis toolkit with beautiful, colorful terminal output. Provides decorators for automatic logging, execution timing, and real-time variable tracking.
Project description
Steely
A Python debugging and analysis toolkit with beautiful, colorful terminal output.
Steely provides powerful decorators that help developers understand their code's behavior at runtime through automatic logging, execution timing, and real-time variable tracking.
Features
- Dan.log - Automatic function logging with start/success/error tracking
- Dan.cronos - Execution time measurement and profiling
- Dan.scan - Real-time variable tracking with type-colored output
- Logger - Flexible, thread-safe logging with color-coded output and file support
- pprint - Pretty print with caller location information for easy debugging
- FastAPI Integration - Record API requests as Postman collections or curl commands
Installation
Using pip
pip install steely
Using uv
uv add steely
From source
git clone https://github.com/your-username/steely.git
cd steely
pip install -e .
Requirements
- Python 3.9 or higher
- No external dependencies
Quick Start
from steely import Dan
from steely.pprint import pprint
from steely.logger import Logger
# Log function execution
@Dan.log
def fetch_data(url):
return {"data": "example"}
# Measure execution time
@Dan.cronos
def slow_operation():
import time
time.sleep(1)
return "done"
# Track variables in real-time
@Dan.scan
def calculate(a, b):
total = a + b
squared = total ** 2
return squared
# Pretty print with location info
def debug_function(x, y):
result = x + y
pprint(f"Result is: {result}")
return result
# Set global app name for consistent logging
Logger.set_global_app_name("MyApp")
# Run the functions
fetch_data("https://api.example.com")
slow_operation()
calculate(3, 4)
debug_function(10, 20)
FastAPI Quick Start
from fastapi import FastAPI
from steely.fastapi import recorder
app = FastAPI()
# Automatically record requests as Postman collections
@app.get("/users/{user_id}")
@recorder.postman()
async def get_user(user_id: int):
return {"user_id": user_id, "name": "John"}
# Also generate curl commands for testing
@app.post("/users")
@recorder.curl()
async def create_user(name: str):
return {"id": 123, "name": name}
Decorators
Dan.log
The log decorator automatically tracks function execution lifecycle - when it starts, succeeds, or fails.
from steely import Dan
@Dan.log
def process_user(user_id):
# Your code here
return {"id": user_id, "status": "processed"}
process_user(123)
Output:
[PROCESS_USER] [START]: Function called
[PROCESS_USER] [SUCCESS]: Function completed
With async functions
@Dan.log
async def fetch_api_data(endpoint):
async with aiohttp.ClientSession() as session:
async with session.get(endpoint) as response:
return await response.json()
Error tracking
When an exception occurs, the decorator logs the error before re-raising it:
@Dan.log
def risky_operation():
raise ValueError("Something went wrong")
risky_operation()
# Output:
# [RISKY_OPERATION] [START]: Function called
# [RISKY_OPERATION] [ERROR]: Something went wrong
# Traceback...
Dan.cronos
Named after the Greek god of time, cronos measures and reports function execution time.
from steely import Dan
import time
@Dan.cronos
def compute_heavy_task(n):
time.sleep(n)
return sum(range(n * 1000000))
compute_heavy_task(2)
Output:
[*-CRONOS-*] [COMPUTE_HEAVY_TASK] [TEST-RESULT]: Total Time Elapsed: 0:00:02.001234
With async functions
import asyncio
@Dan.cronos
async def async_task():
await asyncio.sleep(0.5)
return "completed"
asyncio.run(async_task())
# Output: Total Time Elapsed: 0:00:00.500xxx
Dan.scan
The scan decorator provides real-time variable tracking with beautiful, type-colored output. It shows every variable assignment, its type, value, and the line number where it was set.
from steely import Dan
@Dan.scan
def example(x, y):
name = "Alice"
numbers = [1, 2, 3]
total = x + y
data = {"key": "value"}
return total
example(10, 20)
Output:
┌─────────────────────────────────────────────────────────────────────────────────────
│ ⚡ SCAN: example
│ module: __main__
├─────────────────────────────────────────────────────────────────────────────────────
│ Parameters:
│ ◈ x: int → 10
│ ◈ y: int → 20
├─────────────────────────────────────────────────────────────────────────────────────
│ Variables:
│ ▸ line 4 ◈ name ◉ str → 'Alice'
│ ▸ line 5 ◈ numbers ◉ list → [1, 2, 3]
│ ▸ line 6 ◈ total ◉ int → 30
│ ▸ line 7 ◈ data ◉ dict → {'key': 'value'}
├─────────────────────────────────────────────────────────────────────────────────────
│ ⟼ Return: int → 30
└───────────────────────────────────────────────────────────── elapsed: 0.12ms ──────
Type Colors
Each Python type is displayed in a distinct color for easy visual identification:
| Type | Color |
|---|---|
int |
Blue |
float |
Deep Blue |
str |
Orange |
bool |
Magenta |
None |
Gray |
list |
Sea Green |
dict |
Gold |
tuple |
Light Purple |
set |
Pink |
callable |
Cyan |
class |
Lavender |
Async Functions
For async functions, variable tracking is disabled (due to event loop tracing limitations), but parameters and return values are still displayed:
@Dan.scan
async def fetch_data(url):
response = await some_api_call(url)
return response
await fetch_data("https://api.example.com")
Combining Decorators
You can stack multiple decorators to get combined functionality:
from steely import Dan
@Dan.log
@Dan.cronos
@Dan.scan
def important_operation(data):
processed = data.upper()
result = len(processed)
return result
important_operation("hello world")
This will:
- Log the function start/success
- Measure execution time
- Track all variable assignments
Logger
The Logger class provides a flexible, thread-safe logging system with color-coded terminal output and optional file logging.
Basic Usage
from steely.logger import Logger
# Create a logger instance
logger = Logger("MyApp", "database")
# Log messages with different levels
logger.info("Connected to database")
logger.success("Query executed successfully")
logger.warning("Connection pool running low")
logger.error("Query timeout after 30s")
Output:
25-11-2025 14:30:00 - [MYAPP] [DATABASE] [INFO]: Connected to database
25-11-2025 14:30:00 - [MYAPP] [DATABASE] [SUCCESS]: Query executed successfully
25-11-2025 14:30:01 - [MYAPP] [DATABASE] [WARNING]: Connection pool running low
25-11-2025 14:30:02 - [MYAPP] [DATABASE] [ERROR]: Query timeout after 30s
Log Levels
| Level | Color | Method | Use Case |
|---|---|---|---|
INFO |
Cyan | logger.info() |
General information |
START |
Cyan | logger.start() |
Process initialization |
SUCCESS |
Green | logger.success() |
Successful operations |
OK |
Green | logger.ok() |
Confirmations |
WARNING |
Yellow | logger.warning() |
Warning messages |
ALERT |
Yellow | logger.alert() |
Alert notifications |
ERROR |
Red | logger.error() |
Error messages |
CRITICAL |
Red | logger.critical() |
Critical errors |
FATAL |
Red | logger.fatal() |
Fatal errors |
FAIL |
Red | logger.fail() |
Failed operations |
FAULT |
Red | logger.fault() |
System faults |
TEST |
Blue | logger.test() |
Test messages |
TEST-RESULT |
Blue | logger.test_result() |
Test results |
File Logging
Enable file logging by providing a destination directory:
from steely.logger import Logger
# Logs will be saved to /var/log/myapp/DD-MM-YYYY.log
logger = Logger("MyApp", "api", destination="/var/log/myapp")
logger.info("Request received") # Printed AND saved to file
logger.error("Request failed") # Printed AND saved to file
Log files are automatically named with the current date (e.g., 25-11-2025.log) and appended throughout the day.
Additional Tags
Add custom tags to your log messages for better filtering:
from steely.logger import Logger
# Tags defined at creation apply to all messages
logger = Logger("MyApp", "auth", session_id="abc123", user_id="42")
logger.info("User logged in")
# Output: [MYAPP] [AUTH] [ABC123] [42] [INFO]: User logged in
# Or add tags per-message
logger.info("Action performed", request_id="req-789")
# Output: [MYAPP] [AUTH] [ABC123] [42] [REQ-789] [INFO]: Action performed
Using log() Method Directly
For dynamic log levels, use the log() method:
from steely.logger import Logger
logger = Logger("MyApp", "main")
# Using log() with any level
logger.log("INFO", "Application started")
logger.log("SUCCESS", "Initialization complete")
logger.log("ERROR", "Connection failed")
# Logger instances are callable
logger("WARNING", "This also works!")
Setting Global App Name
Use set_global_app_name() to set a global application name for all @log decorated functions:
from steely.logger import Logger
from steely import Dan
# Set global app name once
Logger.set_global_app_name("MyApp")
# All @log decorated functions will now use "MyApp" instead of module name
@Dan.log
def process_data():
return "processed"
@Dan.log
def fetch_api():
return "fetched"
# Both functions will log with [MYAPP] prefix
process_data() # Logs: [MYAPP] [PROCESS_DATA] [START]: Function Execution Started...
fetch_api() # Logs: [MYAPP] [FETCH_API] [START]: Function Execution Started...
This is particularly useful for applications where you want consistent app naming across all logged functions without specifying it for each logger instance.
Screen Clearing
Enable screen clearing for a cleaner terminal experience:
from steely.logger import Logger
# Clear screen before each log message
logger = Logger("MyApp", "installer", clean=True)
logger.info("Step 1: Downloading...") # Clears screen, then prints
logger.info("Step 2: Installing...") # Clears screen, then prints
logger.success("Installation complete!")
Debug Mode
Control debug output with the debug parameter:
from steely.logger import Logger
# Debug mode enabled (default) - appends "_debug" to log directory
logger = Logger("MyApp", "main", destination="/logs", debug=True)
# Logs go to: /logs_debug/DD-MM-YYYY.log
# Production mode
logger = Logger("MyApp", "main", destination="/logs", debug=False)
# Logs go to: /logs/DD-MM-YYYY.log
Thread Safety
All logging operations run in separate threads for non-blocking behavior:
from steely.logger import Logger
import time
logger = Logger("MyApp", "worker")
def process_items(items):
for item in items:
logger.info(f"Processing {item}") # Non-blocking
# Heavy processing here...
time.sleep(0.1)
logger.success("All items processed")
# Logging won't slow down your processing
process_items(["item1", "item2", "item3"])
Complete Example
from steely.logger import Logger
class DatabaseService:
def __init__(self):
self.logger = Logger("MyApp", "database", destination="./logs")
def connect(self, host, port):
self.logger.start(f"Connecting to {host}:{port}")
try:
# Connection logic here...
self.logger.success("Connected successfully")
return True
except Exception as e:
self.logger.error(f"Connection failed: {e}")
return False
def query(self, sql):
self.logger.info(f"Executing query: {sql[:50]}...")
try:
# Query logic here...
self.logger.success("Query executed")
return {"rows": 42}
except Exception as e:
self.logger.error(f"Query failed: {e}")
raise
# Usage
db = DatabaseService()
db.connect("localhost", 5432)
db.query("SELECT * FROM users WHERE active = true")
pprint
The pprint function enhances the standard print with automatic caller location information, making debugging faster and more efficient. It shows the file path and line number in a format that's clickable in most IDEs and terminals.
Basic Usage
from steely.pprint import pprint
def calculate_total(items):
subtotal = sum(items)
pprint(f"Subtotal: {subtotal}")
tax = subtotal * 0.1
pprint(f"Tax: {tax}")
total = subtotal + tax
pprint(f"Total: {total}")
return total
calculate_total([10, 20, 30])
Output:
[PPRINT] File "/path/to/your/script.py", line 4
Subtotal: 60
[PPRINT] File "/path/to/your/script.py", line 7
Tax: 6.0
[PPRINT] File "/path/to/your/script.py", line 10
Total: 66.0
With Custom Colors
You can customize the color of the output using the color parameter:
from steely.pprint import pprint
from steely.design import UnicodeColors
# Success message in green
pprint("Operation completed!", color=UnicodeColors.success)
# Error message in red
pprint("Something went wrong!", color=UnicodeColors.fail)
# Info message in cyan
pprint("Processing data...", color=UnicodeColors.success_cyan)
# Warning message in yellow
pprint("Low memory warning", color=UnicodeColors.alert)
Clickable Links
The file path and line number are formatted to be clickable in most modern IDEs and terminals:
- VS Code: Ctrl+Click (Cmd+Click on Mac) to jump to the line
- PyCharm: Click the link to navigate to the source
- Terminal: Many terminals support clicking file:line patterns
Use Cases
Perfect for:
- Quick debugging without setting up a full logger
- Tracking execution flow through complex functions
- Comparing values at different points in code
- Temporary debug statements that are easy to locate and remove later
FastAPI Integration
Steely provides decorators for FastAPI that automatically record your API requests for testing, documentation, and debugging purposes.
Postman Recorder
The postman decorator automatically captures requests and responses, generating Postman Collection v2.1 format files that can be imported directly into Postman.
Basic Usage
from fastapi import FastAPI
from steely.fastapi import recorder
app = FastAPI()
@app.get("/users/{user_id}")
@recorder.postman()
async def get_user(user_id: int):
return {"user_id": user_id, "name": "John Doe", "email": "john@example.com"}
@app.post("/users")
@recorder.postman()
async def create_user(name: str, email: str):
return {"id": 123, "name": name, "email": email}
Collections are saved to ./.postman_collections/<function_name>.json by default.
Grouping Endpoints
Group multiple endpoints into a single collection:
@app.get("/users")
@recorder.postman(collection_name="user_api")
async def list_users():
return [{"id": 1, "name": "John"}, {"id": 2, "name": "Jane"}]
@app.get("/users/{user_id}")
@recorder.postman(collection_name="user_api")
async def get_user(user_id: int):
return {"id": user_id, "name": "John"}
@app.post("/users")
@recorder.postman(collection_name="user_api")
async def create_user(name: str):
return {"id": 123, "name": name}
All three endpoints will be saved in ./.postman_collections/user_api.json.
Custom Output Directory
@app.get("/api/data")
@recorder.postman(output_dir="./docs/postman")
async def get_data():
return {"data": "example"}
Curl Recorder
The curl decorator captures requests and generates executable curl commands, perfect for sharing API examples or creating test scripts.
Basic Usage
from fastapi import FastAPI
from steely.fastapi import recorder
app = FastAPI()
@app.get("/users/{user_id}")
@recorder.curl()
async def get_user(user_id: int, include_email: bool = False):
return {"user_id": user_id, "name": "John"}
@app.post("/users")
@recorder.curl()
async def create_user(name: str, email: str):
return {"id": 123, "name": name, "email": email}
Scripts are saved to ./.curl_scripts/<function_name>.sh and are automatically made executable.
Running Generated Scripts
# Execute the generated curl commands
bash ./.curl_scripts/get_user.sh
# Or make it executable and run directly
chmod +x ./.curl_scripts/get_user.sh
./.curl_scripts/get_user.sh
Grouping Commands
Group multiple endpoints into a single script:
@app.get("/users")
@recorder.curl(script_name="user_api")
async def list_users():
return [{"id": 1, "name": "John"}]
@app.post("/users")
@recorder.curl(script_name="user_api")
async def create_user(name: str):
return {"id": 123, "name": name}
All commands will be appended to ./.curl_scripts/user_api.sh.
Custom Output Directory
@app.get("/api/data")
@recorder.curl(output_dir="./scripts")
async def get_data():
return {"data": "example"}
Combining Recorders
You can use both decorators together to generate both Postman collections and curl scripts:
from fastapi import FastAPI
from steely.fastapi import recorder
app = FastAPI()
@app.get("/users/{user_id}")
@recorder.postman(collection_name="api_docs")
@recorder.curl(script_name="api_tests")
async def get_user(user_id: int):
return {"user_id": user_id, "name": "John"}
This will create:
./.postman_collections/api_docs.json- Postman collection for documentation./.curl_scripts/api_tests.sh- Executable curl commands for testing
Benefits
- Automatic Documentation: Generate Postman collections from real API traffic
- Testing: Create reproducible test scripts without manual work
- Team Collaboration: Share API examples with teammates easily
- CI/CD Integration: Use generated curl scripts in your automated tests
- Zero Configuration: Works out of the box with sensible defaults
Advanced Usage
Using Individual Decorators
You can import decorators individually if you prefer:
from steely import cronos, log, scan
@log
def my_function():
pass
@cronos
def timed_function():
pass
@scan
def tracked_function():
pass
Using Design Components
Steely's design module provides terminal styling utilities you can use in your own code:
from steely.design import UnicodeColors as C, Symbols as S, TypeColors
# Colored output
print(f"{C.green}Success!{C.reset}")
print(f"{C.bold}{C.red}Error!{C.reset}")
# Symbols
print(f"{S.CHECK} Task completed")
print(f"{S.ARROW_RIGHT} Next step")
print(f"{S.CROSS} Failed")
# Type-based colors
value = [1, 2, 3]
color = TypeColors.get_color(value)
print(f"{color}{value}{C.reset}")
Examples
Web API Debugging
from steely import Dan
import requests
@Dan.log
@Dan.cronos
def fetch_user(user_id):
response = requests.get(f"https://api.example.com/users/{user_id}")
return response.json()
user = fetch_user(42)
Algorithm Profiling
from steely import Dan
@Dan.cronos
def bubble_sort(arr):
n = len(arr)
for i in range(n):
for j in range(0, n-i-1):
if arr[j] > arr[j+1]:
arr[j], arr[j+1] = arr[j+1], arr[j]
return arr
@Dan.cronos
def quick_sort(arr):
if len(arr) <= 1:
return arr
pivot = arr[len(arr) // 2]
left = [x for x in arr if x < pivot]
middle = [x for x in arr if x == pivot]
right = [x for x in arr if x > pivot]
return quick_sort(left) + middle + quick_sort(right)
import random
data = [random.randint(0, 1000) for _ in range(1000)]
bubble_sort(data.copy()) # See timing
quick_sort(data.copy()) # Compare timing
Debugging Complex Logic
from steely import Dan
@Dan.scan
def calculate_discount(price, quantity, member_status):
subtotal = price * quantity
if member_status == "gold":
discount_rate = 0.20
elif member_status == "silver":
discount_rate = 0.10
else:
discount_rate = 0.0
discount = subtotal * discount_rate
final_price = subtotal - discount
return final_price
calculate_discount(99.99, 3, "gold")
# See exactly how each variable is computed
Framework Compatibility
All Steely decorators preserve the original function's signature, making them compatible with frameworks that rely on function introspection:
FastAPI
from fastapi import FastAPI
from steely import Dan
app = FastAPI()
@app.get("/users/{user_id}")
@Dan.log
@Dan.cronos
async def get_user(user_id: int, include_email: bool = False):
return {"user_id": user_id, "include_email": include_email}
Flask
from flask import Flask
from steely import Dan
app = Flask(__name__)
@app.route("/hello/<name>")
@Dan.log
def hello(name):
return f"Hello, {name}!"
License
MIT License
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 Distributions
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 steely-1.0.1.4-py3-none-any.whl.
File metadata
- Download URL: steely-1.0.1.4-py3-none-any.whl
- Upload date:
- Size: 36.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0ebb22694ebc4a6151997ecacac8faf59f5fe3e02a1200737b46e914e06de619
|
|
| MD5 |
5b9b5228108341dc8ff47b24b8335765
|
|
| BLAKE2b-256 |
2b4ebd3e1af0353dbd6907638e5e1c97948fcc204247fad949630031730f0208
|