以下基于.NET Core 2.1

定义GrayLog日志记录中间件:

中间件代码:

public class GrayLogMiddleware
{
private readonly RequestDelegate _next;
private readonly ILogger _logger;

//在应用程序的生命周期中,中间件的构造函数只会被调用一次
public GrayLogMiddleware(RequestDelegate next, ILoggerFactory loggerFactory)
{
_next = next;
_logger = loggerFactory.CreateLogger("GrayLog");
}

public async Task InvokeAsync(HttpContext context)
{
var additionalFields = new Dictionary<string, object>()
{
["LogId"] = Guid.NewGuid()
};

// 若将该中间件做为第一个请求管道中的第一个中间件进行注册
// 那么在此处就可以进行全局异常处理
try
{
var startTime = DateTime.Now;
await _next(context);
var endTime = DateTime.Now;
additionalFields["Elapsed"] = (endTime - startTime).Milliseconds;
_logger.LogInfo(context, additionalFields);
}
catch (Exception ex)
{
if (context.Response.HasStarted == false)
{
await WriteExceptionInfoIntoResponseAsync(context, ex);
}
_logger.LogException(context, ex, additionalFields);
#if DEBUG
throw;
#endif
}
}

private async Task WriteExceptionInfoIntoResponseAsync(HttpContext context, Exception ex)
{
try
{
var resp = new ApiResponse();
resp = resp.Exception(ex);
var respStr = JsonConvert.SerializeObject(resp);
await context.Response.WriteAsync(respStr, Encoding.UTF8);
}
catch
{
// ignore
}
}

}

日志记录代码:

public static class LogExtension
{
public static void LogInfo(this ILogger logger, HttpContext context, IDictionary<string, object> addtionalFields = null)
{
logger.LogCore(context, LogLevel.Information, addtionalFields: addtionalFields);
}

public static void LogException(this ILogger logger, HttpContext context, Exception ex, IDictionary<string, object> addtionalFields = null)
{
logger.LogCore(context, LogLevel.Error, ex, addtionalFields);
}

private static void LogCore(this ILogger logger, HttpContext context, LogLevel logLevel, Exception ex = null, IDictionary<string, object> addtionalFields = null)
{
try
{
var shortMessage = GetShortMessage(context);
if (addtionalFields == null)
{
addtionalFields = GetAddtionalFields(context);
}
else
{
var temp = GetAddtionalFields(context);
addtionalFields = addtionalFields.Union(temp).ToDictionary(d => d.Key, d => d.Value);
}

// 需要使用Scope才能将additionalFields记录到GrayLog中
using (logger.BeginScope(addtionalFields))
{
logger.Log(logLevel, exception: ex, message: shortMessage);
}
}
catch
{
#if DEBUG
throw;
#endif
// ignore
}
}

/// <summary>
/// 获取请求的短消息
/// <para>
/// 消息格式:HttpMethod RequestUrl HttpStatusCode
/// </para>
/// </summary>
/// <example> GET http://localhost:5000 200</example>
private static string GetShortMessage(HttpContext context)
{
var request = context.Request;

var method = request.Method;
var url = request.GetEncodedUrl();
var statusCode = context.Response.StatusCode;

return $"{method} {url} {statusCode}";
}

/// <summary>
/// 需要写入到日志中的额外字段:请求来源,请求参数
/// </summary>
private static IDictionary<string, object> GetAddtionalFields(HttpContext context)
{
var referer = context.Connection.RemoteIpAddress;
var requestData = GetRequestParameters(context);

return new Dictionary<string, object>()
{
["Referer"] = referer,
["RequestData"] = requestData
};
}

private static string GetRequestParameters(HttpContext context)
{
if (context.Request.ContentLength > )
{
var stream = context.Request.Body;
if (stream.CanRead == false)
{
return null;
}
if (stream.CanSeek == false)
{
// 将HttpRequestStream转换为FileBufferingReadStream
context.Request.EnableBuffering();
stream = context.Request.Body;
}
stream.Position = ;

using (var reader = new StreamReader(stream))
{
var data = reader.ReadToEnd();
return data;
}
}

return null;
}

}
 

Graylog日志配置:

  
 public class Program
{
public static void Main(string[] args) => CreateWebHost().Run();

private static IWebHost CreateWebHost() => CreateWebHostBuilder().Build();

// 这里未使用.NET Core封装的CreateDefaultBuilder方法,因为它封装了过多不需要的东西
private static IWebHostBuilder CreateWebHostBuilder() =>
new WebHostBuilder()
.UseContentRoot(Directory.GetCurrentDirectory())
#if RELEASE
.UseIISIntegration()
#endif
.UseKestrel()
.ConfigureLogging((context, builder) =>
{
ConfigLogger(context, builder);
})
.UseStartup<Startup>();

private static void ConfigLogger(WebHostBuilderContext context, ILoggingBuilder builder)
{
// 使用日志过滤器(log filtering),禁止Kestrel记录访问日志
builder.ClearProviders();
builder.AddFilter("Microsoft", LogLevel.None);
builder.AddFilter("System", LogLevel.Error);

if (context.HostingEnvironment.IsDevelopment())
{
builder.AddDebug();
}

// GrayLog配置(这里使用App.config作为配置文件
builder.AddGelf(option =>
{
option.Host = ConfigurationManager.AppSettings["grayLogHost"];
option.Port = Convert.ToInt32(ConfigurationManager.AppSettings["grayLogPort"]);
option.LogSource = ConfigurationManager.AppSettings["grayLogSource"];
option.Protocol = GelfProtocol.Udp;
});
}
}

注册中间件到请求处理管道:

public static class GrayLogMiddlewareExtension
{
/// <summary>
/// 向请求管道中添加GrayLog记录功能及全局异常处理
/// </summary>
public static IApplicationBuilder UseGrayLog(this IApplicationBuilder builder) =>
builder.UseMiddleware<GrayLogMiddleware>();
}

public class Startup
{
public void Configure(IApplicationBuilder app)
{
app.UseGrayLog()
.UseMvc();
}
}

以上日志记录了如下几个方面:

  1. 日志信息Id

  2. 请求来源

  3. 请求基础信息

    采用类似HTTP请求行格式,即:HttpMethod RequestUrl ResponseStatusCode,如:GET http://localhost 200

  4. 入参

  5. 接口耗时

  6. 若发生异常,则记录异常信息

HttpRequestStream vs FileBufferingReadStream

GET请求参数都体现在Url中了,这里讲述如何获取POST请求的参数。

通常POST请求数据都在请求体中,ASP.NET Core中HttpRequest类型的Body属性是HttpRequestStream类型,该类型源码在Github上可以看到,但在Google和微软关方文档中都没搜索到。反编译Microsoft.AspNetCore.Server.Kestrel.Core.dll只找到了同样继承自ReadOnlyStreamFrameRequestStream

HttpRequestStream类的CanSeek属性返回值为false,不支持多次读取,所以需要先转换为FileBufferingReadStream。转换过程可参考:BufferingHelperHttpRequestRewindExtensions。实现代码如下:

public static class HttpRequestRewindExtensions
{
public static void EnableBuffering(this HttpRequest request, int bufferThreshold, long bufferLimit)
{
BufferingHelper.EnableRewind(request, bufferThreshold, bufferLimit);
}
}


public static class BufferingHelper
{
public static HttpRequest EnableRewind(this HttpRequest request, int bufferThreshold = DefaultBufferThreshold, long? bufferLimit = null)
{
if (request == null)
{
throw new ArgumentNullException(nameof(request));
}

var body = request.Body;
if (!body.CanSeek)
{
var fileStream = new FileBufferingReadStream(body, bufferThreshold, bufferLimit, _getTempDirectory);
request.Body = fileStream;
request.HttpContext.Response.RegisterForDispose(fileStream);
}
return request;
}
}

推荐阅读

Logging in ASP.NET Core

ASP.NET Core Middleware

Stream Class

ASP.NET Core中使用Graylog记录日志的更多相关文章

  1. ASP.NET Core 中使用 GrayLog 记录日志

    使用 UDP 协议发送日志 自定义好的查询 key 存储数据,尽量不要使用 graylog2-server 服务端格式化日志再存储 Ubuntu 安装服务端 sudo apt-get update & ...

  2. ASP.NET Core:ASP.NET Core中使用NLog记录日志

    一.前言 在所有的应用程序中,日志功能是不可或缺的模块,我们可以根据日志信息进行调试.查看产生的错误信息,在ASP.NET Core中我们可以使用log4net或者NLog日志组件来实现记录日志的功能 ...

  3. Asp.Net Core中使用NLog记录日志

    2019/10/28, Asp.Net Core 3.0, NLog 4.6.7, NLog.Web.AspNetCore 4.9.0 摘要:NLog在asp.net网站中的使用,NLog日志写入数据 ...

  4. Asp.Net Core中利用Seq组件展示结构化日志功能

    在一次.Net Core小项目的开发中,掌握的不够深入,对日志记录并没有好好利用,以至于一出现异常问题,都得跑动服务器上查看,那时一度怀疑自己肯定没学好,不然这一块日志不可能需要自己扒服务器日志来查看 ...

  5. 在Asp.Net Core中集成Kafka(中)

    在上一篇中我们主要介绍如何在Asp.Net Core中同步Kafka消息,通过上一篇的操作我们发现上面一篇中介绍的只能够进行简单的首发kafka消息并不能够消息重发.重复消费.乐观锁冲突等问题,这些问 ...

  6. 如何在 ASP.Net Core 中使用 Serilog

    记录日志的一个作用就是方便对应用程序进行跟踪和排错调查,在实际应用上都是引入 日志框架,但如果你的 日志文件 包含非结构化的数据,那么查询起来将是一个噩梦,所以需要在记录日志的时候采用结构化方式. 将 ...

  7. ASP.NET Core 中的那些认证中间件及一些重要知识点

    前言 在读这篇文章之间,建议先看一下我的 ASP.NET Core 之 Identity 入门系列(一,二,三)奠定一下基础. 有关于 Authentication 的知识太广,所以本篇介绍几个在 A ...

  8. Asp.net Core中使用Session

    前言 2017年就这么悄无声息的开始了,2017年对我来说又是特别重要的一年. 元旦放假在家写了个Asp.net Core验证码登录, 做demo的过程中遇到两个小问题,第一是在Asp.net Cor ...

  9. 在ASP.NET Core中使用百度在线编辑器UEditor

    在ASP.NET Core中使用百度在线编辑器UEditor 0x00 起因 最近需要一个在线编辑器,之前听人说过百度的UEditor不错,去官网下了一个.不过服务端只有ASP.NET版的,如果是为了 ...

随机推荐

  1. Java 8 特性

    1.简介 毫无疑问,Java 8是自Java  5(2004年)发布以来Java语言最大的一次版本升级,Java 8带来了很多的新特性,比如编译器.类库.开发工具和JVM(Java虚拟机).在这篇教程 ...

  2. 02 of learning python

    01 input输入的是str类型 如果输入的是数字的话,要记得强制转换一下! 02 isdigit() 这个方法是用来检测字符串是否全部由数字组成 str.isdigit() 如果字符串只包含数字则 ...

  3. linux redis 多实例安装

    前言: Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作数据库.缓存和消息中间件. 它支持多种类型的数据结构,如 字符串(strings), 散列(hashes), 列表( ...

  4. Asp.net Security框架(2)

    Asp.net 的Security框架除了提供Cookies,OAuth,ActiveDirectory等多个用户认证实现,基本上已经满足业务项目的开发需要了. 当需要实现OAuth2.0服务器端实现 ...

  5. day_4流程控制之分支结构循环结构及for循环

    复习一下昨天的内容 1:变量的命名规范 只能由数字 字母 及下划线组成 不能以数字开头 不能与系统关键字重名 _开头有特殊含义 __开头__结尾的变量是魔法变量 支持大小驼峰 ,但建议使用下划线连接语 ...

  6. [转]kaldi 神经网络

    转自:http://blog.csdn.net/xmdxcsj/article/details/54695506 overview type author CPU/GPU feature nnet1 ...

  7. 816. Ambiguous Coordinates

    We had some 2-dimensional coordinates, like "(1, 3)" or "(2, 0.5)".  Then, we re ...

  8. 如何配置React Native真机调试-iOS

    说在前面,本教程是建立在项目已经成功在模拟器上运行的基础上,如果你是还未配置好环境的新手,建议先从官网快速入门开始:官网英文版 . 中文版 ok, 切入正题,当你已经完成好环境配置,在模拟器上成功的运 ...

  9. failed to create pid file /var/run/rsyncd.pid: File exists报错

    [root@pcidata-jenkins ansible_playbooks]# ps aux|grep rsyncroot      1799  0.0  0.0 114652   480 ?   ...

  10. mysql 开发进阶篇系列 16 MySQL Server(myisam key_buffer)

    一.概述 mysql 提供了很多参数来进行服务器的设置,当服务第一次启动的时候,所有启动参数值都是系统默认的.这些参数在很多生产环境下并不能满足实际的应用需求.在这个系列中涉及到了liunx 服务器, ...