-
Notifications
You must be signed in to change notification settings - Fork 780
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'main' into feat/exception-endpoint
- Loading branch information
Showing
53 changed files
with
1,034 additions
and
309 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,175 @@ | ||
# OpenTelemetry .NET Logs | ||
|
||
## Best Practices | ||
|
||
The following tutorials have demonstrated the best practices for logging with | ||
OpenTelemetry .NET: | ||
|
||
* [Getting Started - Console Application](./getting-started-console/README.md) | ||
* [Getting Started - ASP.NET Core | ||
Application](./getting-started-aspnetcore/README.md) | ||
* [Logging with Complex Objects](./complex-objects/README.md) | ||
|
||
## Structured Logging | ||
|
||
:heavy_check_mark: You should use structured logging. | ||
|
||
* Structured logging is more efficient than unstructured logging. | ||
* Filtering and redaction can happen on invidual key-value pairs instead of | ||
the entire log message. | ||
* Storage and indexing are more efficient. | ||
* Structured logging makes it easier to manage and consume logs. | ||
|
||
:stop_sign: You should avoid string interpolation. | ||
|
||
> [!WARNING] | ||
> The following code has bad performance due to [string | ||
interpolation](https://learn.microsoft.com/dotnet/csharp/tutorials/string-interpolation): | ||
|
||
```csharp | ||
var food = "tomato"; | ||
var price = 2.99; | ||
|
||
logger.LogInformation($"Hello from {food} {price}."); | ||
``` | ||
|
||
Refer to the [logging performance | ||
benchmark](../../test/Benchmarks/Logs/LogBenchmarks.cs) for more details. | ||
|
||
## Package Version | ||
|
||
:heavy_check_mark: You should always use the | ||
[`ILogger`](https://docs.microsoft.com/dotnet/api/microsoft.extensions.logging.ilogger) | ||
interface (including | ||
[`ILogger<TCategoryName>`](https://learn.microsoft.com/dotnet/api/microsoft.extensions.logging.ilogger-1)) | ||
from the latest stable version of | ||
[Microsoft.Extensions.Logging](https://www.nuget.org/packages/Microsoft.Extensions.Logging/) | ||
package, regardless of the .NET runtime version being used: | ||
|
||
* If you're using the latest stable version of [OpenTelemetry .NET | ||
SDK](../../src/OpenTelemetry/README.md), you don't have to worry about the | ||
version of `Microsoft.Extensions.Logging` package because it is already taken | ||
care of for you via [package dependency](../../Directory.Packages.props). | ||
* Starting from version `3.1.0`, the .NET runtime team is holding a high bar for | ||
backward compatibility on `Microsoft.Extensions.Logging` even during major | ||
version bumps, so compatibility is not a concern here. | ||
|
||
## Logging API | ||
|
||
:heavy_check_mark: You should use [compile-time logging source | ||
generation](https://docs.microsoft.com/dotnet/core/extensions/logger-message-generator) | ||
pattern to achieve the best performance. | ||
|
||
```csharp | ||
public static partial class Food | ||
{ | ||
[LoggerMessage(Level = LogLevel.Information, Message = "Hello from {food} {price}.")] | ||
public static partial void SayHello(ILogger logger, string food, double price); | ||
} | ||
|
||
var food = "tomato"; | ||
var price = 2.99; | ||
|
||
Food.SayHello(logger, food, price); | ||
``` | ||
|
||
> [!NOTE] | ||
> There is no need to pass in an explicit | ||
[EventId](https://learn.microsoft.com/dotnet/api/microsoft.extensions.logging.eventid) | ||
while using | ||
[LoggerMessageAttribute](https://learn.microsoft.com/dotnet/api/microsoft.extensions.logging.loggermessageattribute). | ||
A durable `EventId` will be automatically assigned based on the hash of the | ||
method name during code generation. | ||
|
||
:heavy_check_mark: You can use | ||
[LogPropertiesAttribute](https://learn.microsoft.com/dotnet/api/microsoft.extensions.logging.logpropertiesattribute) | ||
from | ||
[Microsoft.Extensions.Telemetry.Abstractions](https://www.nuget.org/packages/Microsoft.Extensions.Telemetry.Abstractions/) | ||
if you need to log complex objects. Check out the [Logging with Complex | ||
Objects](./complex-objects/README.md) tutorial for more details. | ||
|
||
:stop_sign: You should avoid the extension methods from | ||
[LoggerExtensions](https://learn.microsoft.com/dotnet/api/microsoft.extensions.logging.loggerextensions), | ||
these methods are not optimized for performance. | ||
|
||
> [!WARNING] | ||
> The following code has bad performance due to | ||
[boxing](https://learn.microsoft.com/dotnet/csharp/programming-guide/types/boxing-and-unboxing): | ||
|
||
```csharp | ||
var food = "tomato"; | ||
var price = 2.99; | ||
|
||
logger.LogInformation("Hello from {food} {price}.", food, price); | ||
``` | ||
|
||
Refer to the [logging performance | ||
benchmark](../../test/Benchmarks/Logs/LogBenchmarks.cs) for more details. | ||
|
||
## Logger Management | ||
|
||
In order to use | ||
[`ILogger`](https://docs.microsoft.com/dotnet/api/microsoft.extensions.logging.ilogger) | ||
interface (including | ||
[`ILogger<TCategoryName>`](https://learn.microsoft.com/dotnet/api/microsoft.extensions.logging.ilogger-1)), | ||
you need to first get a logger. How to get a logger depends on two things: | ||
|
||
* The type of application you are building. | ||
* The place where you want to log. | ||
|
||
Here is the rule of thumb: | ||
|
||
* If you are building an application with [dependency injection | ||
(DI)](https://learn.microsoft.com/dotnet/core/extensions/dependency-injection) | ||
(e.g. [ASP.NET Core](https://learn.microsoft.com/aspnet/core) and [.NET | ||
Worker](https://learn.microsoft.com/dotnet/core/extensions/workers)), in most | ||
cases you should use the logger provided by DI, there are special cases when | ||
you want log before DI logging pipeline is available or after DI logging | ||
pipeline is disposed. Refer to the [.NET official | ||
document](https://learn.microsoft.com/dotnet/core/extensions/logging#integration-with-hosts-and-dependency-injection) | ||
and [Getting Started with OpenTelemetry .NET Logs in 5 Minutes - ASP.NET Core | ||
Application](./getting-started-aspnetcore/README.md) tutorial to learn more. | ||
* If you are building an application without DI, create a `LoggerFactory` | ||
instance and configure OpenTelemetry to work with it. Refer to the [Getting | ||
Started with OpenTelemetry .NET Logs in 5 Minutes - Console | ||
Application](./getting-started-console/README.md) tutorial to learn more. | ||
|
||
:stop_sign: You should avoid creating `LoggerFactory` instances too frequently, | ||
`LoggerFactory` is fairly expensive and meant to be reused throughout the | ||
application. For most applications, one `LoggerFactory` instance per process | ||
would be sufficient. | ||
|
||
:heavy_check_mark: You should properly manage the lifecycle of | ||
[LoggerFactory](https://learn.microsoft.com/dotnet/api/microsoft.extensions.logging.loggerfactory) | ||
instances if they are created by you. | ||
|
||
* If you forget to dispose the `LoggerFactory` instance before the application | ||
ends, logs might get dropped due to the lack of proper flush. | ||
* If you dispose the `LoggerFactory` instance too early, any subsequent logging | ||
API invocation associated with the logger factory could become no-op (i.e. no | ||
logs will be emitted). | ||
|
||
:heavy_check_mark: You should use the fully qualified class name as the log | ||
category name. Refer to the [.NET official | ||
document](https://learn.microsoft.com/dotnet/core/extensions/logging#log-category) | ||
to learn more. | ||
|
||
## Log Correlation | ||
|
||
In OpenTelemetry, logs are automatically correlated to traces. Check the [Log | ||
Correlation](./correlation/README.md) tutorial to learn more. | ||
|
||
## Log Enrichment | ||
|
||
TBD | ||
|
||
## Log Filtering | ||
|
||
Check the [Customizing OpenTelemetry .NET SDK for | ||
Logs](./customizing-the-sdk/README.md#log-filtering) document to learn more. | ||
|
||
## Log Redaction | ||
|
||
Logs might contain sensitive information such as passwords and credit card | ||
numbers, proper redaction is required to prevent privacy and security incidents. | ||
Check the [Log Redaction](./redaction/README.md) tutorial to learn more. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
// Copyright The OpenTelemetry Authors | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
using Microsoft.Extensions.Logging; | ||
|
||
internal static partial class LoggerExtensions | ||
{ | ||
[LoggerMessage(LogLevel.Information, "Food `{name}` price changed to `{price}`.")] | ||
public static partial void FoodPriceChanged(this ILogger logger, string name, double price); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.