Opinionated structured logging library for Python with a fluent interface
Project description
Fluentlog
Opinionated structured logging for Python with a fluent API.
- API inspired by zerolog
- JSON output format
- OpenTelemetry naming conventions when relevant
- Near zero-cost for disabled log levels
Installation
pip install fluentlog
Getting Started
Simple example
import fluentlog
log = fluentlog.Logger().bind().int("request_id", 1).logger()
log.info().str("user", "jmcs").int("uid", 42).msg("user logged in")
# {"level":"INFO","request_id":1,"user":"jmcs","uid":42,"message":"user logged in"}
# Disabled levels have near-zero overhead
log.debug().func(expensive_func).msg("debug info") # expensive_func is never called
Log Levels
fluentlog supports the following log levels, from more to less critical:
FATAL: Errors the application can't recover fromERROR: Errors that make the current context fail, but not the entire applicationWARNING: Recoverable errorsINFO: Expected lifecycle events and relevant business signalsDEBUG: Internal details useful while diagnosing behavior during developmentTRACE: Very fine-grained execution details, usually only useful for deep debugging
You can set the log level for your logger either in the constructor or using a fluent method:
import fluentlog
log = fluentlog.Logger(level=fluentlog.Level.DEBUG)
# or
log = fluentlog.Logger().set_level(fluentlog.Level.DEBUG)
Field Types
Immutable types (bool, bytes, float, int, path, date/datetime, timedelta, str)
log.info().\
bool("is_valid", True).\
int("user_id", 42).\
str("username", "jmcs").\
path("file_path", "/path/to/file").\
time("timestamp", datetime.now()).\
timedelta("duration", timedelta(seconds=30)).msg("User info")
Adds a field of the corresponding type to the context. Since these types are immutable, they are referenced directly without copying, which is more performant.
Dict and List
log.info().dict("user", {"id": 42, "name": "jmcs"}).list("roles", ["admin", "user"]).msg("Logging a dictionary and a list")
Adds a dictionary and a list field to the context. The dictionary and list are deep-copied to prevent mutations after the fact from affecting the log output, so it has a negative impact on performance.
Exception
try:
1 / 0
except ZeroDivisionError as e:
log.error().exception(e).msg("An error occurred")
Stores the exception details in the event fields:
exception.type: the type of the exception (e.g. ValueError)exception.message: the message of the exceptionexception.stacktrace: the stack trace of the exception
Any
obj = SomeComplexObject()
log.info().any("object", obj).msg("Logging a complex object")
Adds a field with any value to the context. If the value is not JSON serializable, it will be converted to a string using repr() when the event is finalized.
Since this method accepts any value, it will be deep-copied to prevent mutations after the fact from affecting the log output, which negatively impacts performance.
Hooks
Hooks are functions that are called with the event context before the event is finalized, allowing for custom processing and enrichment of the event.
Arbitrary/Custom hook
def add_user_info(event: fluentlog.Event) -> None:
user = get_current_user() # example, potentially expensive operation
event.str("user", user.name).int("user_id", user.id)
log.info().func(add_user_info).msg("Logging with a custom hook")
Runs the function if the log level is enabled. The function receives the event as an argument and can add fields to it.
Caller info
log.info().caller().msg("Logging with caller info")
Identifies the caller of the log method and adds it to the log fields, with the following fields:
code.file.path: The full path of the file containing the caller.code.function.name: The name of the function containing the caller.code.line.number: The line number of the caller in the source code.
The optional skip parameter can be used to skip additional stack frames
if the caller is wrapped in helper functions.
Timestamp
log.info().timestamp().msg("Logging with a timestamp")
Adds a timestamp field to the event with the current time in ISO 8601 format. For loggers, this is processed at output time. For events, this is processed when the timestamp() method is called.
Logging context
import fluentlog
def some_func():
log = fluentlog.context()
log.info().msg("From func")
def main():
log = fluentlog.context().bind().str("context", "example").logger()
some_func()
# {"level":"INFO", "message": "From func"}
with fluentlog.context_logger(log):
some_func()
# {"level":"INFO", "context": "example", "message": "From func"}
main()
Performance
Benchmarks show ~2-3x faster than stdlib logging with formatted output, with greater advantages when log levels are filtered.
Design decisions
Why use different methods for different types?
Using different methods for different types allows for optimising serialization strategies for
mutable and immutable types. For example, dict() and list() deep-copy their arguments to prevent
mutations after the event is logged from affecting the output, while int() and str() can safely
reference immutable values directly without copying.
Why dummy events for disabled log levels?
Having dummy events achieves near-zero overhead, as we can avoid unnecessary processing without having to check the log level everywhere.
Why OpenTelemetry naming conventions?
I use OpenTelemetry for distributed tracing, and like consistent and precise naming, even when it comes at the cost of verbosity.
Why no formatted messages?
Formatted messages are familiar because that's how traditional logging usually works. But for structured logs they are a trap, as important data gets buried in strings instead of proper fields, which makes filtering and querying harder.
Why context-based logger passing?
Preserving logging context across boundaries is essential in complex applications, but having bound context inside a function is useful too. Context-based logger passing allows for both options and keeps things purposeful while avoiding cluttering application APIs.
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
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file fluentlog-0.1.2.tar.gz.
File metadata
- Download URL: fluentlog-0.1.2.tar.gz
- Upload date:
- Size: 17.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"25.10","id":"questing","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6c644d447901f25787a66d87008cce0ce7600118a5c975388e38593084371110
|
|
| MD5 |
3193db3584679bd4467b47b5f5c47962
|
|
| BLAKE2b-256 |
85389fc6bb662eab1c6fb8f29bf9b221328bbbaa2be6c3af8e76045b4ea7af32
|
File details
Details for the file fluentlog-0.1.2-py3-none-any.whl.
File metadata
- Download URL: fluentlog-0.1.2-py3-none-any.whl
- Upload date:
- Size: 15.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"25.10","id":"questing","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e5a7d1fc188a2aae103a1eac6df489c6db8838089663d62465b3200a07b0fd4c
|
|
| MD5 |
994cd2ca9c4af2a0fcdd1978ce0b7066
|
|
| BLAKE2b-256 |
a08ee218a1bc4c26fcc8844a4d73f949f27cf274919c254feb15dd6c93d6d70c
|