Ready for Production#

This guide covers the essentials for running Oban in production: logging, job maintenance, and error monitoring.

Running Embedded#

If you’re running Oban embedded in your application instead of using the CLI, you’ll need to configure a few things manually.

Enable Logging#

Attach the telemetry logger during application startup:

from oban.telemetry import logger as telemetry_logger

telemetry_logger.attach()

The logger emits JSON-encoded structured logs at the INFO level by default. You can customize the log level:

telemetry_logger.attach(level=logging.DEBUG)

Configure Job Maintenance#

Pruning and orphan rescue are enabled by default, but you can customize them:

from oban import Oban

pool = await Oban.create_pool()

oban = Oban(
    pool=pool,
    queues={"default": 10},
    pruner={"max_age": 3_600},  # Keep jobs for 1 hour
    lifeline={"interval": 30},  # Check for orphans every 30 seconds
)

See the Job Maintenance guide for more details on job retention.

Enable Metrics for Web Dashboard#

To use the Oban Web dashboard, enable metrics broadcasting:

oban = Oban(pool=pool, queues={"default": 10}, metrics=True)

See the Web Dashboard guide for setup instructions.

Sizing the Connection Pool#

Oban runs its queries through a psycopg connection pool bounded by pool_min_size and pool_max_size, which default to 1 and 10.

A common mistake is to assume you need one connection per concurrently running job. You don’t. Workers only borrow a connection briefly—to fetch a batch of jobs or acknowledge results—then return it. A queue with a limit of 100 does not hold 100 connections. What draws from the pool are short, concurrent operations: producers fetching jobs, maintenance tasks (stager, scheduler, pruner, lifeline), and your application calling enqueue. The PostgreSQL notifier keeps its own dedicated connection outside the pool, so pubsub needs no pool slot.

The default of 10 is comfortable for most deployments. Raise pool_max_size if you enqueue at high volume from many concurrent requests, or if you see timeouts acquiring a connection. Don’t oversize: every connection is a real backend, and the total across all your nodes counts against Postgres’s max_connections (and any PgBouncer limits).

Set the bounds in oban.toml:

pool_min_size = 2
pool_max_size = 20

Or directly in embedded mode via Oban.create_pool(min_size=2, max_size=20).

Ship It!#

Whether you’re using the CLI or embedded mode, you now have:

  • Structured logging for debugging and monitoring

  • Automatic job pruning to prevent unbounded table growth

  • Orphan recovery for interrupted jobs

The CLI handles all of this automatically, making it the recommended approach for most production deployments.