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.0.tar.gz (27.5 kB view details)

Uploaded Source

Built Distributions

structlog_sentry_logger-0.12.0-cp39-cp39-win_amd64.whl (84.0 kB view details)

Uploaded CPython 3.9 Windows x86-64

structlog_sentry_logger-0.12.0-cp39-cp39-manylinux_2_31_x86_64.whl (316.9 kB view details)

Uploaded CPython 3.9 manylinux: glibc 2.31+ x86-64

structlog_sentry_logger-0.12.0-cp39-cp39-macosx_10_16_x86_64.whl (87.7 kB view details)

Uploaded CPython 3.9 macOS 10.16+ x86-64

structlog_sentry_logger-0.12.0-cp38-cp38-win_amd64.whl (82.9 kB view details)

Uploaded CPython 3.8 Windows x86-64

structlog_sentry_logger-0.12.0-cp38-cp38-manylinux_2_31_x86_64.whl (315.8 kB view details)

Uploaded CPython 3.8 manylinux: glibc 2.31+ x86-64

structlog_sentry_logger-0.12.0-cp38-cp38-macosx_10_16_x86_64.whl (86.9 kB view details)

Uploaded CPython 3.8 macOS 10.16+ x86-64

structlog_sentry_logger-0.12.0-cp37-cp37m-win_amd64.whl (83.1 kB view details)

Uploaded CPython 3.7m Windows x86-64

structlog_sentry_logger-0.12.0-cp37-cp37m-manylinux_2_31_x86_64.whl (295.7 kB view details)

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

structlog_sentry_logger-0.12.0-cp37-cp37m-macosx_10_16_x86_64.whl (86.1 kB view details)

Uploaded CPython 3.7m macOS 10.16+ x86-64

File details

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

File metadata

  • Download URL: structlog-sentry-logger-0.12.0.tar.gz
  • Upload date:
  • Size: 27.5 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.0.tar.gz
Algorithm Hash digest
SHA256 21a317a939b88d959e897570c2a3d0df08d37cf9f9171c01d89fd937667f85c6
MD5 a7354e774f923823f3e30c72b49771b4
BLAKE2b-256 ca78d46e56287eecfa064de7afaa0549ae84eb5a55ddb5452c66add2075bc9f6

See more details on using hashes here.

File details

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

File metadata

  • Download URL: structlog_sentry_logger-0.12.0-cp39-cp39-win_amd64.whl
  • Upload date:
  • Size: 84.0 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.0-cp39-cp39-win_amd64.whl
Algorithm Hash digest
SHA256 42fc6710fcaf8edc5dd0dfc9b4d11450096b8336ba4b9597397d2389d4439e87
MD5 d1f673b8f903663f4d2f6b181af46586
BLAKE2b-256 c03686e980ae6d907ea7613a29305a960f664f0ba59dbfbb788c0dac95aed15f

See more details on using hashes here.

File details

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

File metadata

  • Download URL: structlog_sentry_logger-0.12.0-cp39-cp39-manylinux_2_31_x86_64.whl
  • Upload date:
  • Size: 316.9 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.0-cp39-cp39-manylinux_2_31_x86_64.whl
Algorithm Hash digest
SHA256 5e656505be3f9fe221ff774835c50a57a285cb1362c3805263c74fa5f30e3e9a
MD5 df700973f338049805e01d9e9902bcfc
BLAKE2b-256 599f14b677736e1644835dfbfd970cf8af911de1c2f03252365c0ccedb858344

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.0-cp39-cp39-macosx_10_16_x86_64.whl
Algorithm Hash digest
SHA256 b2ef9c46e90b8696ff466be652e37a1af1df0110ad24cdd47a10fa5ccf82a9f8
MD5 355b6e204dd9bef744bac91a84cba43e
BLAKE2b-256 0ae862c2cab6d5d610ec0f928adca575c519539a94d676fa22c011171c3c5185

See more details on using hashes here.

File details

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

File metadata

  • Download URL: structlog_sentry_logger-0.12.0-cp38-cp38-win_amd64.whl
  • Upload date:
  • Size: 82.9 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.0-cp38-cp38-win_amd64.whl
Algorithm Hash digest
SHA256 e7368f1481e1085a69d6a9b922df2ca53833f9b8159fa3a71aad0a96029625fc
MD5 378e379d27bea54f02a96831316d7687
BLAKE2b-256 9f47f97f9bb0c415806fe4ce8aea8678172a091ac7be328a0dbf0308645ea510

See more details on using hashes here.

File details

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

File metadata

  • Download URL: structlog_sentry_logger-0.12.0-cp38-cp38-manylinux_2_31_x86_64.whl
  • Upload date:
  • Size: 315.8 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.0-cp38-cp38-manylinux_2_31_x86_64.whl
Algorithm Hash digest
SHA256 5bb4d33a42f686239e650c1cf393d4fdaae4afdf3ba5feab13b3df338647dd76
MD5 e31e7aca1e30c61b61c406bb43586693
BLAKE2b-256 2b9bd323c6c9fa289510c2f89abe42d865c678600ef96e002e9fa829f0c17dbd

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.0-cp38-cp38-macosx_10_16_x86_64.whl
Algorithm Hash digest
SHA256 329e09541996df9b7950d62269338825183034fdc545d478b7050c718d9563ec
MD5 1b16eb27fba16ef908018505c22d1a8f
BLAKE2b-256 1a23e57face2fc062cbf089ec638d2573be2c2ecfdc3afd59b6d86c7fbbb3612

See more details on using hashes here.

File details

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

File metadata

  • Download URL: structlog_sentry_logger-0.12.0-cp37-cp37m-win_amd64.whl
  • Upload date:
  • Size: 83.1 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.0-cp37-cp37m-win_amd64.whl
Algorithm Hash digest
SHA256 e6558288108445f56e83b77e72b5628dcc09ab3494bb332cf60591d08ca84d9f
MD5 97bcc1f78be2250fa72c4ebd766cf8ea
BLAKE2b-256 09e376f796b471820c5bb9f2a8ea93646d9b194f40eb929f2a4af9712b11ae80

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.0-cp37-cp37m-manylinux_2_31_x86_64.whl
Algorithm Hash digest
SHA256 b59e87e7972e77e42d220d1fbe682bd1e5fb8cb43a3e7f23a4c7b3d37fd197a7
MD5 16fe3a4a9ad210c4858995d2c9524d2f
BLAKE2b-256 0bd07dda418724a6d8207653e93565ab7799952ca15ab542cd120e8694f53662

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for structlog_sentry_logger-0.12.0-cp37-cp37m-macosx_10_16_x86_64.whl
Algorithm Hash digest
SHA256 ec99f85ed806f460fae40e0d490eda621c95bd82c9d1415c0842c73e4d761b67
MD5 2f84ad6bce171c817b9c1f1a31691d00
BLAKE2b-256 573dc38969cab9d52481c4b70eddc85911ecf8c94b528dab7dc07b9bc77ea546

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