10

最近我已经将 Serilog 配置为与 ASP.NET Core 2 MVC 应用程序配合得很好,下一个任务是在系统的每一层上跟踪对 Web 应用程序的传入请求。所以基本上,我们想要传播一些令牌(比如RequestId由 Serilog 生成到较低层的应用程序)。

"Serilog": {
    "Using": [ "Serilog.Sinks.Console", "Serilog.Sinks.RollingFile" ],
    "MinimumLevel": {
      "Default": "Debug",
      "Override": {
        "Microsoft": "Warning"
      }
    },
    "WriteTo": [
      {
        "Name": "RollingFile",
        "Args": {
          "pathFormat": "log-{Hour}.txt",
          "fileSizeLimitBytes": "",
          "retainedFileCountLimit": "",
          "outputTemplate": "{Timestamp:yyyy-MM-dd HH:mm:ss.fff zzz} [{Application}] [{Level}] [{RequestId}] - {Message}{NewLine}{Exception}"
        }
      }
    ],
    "Enrich": [ "FromLogContext", "WithMachineName", "WithThreadId" ],
    "Properties": {
      "Application": "MultilayerApp"
    }
},

在日志中,我们有很好的输入,比如

2018-01-19 14:06:01.165 +00:00 [App] [Warning] [0HLAV6PIMA9M0:00000001] - Accessing expired session, Key:"6e7f3ab5-db62-335d-1bc7-6824e5b918f5"

但我的问题是 Serilog 在哪里实施丰富器RequestId?老实说,我找不到它。

4

1 回答 1

22

在 ASP.NET Core 中,RequestId一些记录器公开的TraceIdentifierHttpContext. 此属性可在整个应用程序中用于标识当前请求。

出于记录目的,回退到HttpContext不是要走的路。Microsoft.Extensions.Logging抽象支持日志记录范围,这是一种向适用于该范围的记录器提供附加信息的方法。

默认情况下,ASP.NET Core 打开了两个日志记录范围。其中之一是在每个请求开始时HostingLogScope打开的(如果至少启用了关键日志记录)。

记录器可以通过实现在每个请求开始时将对象传递给它的BeginScope方法来访问信息,HostingLogScope并简单地遍历对象直到找到属性:

string requestId = null;
if (state is IEnumerable<KeyValuePair<string, object>> properties)
{
    foreach (var property in properties)
    {
        if (property.Key == "RequestId")
        {
            requestId = property.Value as string;
        }
    }
}

Serilog的作用几乎相同,但将所有属性存储在日志事件中。这就是为什么您没有找到明确的引用,RequestId但是当您指定要包含它的日志记录字符串格式时它仍然存在。

于 2018-02-05T15:46:25.763 回答