Skip to main content

Static analysis tool for tracing exception flow through Python codebases

Project description

flow

Static analysis tool for tracing exception flow through Python codebases.

What can escape from my API endpoints? Flow answers this by parsing your code, building a call graph, and computing which exceptions propagate to each entrypoint.

Quick Start

pip install bubble-analysis
# Check Flask routes for uncaught exceptions
bubble flask audit -d /path/to/project

# Check FastAPI routes
bubble fastapi audit -d /path/to/project

# Deep dive into one function
bubble escapes create_user -d /path/to/project

# Visualize the call tree
bubble trace create_user -d /path/to/project

What It Does

Flow finds your HTTP routes and CLI scripts, traces the call graph, and reports which exceptions can escape:

$ flow flask audit

Scanning 23 flask entrypoints...

3 entrypoints have uncaught exceptions:

  POST /users/import
    └─ FileNotFoundError (importers.py:45)
    └─ ValidationError (validators.py:12)

  GET /reports/{id}
    └─ PermissionError (auth.py:89)

20 entrypoints fully covered by exception handlers

For a specific endpoint, see the full picture:

$ flow escapes create_user

Exceptions that can escape from POST /users:

  FRAMEWORK-HANDLED (converted to HTTP response):
    HTTPException
      └─ becomes: HTTP 404
      └─ raised in: routes/users.py:45 (get_user) [high confidence]

  CAUGHT BY GLOBAL HANDLER:
    ValidationError (@errorhandler(AppError))
      └─ raised in: validators.py:27 (validate_input) [high confidence]

  UNCAUGHT (will propagate to caller):
    ConnectionError
      └─ raised in: db/client.py:45 (execute) [medium confidence]
      └─ call path: create_user → save_user → db.execute

Visualize as a tree:

$ flow trace create_user

POST /users  → escapes: ValidationError, ConnectionError
├── validate_input()  → ValidationError
│   └── raises ValidationError (validators.py:27)
└── save_user()  → ConnectionError
    └── db.execute()  → ConnectionError
        └── raises ConnectionError (db/client.py:45)

Features

  • Entrypoint detection: Flask routes, FastAPI routes, CLI scripts (if __name__ == "__main__")
  • Global handler awareness: Understands @errorhandler, add_exception_handler
  • Exception hierarchy: Knows that catching AppError also catches ValidationError if it's a subclass
  • Polymorphism: Expands abstract method calls to all concrete implementations
  • Framework-handled exceptions: Detects HTTPException, ValidationError → HTTP responses
  • Confidence levels: Shows high/medium/low confidence based on resolution quality
  • Resolution modes: --strict for precision, --aggressive for recall
  • Exception stubs: Declare what external libraries can raise (requests, sqlalchemy, etc.)
  • JSON output: All commands support -f json for CI/automation
  • Caching: SQLite-based caching for fast repeated analysis

Commands

Core Commands (framework-agnostic)

Command Description
bubble raises <exception> Find all places an exception is raised
bubble escapes <function> Show what can escape from a specific function
bubble callers <function> Find all callers of a function
bubble catches <exception> Find all places an exception is caught
bubble trace <function> Visualize exception flow as a call tree
bubble exceptions Show the exception class hierarchy
bubble subclasses <class> Show class inheritance tree
bubble stubs <action> Manage exception stubs (list, init, validate)
bubble stats Show codebase statistics

Framework-Specific Commands

Command Description
bubble flask audit Check Flask routes for escaping exceptions
bubble flask entrypoints List Flask HTTP routes
bubble flask routes-to <exc> Which Flask routes can trigger this exception?
bubble fastapi audit Check FastAPI routes for escaping exceptions
bubble fastapi entrypoints List FastAPI HTTP routes
bubble fastapi routes-to <exc> Which FastAPI routes can trigger this exception?
bubble cli audit Check CLI scripts for escaping exceptions
bubble cli entrypoints List CLI scripts
bubble cli scripts-to <exc> Which CLI scripts can trigger this exception?

All commands accept:

  • -d, --directory: Directory to analyze (default: current)
  • -f, --format: Output format (text or json)
  • --no-cache: Disable caching

The escapes command accepts additional flags:

  • --strict: High precision mode - only includes precisely resolved calls
  • --aggressive: High recall mode - includes fuzzy matches

Supported Frameworks

Detected automatically:

  • Flask (@app.route, @blueprint.route, @app.errorhandler)
  • FastAPI (@router.get/post/put/delete, add_exception_handler)
  • CLI scripts (if __name__ == "__main__")

Not yet supported:

  • Django
  • Celery tasks
  • Scheduled jobs (APScheduler, etc.)

Custom patterns can be added via .flow/detectors/ (run bubble init to set up).

Adding Custom Detectors

Flow is designed to be extended with AI coding agents. The detector interface is intentionally simple: implement a protocol that returns entrypoints and handlers from parsed code.

To add support for a new framework (Django, Celery, your internal RPC layer, etc.):

  1. Run bubble init to create the .flow/ directory structure
  2. Point your AI agent at flow/protocols.py to see the EntrypointDetector interface
  3. Ask it to implement a detector for your framework in .flow/detectors/

Example prompt for an AI agent:

Read flow/protocols.py and flow/detectors.py to understand how entrypoint
detection works. Then implement a detector for Django that finds:
- Views decorated with @api_view
- Class-based views inheriting from APIView
- URL patterns in urls.py

Put the implementation in .flow/detectors/django.py

The detector just needs to implement:

  • detect_entrypoints(functions, classes, ...) → list of Entrypoint
  • detect_global_handlers(...) → list of GlobalHandler

Flow will automatically load any .py files in .flow/detectors/ and use them alongside the built-in Flask/FastAPI detectors.

Configuration

Flow can be configured via .flow/config.yaml:

resolution_mode: default  # "strict", "default", or "aggressive"
exclude:
  - vendor
  - migrations

Exception Stubs

Flow includes built-in stubs for common libraries (requests, sqlalchemy, httpx, redis, boto3). These declare what exceptions external library functions can raise.

Add custom stubs in .flow/stubs/:

# .flow/stubs/mylib.yaml
mylib:
  do_thing:
    - MyLibError
    - TimeoutError

Manage stubs with bubble stubs list and bubble stubs validate.

How It Works

  1. Parse: LibCST parses all Python files
  2. Extract: Find functions, classes, raise/catch sites, calls, entrypoints
  3. Build call graph: Track who calls whom, resolve method calls
  4. Propagate: Fixed-point iteration computes which exceptions escape each function
  5. Report: For each entrypoint, show caught vs uncaught exceptions

Limitations

  • Over-approximation: May report more exceptions than actually possible (e.g., all implementations of an abstract method)
  • Under-approximation: Dynamic dispatch, eval(), and external libraries can't be fully traced
  • No runtime info: Analysis is purely static

Development

git clone https://github.com/ianm199/flow
cd flow
pip install -e ".[dev]"
pytest

License

MIT

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

bubble_analysis-0.2.0.tar.gz (75.5 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

bubble_analysis-0.2.0-py3-none-any.whl (79.0 kB view details)

Uploaded Python 3

File details

Details for the file bubble_analysis-0.2.0.tar.gz.

File metadata

  • Download URL: bubble_analysis-0.2.0.tar.gz
  • Upload date:
  • Size: 75.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.2

File hashes

Hashes for bubble_analysis-0.2.0.tar.gz
Algorithm Hash digest
SHA256 aa644bdf3e05f3e4c96ace1f9bdd92c0098638c8a2e3d23ecdb56f87db2c7fff
MD5 94970fd38164ff9cd7d7a618cb8fdf8a
BLAKE2b-256 62a2b7e7eb424b5a9a249d6939ce7f239861befd7b114f5981fe43c1a73f82de

See more details on using hashes here.

File details

Details for the file bubble_analysis-0.2.0-py3-none-any.whl.

File metadata

File hashes

Hashes for bubble_analysis-0.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 0db108b1bf860344b612dd4aaf82a4312cbe1e906575f87562ef53252ee972a6
MD5 8dc350e66f962feb7107164a1ac560ce
BLAKE2b-256 aca93527a42f0a045dbcaf3834b784b24f65415ca744f0c8dc0b763de70ce315

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page