Skip to main content

Log without the setup via a pre-configured structlog logger with optional Sentry integration

Project description

Structlog-Sentry-Logger

CI codecov License PyPI - Python Version PyPI pre-commit Code style: black powered by semgrep


Documentation: https://structlog-sentry-logger.readthedocs.io

Source Code: https://github.com/TeoZosa/structlog-sentry-logger


:teacher: Overview

A multi-purpose, pre-configured, performance-optimized structlog logger with (optional) Sentry integration via structlog-sentry.

:sparkles: Features

  1. Makes logging as easy as using print statements, but prettier and easier to capture and filter!
  2. Highly opinionated! There are only two (2) distinct configurations.
  3. Structured logs in JSON format means they are ready to be ingested by many of your favorite log analysis tools!
  4. Cloud Logging compatible!

:confetti_ball: What You Get

:muscle: Powerful Automatic Context Fields

The pre-configured options include:

  1. :watch: Timestamps
    • DATETIME_FORMAT = "%Y-%m-%d %H:%M:%S"
  2. :vertical_traffic_light: Log levels
    • Added to the JSON context for filtering and categorization
  3. :mag: Logger names
    • Automatically assigned to namespaced versions of the initializing python modules (.py files), relative to your project directory.
      • e.g., the logger in docs_src/sentry_integration.py is named docs_src.sentry_integration

With fields sorted by key for easier at-a-glance analysis.

:zap: Performance

structlog-sentry-logger is C-compiled and fully-tuned, leveraging orjson as the JSON serializer for lightning-fast logging (more than a 4x speedup over Python's built-in JSON library[1]; see here for sample performance benchmarks). Don't let your obligate cross-cutting concerns cripple performance any longer!

For further reference, see:

[1] source: Choosing a faster JSON library for Python: Benchmarking

:robot: Built-in Sentry Integration (Optional)

Automatically add much richer context to your Sentry reports.

  • Your entire logging context is sent as a Sentry event when the structlog-sentry-logger log level is error or higher.
    • i.e., logger.error(""), logger.exception("")
  • See structlog-sentry for more details.

Table of Contents

:tada: Installation

pip install structlog-sentry-logger

:rocket: Usage

:loud_sound: Pure structlog Logging (Without Sentry)

At the top of your Python module, import and instantiate the logger:

import structlog_sentry_logger

LOGGER = structlog_sentry_logger.get_logger()

Now anytime you want to print anything, don't. Instead do this:

LOG_MSG = "Information that's useful for future me and others"
LOGGER.info(LOG_MSG, extra_field="extra_value")

:memo: Note
All the regular Python logging levels are supported.

Which automatically produces this:

{
    "event": "Information that's useful for future me and others",
    "extra_field": "extra_value",
    "level": "info",
    "logger": "docs_src.pure_structlog_logging_without_sentry",
    "sentry": "skipped",
    "timestamp": "2020-10-18 15:30:05"
}

:goal_net: Sentry Integration

Export your Sentry DSN into your local environment.

  • An easy way to do this is to put it into a local .env file and use python-dotenv to populate your environment:
# On the command line:
SENTRY_DSN=YOUR_SENTRY_DSN
echo "SENTRY_DSN=${SENTRY_DSN}" >> .env

Then load the .env file in your Python code prior to instantiating the logger, e.g.:

from dotenv import find_dotenv, load_dotenv

load_dotenv(find_dotenv())

import structlog_sentry_logger

LOGGER = structlog_sentry_logger.get_logger()

:outbox_tray: Log Custom Context Directly to Sentry

With structlog, you can even incorporate custom messages in your exception handling which will automatically be reported to Sentry (thanks to the structlog-sentry module):

import uuid

import structlog_sentry_logger

LOGGER = structlog_sentry_logger.get_logger()

curr_user_logger = LOGGER.bind(uuid=uuid.uuid4().hex)  # LOGGER instance with bound UUID
try:
    curr_user_logger.warn("A dummy error for testing purposes is about to be thrown!")
    x = 1 / 0
except ZeroDivisionError as err:
    ERR_MSG = (
        "I threw an error on purpose for this example!\n"
        "Now throwing another that explicitly chains from that one!"
    )
    curr_user_logger.exception(ERR_MSG)
    raise RuntimeError(ERR_MSG) from err
{
    "event": "A dummy error for testing purposes is about to be thrown!",
    "level": "warning",
    "logger": "docs_src.sentry_integration",
    "sentry": "skipped",
    "timestamp": "2020-10-18 15:29:55",
    "uuid": "181e0e00b9034732af4fed2b8424fb11"
}
{
    "event": "I threw an error on purpose for this example!\nNow throwing another that explicitly chains from that one!",
    "exception": 'Traceback (most recent call last):\n  File "/app/structlog-sentry-logger/docs_src/sentry_integration.py", line 10, in <module>\n    x = 1 / 0\nZeroDivisionError: division by zero',
    "level": "error",
    "logger": "docs_src.sentry_integration",
    "sentry": "sent",
    "sentry_id": null,
    "timestamp": "2020-10-18 15:29:55",
    "uuid": "181e0e00b9034732af4fed2b8424fb11"
}
Traceback (most recent call last):
  File "/app/structlog-sentry-logger/docs_src/sentry_integration.py", line 10, in <module>
    x = 1 / 0
ZeroDivisionError: division by zero

The above exception was the direct cause of the following exception:

Traceback (most recent call last):
  File "/app/structlog-sentry-logger/docs_src/sentry_integration.py", line 17, in <module>
    raise RuntimeError(ERR_MSG) from err
RuntimeError: I threw an error on purpose for this example!
Now throwing another that explicitly chains from that one!

:cloud: Cloud Logging Compatibility

The logger will attempt to infer if an application is running in a cloud environment by inspecting for the presence of environment variables that may be automatically injected by cloud providers (namely, KUBERNETES_SERVICE_HOST, GCP_PROJECT, and GOOGLE_CLOUD_PROJECT).

If any of these environment variables are detected, log levels will be duplicated to a reserved severity key in the emitted logs to enable parsing of the log level and the remaining log context (as jsonPayload) by Cloud Logging (see: Cloud Logging: Structured logging).

:memo: ️Note
This behavior can also be manually enabled by adding the STRUCTLOG_SENTRY_LOGGER_CLOUD_LOGGING_COMPATIBILITY_MODE_ON variable to your environment.

:warning:️ Warning
If a user manually specifies a value for the severity key, it will be overwritten! Avoid using this key if possible to preempt any future issues.

:chart_with_downwards_trend: Output: Formatting & Storage

The default behavior is to stream JSON logs directly to the standard output stream like a proper 12 Factor App.

For local development, it often helps to prettify logging to stdout and save JSON logs to a .logs folder at the root of your project directory for later debugging. To enable this behavior, set the following environment variable (assuming you are populating environment variables via a .env file and python-dotenv, as in the Sentry Integration section):

echo "STRUCTLOG_SENTRY_LOGGER_LOCAL_DEVELOPMENT_LOGGING_MODE_ON=" >> .env

In doing so, with our previous exception handling example we would get:

Output_Formatting_example

:wrench: Development

For convenience, implementation details of the below processes are abstracted away and encapsulated in single Make targets.

:fire: Tip
Invoking make without any arguments will display auto-generated documentation on available commands.

:building_construction: Package and Dependencies Installation

Make sure you have Python 3.7+ and poetry installed and configured.

To install the package and all dev dependencies, run:

make provision-environment

:fire: Tip
Invoking the above without poetry installed will emit a helpful error message letting you know how you can install poetry.

:package: Python Module to C-Extension Compilation

The projects's build.py file specifies which modules to package.

For manual per-module compilation, see: Mypyc Documentation: Getting started - Compiling and running

:white_check_mark: Testing

We use tox and pytest for our test automation and testing frameworks, respectively.

To invoke the tests, run:

make test

Run mutation tests to validate test suite robustness (Optional):

make test-mutations

:information_source: Technical Details
Test time scales with the complexity of the codebase. Results are cached in .mutmut-cache, so once you get past the initial cold start problem, subsequent mutation test runs will be much faster; new mutations will only be applied to modified code paths.

:rotating_light: Code Quality

We use pre-commit for our static analysis automation and management framework.

To invoke the analyses and auto-formatting over all version-controlled files, run:

make lint

:rotating_light: Danger
CI will fail if either testing or code quality fail, so it is recommended to automatically run the above locally prior to every commit that is pushed.

:arrows_counterclockwise: Automate via Git Pre-Commit Hooks

To automatically run code quality validation on every commit (over to-be-committed files), run:

make install-pre-commit-hooks

:warning:️ Warning
This will prevent commits if any single pre-commit hook fails (unless it is allowed to fail) or a file is modified by an auto-formatting job; in the latter case, you may simply repeat the commit and it should pass.

:memo: Documentation

make docs-clean docs-html

:fire: Tip
For faster feedback loops, this will attempt to automatically open the newly built documentation static HTML in your browser.

:clipboard: Summary

That's it. Now no excuses. Get out there and program with pride knowing no one will laugh at you in production! For not logging properly, that is. You're on your own for that other observability stuff.

:books: Further Reading

:one: structlog: Structured Logging for Python

:two: Sentry: Monitor and fix crashes in realtime

:three: structlog-sentry: Provides the structlog integration for Sentry


:judge: Legal

:page_facing_up: License

Structlog-Sentry-Logger is licensed under the Apache License, Version 2.0. See LICENSE for the full license text.

:busts_in_silhouette: Credits

This project was generated from @TeoZosa's cookiecutter-cruft-poetry-tox-pre-commit-ci-cd template.

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

structlog-sentry-logger-0.12.1.tar.gz (27.4 kB view details)

Uploaded Source

Built Distributions

structlog_sentry_logger-0.12.1-cp39-cp39-win_amd64.whl (83.3 kB view details)

Uploaded CPython 3.9 Windows x86-64

structlog_sentry_logger-0.12.1-cp39-cp39-manylinux_2_31_x86_64.whl (322.4 kB view details)

Uploaded CPython 3.9 manylinux: glibc 2.31+ x86-64

structlog_sentry_logger-0.12.1-cp39-cp39-macosx_10_16_x86_64.whl (86.8 kB view details)

Uploaded CPython 3.9 macOS 10.16+ x86-64

structlog_sentry_logger-0.12.1-cp38-cp38-win_amd64.whl (82.4 kB view details)

Uploaded CPython 3.8 Windows x86-64

structlog_sentry_logger-0.12.1-cp38-cp38-manylinux_2_31_x86_64.whl (315.9 kB view details)

Uploaded CPython 3.8 manylinux: glibc 2.31+ x86-64

structlog_sentry_logger-0.12.1-cp38-cp38-macosx_10_16_x86_64.whl (85.7 kB view details)

Uploaded CPython 3.8 macOS 10.16+ x86-64

structlog_sentry_logger-0.12.1-cp37-cp37m-win_amd64.whl (82.2 kB view details)

Uploaded CPython 3.7m Windows x86-64

structlog_sentry_logger-0.12.1-cp37-cp37m-manylinux_2_31_x86_64.whl (293.2 kB view details)

Uploaded CPython 3.7m manylinux: glibc 2.31+ x86-64

structlog_sentry_logger-0.12.1-cp37-cp37m-macosx_10_16_x86_64.whl (84.8 kB view details)

Uploaded CPython 3.7m macOS 10.16+ x86-64

File details

Details for the file structlog-sentry-logger-0.12.1.tar.gz.

File metadata

  • Download URL: structlog-sentry-logger-0.12.1.tar.gz
  • Upload date:
  • Size: 27.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.9.7

File hashes

Hashes for structlog-sentry-logger-0.12.1.tar.gz
Algorithm Hash digest
SHA256 ccba9fd36d7e3dbd21e87aec3e3ac8f3d128f93a4ec5dc4f49a20a0ba57880e0
MD5 9333cfa6ac4e20e04eb545cf93c38ceb
BLAKE2b-256 79dd95101117c52c25ceddd2853b84d55087b48b6be4175a489ca90b2bb4ca99

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp39-cp39-win_amd64.whl.

File metadata

  • Download URL: structlog_sentry_logger-0.12.1-cp39-cp39-win_amd64.whl
  • Upload date:
  • Size: 83.3 kB
  • Tags: CPython 3.9, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.9.7

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp39-cp39-win_amd64.whl
Algorithm Hash digest
SHA256 c85a83f435e3d9b604f94807b33c6c857be7f7635835e48a1ed741ccc4c00628
MD5 3d16647740f88706afea5c482d070002
BLAKE2b-256 0c594bc094d644c1550924b3979c31ce8b10c2bf93a7673bc29729b4fb42c8a7

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp39-cp39-manylinux_2_31_x86_64.whl.

File metadata

  • Download URL: structlog_sentry_logger-0.12.1-cp39-cp39-manylinux_2_31_x86_64.whl
  • Upload date:
  • Size: 322.4 kB
  • Tags: CPython 3.9, manylinux: glibc 2.31+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.9.7

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp39-cp39-manylinux_2_31_x86_64.whl
Algorithm Hash digest
SHA256 9f79aa21aae2b9cd5d4f04cbb139b9b9dd3b8a1fa1f7c6beddcbb120c3e16e0e
MD5 4c864ac5f0104a3ad29cb2e728677747
BLAKE2b-256 2ac30b046cfb3113de934d8842a4e9cbea7aca55cf589704cef53b22b41499db

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp39-cp39-macosx_10_16_x86_64.whl.

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp39-cp39-macosx_10_16_x86_64.whl
Algorithm Hash digest
SHA256 f614384817910d7c67c4cb8d2292ef266b29bf193bc733a9cf53b848dde7244c
MD5 643cc7b13c9cce091bcc18998f31eb6d
BLAKE2b-256 9df08678dc5dd374df7ec6ae815157332a77e606c21b698af5718a074fdea0cc

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp38-cp38-win_amd64.whl.

File metadata

  • Download URL: structlog_sentry_logger-0.12.1-cp38-cp38-win_amd64.whl
  • Upload date:
  • Size: 82.4 kB
  • Tags: CPython 3.8, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.9.7

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp38-cp38-win_amd64.whl
Algorithm Hash digest
SHA256 fd2539441cc7415c7791db795bf0555499e147e31571b6844802bca88a662847
MD5 f58d4840cbf7734be951af59656cffc7
BLAKE2b-256 544fbd3f46e2170fcdcbd786e9516028fca5f83cdad35673d0bf7e177ced6acb

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp38-cp38-manylinux_2_31_x86_64.whl.

File metadata

  • Download URL: structlog_sentry_logger-0.12.1-cp38-cp38-manylinux_2_31_x86_64.whl
  • Upload date:
  • Size: 315.9 kB
  • Tags: CPython 3.8, manylinux: glibc 2.31+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.9.7

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp38-cp38-manylinux_2_31_x86_64.whl
Algorithm Hash digest
SHA256 480c14fffe1963f0b9f3949b3fc8c9f1811b6dc1d7a5a6552f4fcd6618c8cea3
MD5 e0e7bb209b7c1e1d599c3afe14c46dfe
BLAKE2b-256 39a0c6c6e3bbca87bbaca4e490725203e7445a00509bc4e1bf5a98a08b543e38

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp38-cp38-macosx_10_16_x86_64.whl.

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp38-cp38-macosx_10_16_x86_64.whl
Algorithm Hash digest
SHA256 6b1602516a0bf20ae342829cfc866607f8105d384109f63c6f039a760bc215ad
MD5 52c8e780205cc88ef7be6620aad56c9c
BLAKE2b-256 c2bdac722166bd5a5e1cfca43438ad854bab7b8feccee653af3f1d4667684db7

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp37-cp37m-win_amd64.whl.

File metadata

  • Download URL: structlog_sentry_logger-0.12.1-cp37-cp37m-win_amd64.whl
  • Upload date:
  • Size: 82.2 kB
  • Tags: CPython 3.7m, Windows x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.9.7

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp37-cp37m-win_amd64.whl
Algorithm Hash digest
SHA256 1888da9ff73f5e83ff990e8f69c2682a8d1d4169ba1f7f52c6eba4dfe6c825e3
MD5 5c4f8e824328cdf930ee2ce62c25a9d2
BLAKE2b-256 b21282456ad9ce8c76e371d4437aab4f654740a4806875b744adaef3b9c468f3

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp37-cp37m-manylinux_2_31_x86_64.whl.

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp37-cp37m-manylinux_2_31_x86_64.whl
Algorithm Hash digest
SHA256 0511166f9a69f9517d6f32382b000297c4cd2ea998efaa8ec7dea0b54010d7ae
MD5 69ddd2d6296ea3b7bce838385b69b40b
BLAKE2b-256 794fd046d4d09dad78dc1f102f989e8e23ac25c8ae7a6e9de9ab203e6e6f33ab

See more details on using hashes here.

File details

Details for the file structlog_sentry_logger-0.12.1-cp37-cp37m-macosx_10_16_x86_64.whl.

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.1-cp37-cp37m-macosx_10_16_x86_64.whl
Algorithm Hash digest
SHA256 33df61c2ec287fd24b3fbed269f2d2401433ccca91478556c129956dd843ab3e
MD5 684067ad1b5d53af3296e156b6c568a4
BLAKE2b-256 d92d2ea81019be361ba41bd3885db12425df5e23385aecec64574bb32c198822

See more details on using hashes here.

Supported by

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