Skip to main content

A production-ready structured logging library for Python

Project description

LogCore 🔥

PyPI version Python versions CI License

A production-ready logging library for Python

LogCore provides a simple, structured, and extensible logging solution that works seamlessly for both small scripts and large microservices. It's designed as a drop-in alternative to Python's built-in logging with a focus on developer experience, observability, and production readiness.

✨ Features

  • 🚀 Simple API: Single entrypoint with intuitive configuration
  • 📊 Structured Logging: JSON and human-readable output formats
  • 🔗 Correlation IDs: Built-in request tracing support
  • ⏱️ Built-in Timing: Context managers for performance monitoring
  • 🛡️ Security: Automatic redaction of sensitive fields
  • 📁 File Rotation: Configurable log rotation and archival
  • 🎨 Colorized Output: Beautiful console logging with colors
  • ⚡ Async Support: Safe for asyncio applications
  • 🧵 Thread-safe: Concurrent logging without issues
  • 🌍 Environment Configuration: Configure via environment variables

🚀 Quick Start

Installation

pip install logcore

For colored output support:

pip install logcore[colors]

Basic Usage

from logcore import get_logger

# Create a logger
log = get_logger("myapp", level="INFO", json=True)

# Simple logging
log.info("Application started")
log.error("Something went wrong")

# Structured logging with extra fields
log.info("User login", user="alice", role="admin", success=True)

# Exception logging with automatic traceback
try:
    1 / 0
except Exception:
    log.exception("Division failed")

📖 Documentation

Configuration Options

LogCore can be configured through code or environment variables:

from logcore import get_logger

log = get_logger(
    name="myapp",              # Logger name
    level="INFO",              # DEBUG, INFO, WARNING, ERROR, CRITICAL
    json=True,                 # JSON output (False for human-readable)
    file="/path/to/app.log",   # Optional file logging
    correlation_id="req-123",  # Optional correlation ID
    max_file_size=10*1024*1024, # 10MB file size limit
    backup_count=5,            # Keep 5 backup files
    redact_fields={"password", "secret"}  # Fields to redact
)

Calling get_logger with the same name a second time and no extra arguments returns the cached instance. Passing configuration arguments when a logger already exists replaces it and emits a UserWarning — existing references to the old logger will stop receiving records.

Public API

from logcore import get_logger, LogLevel, set_correlation_id, get_correlation_id
Symbol Description
get_logger(name, ...) Create or retrieve a logger
LogLevel Enum of valid log levels (DEBUG, INFO, WARNING, ERROR, CRITICAL)
set_correlation_id(id) Set a correlation ID on the current context (thread/task) without a logger instance
get_correlation_id() Read the current correlation ID, or None if unset

set_correlation_id and get_correlation_id are useful in middleware that sets the ID before a logger is available:

from logcore import set_correlation_id, get_correlation_id

# In ASGI/WSGI middleware, before any logger is called:
set_correlation_id(request.headers.get("x-correlation-id"))

Environment Variables

Set configuration via environment variables:

export LOGCORE_LEVEL=DEBUG
export LOGCORE_JSON=true
export LOGCORE_FILE=/var/log/app.log
export LOGCORE_CORRELATION_ID=req-abc-123
export LOGCORE_REDACT_FIELDS=password,token,secret

Output Formats

JSON Format

{
  "timestamp": "2025-01-15T10:30:45.123456+00:00",
  "level": "INFO",
  "logger": "myapp",
  "message": "User login",
  "correlation_id": "req-123",
  "user": "alice",
  "success": true
}

Human-Readable Format

2025-01-15 10:30:45.123 INFO     myapp [cid=req-123]: User login user=alice success=true

Advanced Features

Correlation IDs for Request Tracing

from logcore import get_logger

log = get_logger("api")

# Set correlation ID for the entire request context
with log.with_correlation_id("req-abc-123"):
    log.info("Processing request")
    process_request()
    log.info("Request completed")

Performance Timing

# Measure execution time automatically
with log.time("database_query", level="DEBUG"):
    result = expensive_database_operation()

# Outputs:
# Starting database_query
# Completed database_query duration_ms=234.56

Exception Handling

try:
    risky_operation()
except Exception as e:
    log.exception("Operation failed", operation="risky_operation", user_id=123)
    # Automatically includes full traceback

Sensitive Data Redaction

Fields are partially masked — enough to confirm a value was present without leaking it:

log = get_logger("secure", redact_fields={"password", "token", "ssn"})

log.info("User data", username="alice", password="secret123", token="abc123", role="admin")
# Output: ... username=alice password=se*** token=a*** role=admin

Values of 4 characters or fewer are fully redacted ([REDACTED]). Longer values reveal a short prefix so you can correlate log lines without exposing the secret.

Default redacted fields: password, passwd, secret, token, key, api_key, access_token, auth, authorization, credential, private_key, cert, certificate.

OpenTelemetry Integration

When an active OpenTelemetry span exists, LogCore automatically injects trace_id and span_id into every log record — zero configuration required.

pip install logcore[otel]
from opentelemetry import trace
from opentelemetry.sdk.trace import TracerProvider
from logcore import get_logger

trace.set_tracer_provider(TracerProvider())
tracer = trace.get_tracer("myapp")
log = get_logger("myapp", json=True)

with tracer.start_as_current_span("handle-request"):
    log.info("Processing order", order_id=42)
    # {"trace_id": "4bf92f3577b34da6a3ce929d0e0e4736",
    #  "span_id": "00f067aa0ba902b7", "message": "Processing order", ...}

Outside a span the fields are simply absent — no noise in non-traced code paths. Works with any OTel-compatible backend (Jaeger, Zipkin, Honeycomb, Datadog, etc.).

File Logging with Rotation

log = get_logger(
    "myapp",
    file="/var/log/myapp.log",
    max_file_size=10 * 1024 * 1024,  # 10MB
    backup_count=5                    # Keep 5 old files
)

Files are automatically rotated:

  • myapp.log (current)
  • myapp.log.1 (previous)
  • myapp.log.2 (older)
  • etc.

Async Support

LogCore is fully compatible with asyncio:

import asyncio
from logcore import get_logger

async def main():
    log = get_logger("async_app")

    # Correlation IDs work across await boundaries
    with log.with_correlation_id():
        log.info("Starting async operation")
        await some_async_task()
        log.info("Async operation completed")

    # Async timing context manager
    async with log.time("async_operation"):
        await another_async_task()

asyncio.run(main())

Integration with Web Frameworks

Flask Example

from flask import Flask, request, g
from logcore import get_logger
import uuid

app = Flask(__name__)
log = get_logger("webapp")

@app.before_request
def before_request():
    g.correlation_id = request.headers.get('X-Correlation-ID', str(uuid.uuid4()))

@app.after_request
def after_request(response):
    with log.with_correlation_id(g.correlation_id):
        log.info(
            "Request completed",
            method=request.method,
            path=request.path,
            status_code=response.status_code,
            duration_ms=...  # Add timing logic
        )
    return response

@app.route('/users/<user_id>')
def get_user(user_id):
    with log.with_correlation_id(g.correlation_id):
        log.info("Fetching user", user_id=user_id)
        # ... your logic here

FastAPI Example

from fastapi import FastAPI, Request
from logcore import get_logger
import time
import uuid

app = FastAPI()
log = get_logger("api")

@app.middleware("http")
async def logging_middleware(request: Request, call_next):
    correlation_id = request.headers.get("x-correlation-id", str(uuid.uuid4()))
    start_time = time.time()

    with log.with_correlation_id(correlation_id):
        log.info("Request started", method=request.method, url=str(request.url))

        response = await call_next(request)

        duration = (time.time() - start_time) * 1000
        log.info(
            "Request completed",
            status_code=response.status_code,
            duration_ms=round(duration, 2)
        )

    response.headers["x-correlation-id"] = correlation_id
    return response

⚡ Performance

Measured on Python 3.12, Apple M-series, writing to /dev/null (I/O excluded):

Mode µs / call Notes
stdlib logging (text) ~6 µs baseline
stdlib + manual JSON formatter ~7 µs +1 µs
LogCore JSON ~13 µs +7 µs for structured output
LogCore text (colored) ~36 µs +30 µs for strftime + color

JSON mode is the recommended default for production — it costs ~7 µs per call over stdlib and produces machine-readable output that log aggregators can query directly.

Run the benchmark yourself: python examples/benchmark.py

🆚 Comparison with Other Libraries

vs. Built-in logging

Feature LogCore Built-in logging
Setup complexity ⭐⭐⭐⭐⭐ Single line ⭐⭐ Complex setup
Structured logging ⭐⭐⭐⭐⭐ Built-in ⭐⭐ Manual implementation
JSON output ⭐⭐⭐⭐⭐ Automatic ⭐⭐ Custom formatter needed
Correlation IDs ⭐⭐⭐⭐⭐ Built-in ⭐ Custom context needed
Security ⭐⭐⭐⭐⭐ Auto-redaction ⭐ Manual filtering
Colors ⭐⭐⭐⭐⭐ Auto-detected ⭐⭐ Third-party needed

vs. loguru

Feature LogCore Loguru
Production focus ⭐⭐⭐⭐⭐ Enterprise-ready ⭐⭐⭐⭐ Great for development
Correlation IDs ⭐⭐⭐⭐⭐ Built-in context ⭐⭐ Manual binding
Security ⭐⭐⭐⭐⭐ Auto-redaction ⭐⭐ Manual filtering
Async support ⭐⭐⭐⭐⭐ Context-aware ⭐⭐⭐ Basic support
Performance ⭐⭐⭐⭐ Good ⭐⭐⭐⭐⭐ Excellent
Ecosystem ⭐⭐⭐⭐⭐ Standard logging ⭐⭐⭐ Custom approach

🛠️ Development

Setup

git clone https://github.com/SarkarRana/logcore.git
cd logcore

# Install development dependencies
pip install -e ".[dev]"

# Install pre-commit hooks
pre-commit install

Running Tests

# Run all tests
pytest

# Run with coverage
pytest --cov=logcore

# Run specific test categories
pytest -m "not slow"          # Skip slow tests
pytest -m integration         # Run integration tests only

Code Quality

# Format code
black logcore tests
isort logcore tests

# Lint
flake8 logcore tests

# Type checking
mypy logcore

📄 License

This project is licensed under the MIT License - see the LICENSE file for details.

🤝 Contributing

Contributions are welcome! Please read our Contributing Guide for details on our code of conduct and the process for submitting pull requests.

🎯 Roadmap

Shipped

  • OpenTelemetry: Automatic trace/span ID injection from active spans (v0.1.4)
  • Async support: AsyncTimer with isolated correlation IDs per task (v0.1.4)
  • Partial masking: Secrets show a short prefix, not just [REDACTED] (v0.1.4)
  • Accurate caller info: filename, lineno, and funcName now reflect the real call site (v0.1.5)
  • Reconfiguration warning: get_logger emits UserWarning when replacing a cached logger (v0.1.5)
  • LogLevel, set_correlation_id, get_correlation_id promoted to top-level public API (v0.1.5)

Planned

  • Sentry integration: Automatic error forwarding with structured context
  • Log sampling: Rate-based sampling to protect downstream systems under load
  • Async batching: Buffer and flush writes for lower-latency hot paths
  • OTLP export: Direct log shipping to OpenTelemetry collectors
  • Kubernetes metadata: Pod/node/namespace injection via downward API env vars

💖 Support

If you find LogCore useful, please consider:

  • ⭐ Starring the repository
  • 🐛 Reporting bugs and issues
  • 💡 Suggesting new features
  • 📖 Improving documentation
  • 💻 Contributing code

Built with ❤️ for the Python community

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

logcore-0.1.5.tar.gz (25.2 kB view details)

Uploaded Source

Built Distribution

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

logcore-0.1.5-py3-none-any.whl (16.5 kB view details)

Uploaded Python 3

File details

Details for the file logcore-0.1.5.tar.gz.

File metadata

  • Download URL: logcore-0.1.5.tar.gz
  • Upload date:
  • Size: 25.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for logcore-0.1.5.tar.gz
Algorithm Hash digest
SHA256 c7e9ddaec2529ae1aa61f71cc561064a8f940458cf649156691be5cbeeed683d
MD5 a78bcc8b06dc35c62ff63f3c8600fefb
BLAKE2b-256 44b14d3c01d32d364c93566ceb7a719e5fc24ab75ff73f4cded0067f2f23c10b

See more details on using hashes here.

File details

Details for the file logcore-0.1.5-py3-none-any.whl.

File metadata

  • Download URL: logcore-0.1.5-py3-none-any.whl
  • Upload date:
  • Size: 16.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for logcore-0.1.5-py3-none-any.whl
Algorithm Hash digest
SHA256 47464eb946e18f8e889413e43d0883b95126d54ac5f173975c0baab1a3df0b21
MD5 08fb4530077c821c872c65f87389bdd2
BLAKE2b-256 631bd69a1830dffe4d41e17e6e7461faa444d01695c3b10555a250492cc924bd

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