Log without the setup via a pre-configured structlog logger with optional Sentry integration
Project description
Structlog-Sentry-Logger
A multi-purpose, pre-configured, performance-optimized structlog
logger
with (optional) Sentry integration
via structlog-sentry
.
Why
- Makes logging as easy as using print statements, but prettier and less smelly!
- Highly opinionated! There are only two (2) distinct configurations.
- Structured logs in JSON format means they are ready to be ingested by many of your favorite log analysis tools!
What You Get
Powerful Automatic Context Fields
The pre-configured options include:
- Timestamps
DATETIME_FORMAT = "%Y-%m-%d %H:%M:%S"
- Log levels
- Added to the JSON context for filtering and categorization
- 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 nameddocs_src.sentry_integration
- e.g., the logger in
- Automatically assigned to namespaced versions of the initializing
python modules (
With fields sorted by key for easier at-a-glance analysis.
Performance
structlog-sentry-logger
is fully-tuned and leverages ORJSON
as the JSON serializer for lightning-fast logging (more than a 4x speedup over
Python's built-in JSON library[1]). It's 2020, you don't have to let your
obligate cross-cutting concerns cripple performance any longer!
For further reference, see:
- "
ORJSON
: Serialize" for benchmarks - "
structlog
: Performance" for salient performance-related configurations.
[1] source: Choosing a faster JSON library for Python: Benchmarking
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 iserror
or higher.- i.e.,
logger.error("")
,logger.exception("")
- i.e.,
- See
structlog-sentry
for more details.
Table of Contents
Installation
pip install structlog-sentry-logger
Usage
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")
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"
}
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 usepython-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()
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!
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
(e.g. via python-dotenv
as in
the Sentry Integration section):
CI_ENVIRONMENT_SLUG=dev-local
In doing so, with our previous exception handling example we would get:
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.
Further Reading
structlog
: Structured Logging for Python
Sentry
: Monitor and fix crashes in realtime.
structlog-sentry
: Provides the structlog
SentryProcessor
for Sentry integration.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file structlog-sentry-logger-0.5.2.tar.gz
.
File metadata
- Download URL: structlog-sentry-logger-0.5.2.tar.gz
- Upload date:
- Size: 15.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.0.10 CPython/3.8.5 Darwin/19.6.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 1b34abb879fc94ea43c23e2af367697b32fdf1e66bc38b09427cdbbfa595c54c |
|
MD5 | e62f8d7de5b652f8047b63cd1766155c |
|
BLAKE2b-256 | 6e3e8d4d085380d53cbd0b28fa7fec92c3ef17d1b1a60fe43751746f98e10ea7 |
File details
Details for the file structlog_sentry_logger-0.5.2-py3-none-any.whl
.
File metadata
- Download URL: structlog_sentry_logger-0.5.2-py3-none-any.whl
- Upload date:
- Size: 13.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.0.10 CPython/3.8.5 Darwin/19.6.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 54341834bb4b1006bcf04c3ff50ad0e3ba780a2d1f94c5dfc4fbddd2f1cbdab1 |
|
MD5 | 9009a40047e09c44501bce8e59387e74 |
|
BLAKE2b-256 | a113fec7e00f0effe4f2a6f58b904d52c018cc7de7bfec9e605988b9eb9ddcb8 |