微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

在 asp.net core 中使用 serilog 维护单独的范围上下文

如何解决在 asp.net core 中使用 serilog 维护单独的范围上下文

我遇到了在 ILogger<T> 实例之间共享范围上下文的问题。

下面是我如何在我的 asp.net core 3.1 服务中配置 Serilog 并注入 ILogger<T>我有一个 WithClassContext 扩展,我在每个类的构造函数调用它,以便将类名作为上下文属性推送。我发现我的日志中出现了错误ClassName 值。

当我在调试器中检查注入的 ILogger 对象时,我发现以下内容

ILogger<T> -> Logger<T>
{
  _logger -> Serilog.Extensions.Logger.SerilogLogger
   {
      _logger -> Serilog.Core.Logger
      _provider -> Serilog.Extensions.Logger.SerilogLoggerProvider
      {
         CurrentScope -> Serilog.Extensions.Logger.SerilogLoggerScope
         {
           Parent -> Serilog.Extensions.Logger.SerilogLoggerScope
           _state -> Dictionary<string,object>
         }
      }
   }
}

所以我观察到的是,每个注入的 ILogger<T> 都具有由所有记录器共享的相同 _provider 对象。 _provider.CurrentScope 似乎是一个 SerilogLoggerScope 对象的链表,其中 _provider.CurrentScope 指向列表中的最后一个节点,_provider.CurrentScope.Parent 是前一个节点。还有一个 CurrentScope._state Dictionary<string,object> 包含属性名称和值。写出范围上下文时,如果存在任何冲突的属性名称,则使用列表中的最后一个 SerilogLoggerScope

所以使用我下面的例子:

  1. FooService 已创建并推送 ClassName

    • CurrentScope._state -> {"ClassName","FooService"}
  2. FooController 已创建并推送 ClassName

    • CurrentScope._state -> {"ClassName","FooController"}
    • CurrentScope.Parent._state -> {"ClassName","FooService"}
  3. FooService:CurrentScope 现在与 FooController 相同。

  4. 所有类的日志记录现在推送 "ClassName": "FooController"

我本以为通过注入 ILogger<T>,范围上下文不会被其他实例共享。我还研究了其他人如何将类名推送到日志记录上下文中,我相信我也在做同样的事情。

Program.cs:

public static IHostBuilder CreateHostBuilder(string[] args) =>
    Host.CreateDefaultBuilder(args)
        .UseSerilog(ConfigureLogger)
        .ConfigureWebHostDefaults(webBuilder =>
        {
            webBuilder.UseStartup<Startup>();
            webBuilder.UseUrls("http://*:6050");
        });


private static void ConfigureLogger(HostBuilderContext ctx,IServiceProvider sp,LoggerConfiguration config)
{
    var shouldFormatElastic = !ctx.HostingEnvironment.EnvironmentName.Equals("local",StringComparison.OrdinalIgnoreCase);
    config.ReadFrom.Configuration(ctx.Configuration)
          .Enrich.FromLogContext()
          .Enrich.WithExceptionDetails();

    if (shouldFormatElastic)
    {
        var logFormatter = new ExceptionAsObjectJsonFormatter(renderMessage: true);
        config.Writeto.Async(a =>
                                 a.Console(logFormatter,standardErrorFromLevel: LogEventLevel.Error));
    }
    else
    {
        config.Writeto.Async(a =>
                                 a.Console(
                                     outputTemplate: "[{Timestamp:HH:mm:ss} {Level:u3}] {Message:lj}{NewLine}{Properties:j}{Exception}{NewLine}",theme: SystemConsoleTheme.Literate));
    }
}

控制器:

[ApiController]
[Route("api/[controller]")]
public class FooController : ControllerBase
{
    private ILogger _logger;
    private readonly IFooService _fooService;

    /// <inheritdoc />
    public FooController(ILogger<FooController> logger,IFooService fooService)
    {
        _logger = logger.WithClassContext();
        _fooService = fooService;
    }
}

服务:

public class FooService : IFooService
{
    private readonly ILogger _logger;

    public FooService(ILogger<IFooService> logger)
    {
        _logger = logger.WithClassContext();
    }

LoggingExtensions.cs

public static class FooLoggingExtensions
{
    public static ILogger Here(this ILogger logger,[CallerMemberName] string memberName = null)
    {
        var state = new Dictionary<string,object>
        {
            {"MemberName",memberName }
        };

        // Don't need to dispose since this scope will last through the call stack.
        logger.BeginScope(state);
        return logger;
    }

    public static ILogger WithClassContext<T>(this ILogger<T> logger)
    {
        var state = new Dictionary<string,object>
        {
            {"ClassName",typeof(T).Name }
        };

        // Don't need to dispose since this scope will last through the call stack.
        logger.BeginScope(state);
        return logger;
    }
}

解决方法

根据您尝试执行的操作,BeginScope 似乎不是解决问题的正确工具。您要解决的问题是每个日志都需要将 ClassName 作为日志消息的一部分。为此,您可以修改您使用的 outputTemplate 以包含 {SourceContext},然后在您的构造函数内部,而不是调用 logger.WithClassContext(),而是调用 logger.ForContext<ClassName>()

请注意,我只修改了以下示例中的本地环境日志记录。

private static void ConfigureLogger(HostBuilderContext ctx,IServiceProvider sp,LoggerConfiguration config)
{
    var shouldFormatElastic = !ctx.HostingEnvironment.EnvironmentName.Equals("local",StringComparison.OrdinalIgnoreCase);
    config.ReadFrom.Configuration(ctx.Configuration)
          .Enrich.FromLogContext()
          .Enrich.WithExceptionDetails();

    if (shouldFormatElastic)
    {
        var logFormatter = new ExceptionAsObjectJsonFormatter(renderMessage: true);
        config.WriteTo.Async(a =>
                                 a.Console(logFormatter,standardErrorFromLevel: LogEventLevel.Error));
    }
    else
    {
        config.WriteTo.Async(a =>
                                 a.Console(
                                     outputTemplate: "[{Timestamp:HH:mm:ss} {Level:u3}] {SourceContext} {Message:lj}{NewLine}{Properties:j}{Exception}{NewLine}",theme: SystemConsoleTheme.Literate));
    }
}
[ApiController]
[Route("api/[controller]")]
public class FooController : ControllerBase
{
    private ILogger _logger;
    private readonly IFooService _fooService;

    /// <inheritdoc />
    public FooController(ILogger<FooController> logger,IFooService fooService)
    {
        _logger = logger.ForContext<FooController>();
        _fooService = fooService;
    }
}

我使用以下网站/博客作为参考来制定此答案。
https://benfoster.io/blog/serilog-best-practices/#source-context
C# ASP.NET Core Serilog add class name and method to log
https://github.com/serilog/serilog/issues/968

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?