README.md |
---|
If you like or are using this project please give it a star. Thanks!
Install the Serilog.Enrichers.GlobalLogContext package from NuGet:
Install-Package Serilog.Enrichers.GlobalLogContext
Add the Serilog namespaces to your C# file:
using Serilog;
using Serilog.Context;
Include the global log context enricher in your logger configuration:
Log.Logger = new LoggerConfiguration()
.Enrich.FromGlobalLogContext()
// ... other configuration ...
.CreateLogger();
The FromGlobalLogContext()
enricher dynamically adds properties present in the Serilog.Context.GlobalLogContext
, to all produced events in your application.
Then, properties can be added and removed from the global log context using GlobalLogContext.PushProperty()
:
GlobalLogContext.PushProperty("AppVersion", GetThisAppVersion());
GlobalLogContext.PushProperty("OperatingSystem", GetCurrentOS());
After the code above is executed, any log event written to any Serilog sink will carry the properties AppVersion
and OperatingSystem
automatically.
When properties are added to the global log context, a disposable object is returned, which can be used to remove the property from the global stack at a later time:
GlobalLogContext.PushProperty("A", 1);
Log.Information("Carries property A = 1");
using (GlobalLogContext.PushProperty("A", 2))
using (GlobalLogContext.PushProperty("B", 1))
{
Log.Information("Carries A = 2 and B = 1");
}
Log.Information("Carries property A = 1, again");
Pushing a property onto the global log context will override any existing properties with the same name, until the object returned from PushProperty()
is disposed, as the property A
in the example demonstrates.
Important: properties must be popped from the global log context in the precise order in which they were added. Behavior otherwise is undefined.
By design, for performance reasons, adding or removing properties in the global log context are not thread-safe operations. If you have multiple threads modifying the GlobalLogContext
at the same time, you should call Lock
or LockAsync
before making changes to the GlobalLogContext
. e.g.:
// sync
using (GlobalLogContext.Lock())
{
GlobalLogContext.PushProperty("A", 3);
}
// async
using (await GlobalLogContext.LockAsync())
{
GlobalLogContext.PushProperty("A", 3);
}
In the sample folder, there's an example of a .NET app that uses Serilog for logging to the Console
using this enricher, as described above.
When setting up the Serilog logging pipeline, you have the option to include properties in all log events using a native feature of Serilog:
Log.Logger = new LoggerConfiguration()
.Enrich.WithProperty("AppVersion", GetThisAppVersion())
.Enrich.WithProperty("OperatingSystem", GetCurrentOS())
// ... other configuration ...
.CreateLogger();
The equivalent using Serilog.Enrichers.GlobalLogContext
would be:
Log.Logger = new LoggerConfiguration()
.Enrich.FromGlobalLogContext()
// ... other configuration ...
.CreateLogger();
// ... other code ...
GlobalLogContext.PushProperty("AppVersion", GetThisAppVersion());
GlobalLogContext.PushProperty("OperatingSystem", GetCurrentOS());
As you can see, the main difference is that Serilog.Enrichers.GlobalLogContext
allows you to add properties at a later time, long after your Serilog logging pipeline has been configured, which can be very useful in scenarios where the properties being added are not immediately available during the bootstrapping of the app. In addition, properties in the GlobalLogContext
can be replaced or removed at any time during the execution of the app.
Click on the Releases tab on GitHub.
Copyright © 2021-2023 C. Augusto Proiete & Contributors - Provided under the Apache License, Version 2.0.