问题:
是这样的,我想实现请求日志记录的功能,能够记录每个请求连接的Request请求信息和Response输出信息。
现在OnResultExecuted方法中,能读取到Request的基本信息,但是Response输出的信息不知道怎么获取,使用下面的代码每次请求都进不了if里面
if (context.HttpContext.Response.Body.CanRead) { var responseReader = new StreamReader(context.HttpContext.Response.Body); log.Response = responseReader.ReadToEnd(); }
我想要,不管这次请求是输出视图还是字符串还是json,能不能都转成对应内容的字符串
回答1:
既然是aspnet core那么Request和Response信息弄一个中间件拦截处理就好了。
大致思想如下:
public class HttpContextMiddleware
{
private readonly RequestDelegate _next;
private readonly ILogger _logger;
/// <summary>
/// 计时器
/// </summary>
private Stopwatch _stopwatch;
/// <summary>
/// 构造 Http 请求中间件
/// <param name="next"></param>
/// <param name="loggerFactory"></param>
/// </summary>
public HttpContextMiddleware(RequestDelegate next,
ILoggerFactory loggerFactory)
{
_next = next;
_logger = loggerFactory.CreateLogger<HttpContextMiddleware>();
}
public async Task InvokeAsync(HttpContext context)
{
if (context.Request.Method == "GET")
{
await _next(context);
return;
}
context.Request.EnableBuffering();
_stopwatch = new Stopwatch();
_stopwatch.Start();
//默认值
var request = context.Request.Body;
var response = context.Response.Body;
try
{
using (var newRequest = new MemoryStream())
{
//替换request流
context.Request.Body = newRequest;
using (var newResponse = new MemoryStream())
{
//替换response流
context.Response.Body = newResponse;
using (var writer = new StreamWriter(newRequest))
{
newRequest.Position = 0;
context.Request.Body = newRequest;
await _next(context);
}
}
}
}
catch (Exception ex)
{
_logger.LogError($" http中间件发生错误: " + ex.ToString());
//定义返回类型
context.Response.ContentType = "application/json";
//错误异常的统一返回
var exOutput = new OutputBaseDto
{
ErrorMsg = ex.Message,
GroupID = log.GroupId,
RequestID = log.RequestId,
ResultCode = Application.Main.Dtos.ResultCode.InternetError
};
using (var sw = new StreamWriter(response))
{
await sw.WriteAsync(exOutput.Obj2Json());
await sw.FlushAsync();
}
//数据库记录堆栈
exOutput.ErrorMsg += ex.StackTrace;
log.Response = exOutput.Obj2Json();
}
finally
{
context.Request.Body = request;
context.Response.Body = response;
}
// 响应完成时存入缓存
context.Response.OnCompleted(() =>
{
_stopwatch.Stop();
return Task.CompletedTask;
});
}
}
回答2:
一般记录请求log 是通过中间件Middlerware实现,如楼上。
你这里是在 ActionFilterAttribute 来实现的话,需要通过ActionExecutedContext.Result 来拿到Action的返回值。而非Response。因为这时候请求流程管道实际还未结束。
具体获取可以参考:
https://stackoverflow.com/questions/49013678/get-values-from-resultexecutedcontext-result