巧用ActionFilterAttribute实现API日志的记录

 本文转载自微信公众号「UP技术控」,作者conan5566。转载本文请联系UP技术控公众号。  

背景

上回提到开发web api的时候,一般是需要记录api的输入输出信息,方便后续排查问题;使用的是委托的形式进行记录日志。

使用Func

这次我们使用另外一种方式,Filter来记录输入输出日志。

实现方式

1、首先在进入action的时候,定义OnActionExecuting。

 
 
 
  1. public override void OnActionExecuting(ActionExecutingContext context) 
  2.         { 
  3.             base.OnActionExecuting(context); 
  4.  
  5.             // 后续添加了获取请求的请求体,如果在实际项目中不需要删除即可 
  6.             long contentLen = context.HttpContext.Request.ContentLength == null ? 0 : context.HttpContext.Request.ContentLength.Value; 
  7.             if (contentLen > 0) 
  8.             { 
  9.                 // 读取请求体中所有内容 
  10.                 System.IO.Stream stream = context.HttpContext.Request.Body; 
  11.                 if (context.HttpContext.Request.Method == "POST") 
  12.                 { 
  13.                     stream.Position = 0; 
  14.                 } 
  15.                 byte[] buffer = new byte[contentLen]; 
  16.                 stream.Read(buffer, 0, buffer.Length); 
  17.                 // 转化为字符串 
  18.                 RequestBody = System.Text.Encoding.UTF8.GetString(buffer); 
  19.             } 
  20.  
  21.             ActionArguments = Newtonsoft.Json.JsonConvert.SerializeObject(context.ActionArguments); 
  22.  
  23.             Stopwatch = new Stopwatch(); 
  24.             Stopwatch.Start(); 
  25.         } 

2、定义Stopwatch ,计算方法的耗时。

 
 
 
  1. private string ActionArguments { get; set; } 
  2.  
  3.         ///  
  4.         /// 请求体中的所有值 
  5.         ///  
  6.         private string RequestBody { get; set; } 
  7.  
  8.         private Stopwatch Stopwatch { get; set; } 

3、结束的时候,把信息打印出来OnActionExecuted。

 
 
 
  1. public override void OnActionExecuted(ActionExecutedContext context) 
  2.         { 
  3.             base.OnActionExecuted(context); 
  4.             Stopwatch.Stop(); 
  5.  
  6.             string url = context.HttpContext.Request.Host + context.HttpContext.Request.Path + context.HttpContext.Request.QueryString; 
  7.             string method = context.HttpContext.Request.Method; 
  8.             string controller = context.Controller.ToString(); 
  9.             string action = context.ActionDescriptor.DisplayName; 
  10.             string token = ""; 
  11.             if (context.HttpContext.Request != null && context.HttpContext.Request.Headers != null && context.HttpContext.Request.Headers["Authorization"].Count > 0) 
  12.             { 
  13.                 token = context.HttpContext.Request.Headers["Authorization"]; 
  14.             } 
  15.             string qs = ActionArguments; 
  16.             dynamic result = context?.Result?.GetType()?.Name == "EmptyResult" ? new { Value = "无返回结果" } : context?.Result as dynamic; 
  17.  
  18.             string res = "在返回结果前发生了异常"; 
  19.             try 
  20.             { 
  21.                 if (result != null) 
  22.                 { 
  23.                     res = Newtonsoft.Json.JsonConvert.SerializeObject(result.Value); 
  24.                 } 
  25.             } 
  26.             catch (System.Exception) 
  27.             { 
  28.                 res = "日志未获取到结果,返回的数据无法序列化"; 
  29.             } 
  30.  
  31.             NLogger.Info( 
  32.                 $"地址:{url} \n " + 
  33.                   $"controller:{controller} \n " + 
  34.                     $"action:{action} \n " + 
  35.                       $"token:{token} \n " + 
  36.                 $"方式:{method} \n " + 
  37.                 $"请求体:{RequestBody} \n " + 
  38.                 $"参数:{qs}\n " + 
  39.                 $"结果:{res}\n " + 
  40.                 $"耗时:{Stopwatch.Elapsed.TotalMilliseconds} 毫秒(指控制器内对应方法执行完毕的时间)"); 
  41.         } 

4、控制器调用LogAttribute。

 
 
 
  1. ///  
  2.     /// 
  3.     ///  
  4.     [Produces("application/json")] 
  5.     [LogAttribute] 
  6.     [CustomExceptionFilterAttribute] 
  7.     public class DefaultController : Controller 
  8.     { 
  9.     } 

 

完整代码



 
 
 
  1. using CompanyName.ProjectName.Core; 
  2. using Microsoft.AspNetCore.Mvc.Filters; 
  3. using System.Diagnostics; 
  4.  
  5. namespace CompanyName.ProjectName.HttpApi.Host.Code 
  6.     ///  
  7.     /// 拦截器 
  8.     ///  
  9.     public class LogAttribute : ActionFilterAttribute 
  10.     { 
  11.         private string ActionArguments { get; set; } 
  12.  
  13.         ///  
  14.         /// 请求体中的所有值 
  15.         ///  
  16.         private string RequestBody { get; set; } 
  17.  
  18.         private Stopwatch Stopwatch { get; set; } 
  19.  
  20.         ///  
  21.         /// 
  22.         ///  
  23.         ///  
  24.         public override void OnActionExecuting(ActionExecutingContext context) 
  25.         { 
  26.             base.OnActionExecuting(context); 
  27.  
  28.             // 后续添加了获取请求的请求体,如果在实际项目中不需要删除即可 
  29.             long contentLen = context.HttpContext.Request.ContentLength == null ? 0 : context.HttpContext.Request.ContentLength.Value; 
  30.             if (contentLen > 0) 
  31.             { 
  32.                 // 读取请求体中所有内容 
  33.                 System.IO.Stream stream = context.HttpContext.Request.Body; 
  34.                 if (context.HttpContext.Request.Method == "POST") 
  35.                 { 
  36.                     stream.Position = 0; 
  37.                 } 
  38.                 byte[] buffer = new byte[contentLen]; 
  39.                 stream.Read(buffer, 0, buffer.Length); 
  40.                 // 转化为字符串 
  41.                 RequestBody = System.Text.Encoding.UTF8.GetString(buffer); 
  42.             } 
  43.  
  44.             ActionArguments = Newtonsoft.Json.JsonConvert.SerializeObject(context.ActionArguments); 
  45.  
  46.             Stopwatch = new Stopwatch(); 
  47.             Stopwatch.Start(); 
  48.         } 
  49.  
  50.         ///  
  51.         /// 
  52.         ///  
  53.         ///  
  54.         public override void OnActionExecuted(ActionExecutedContext context) 
  55.         { 
  56.             base.OnActionExecuted(context); 
  57.             Stopwatch.Stop(); 
  58.  
  59.             string url = context.HttpContext.Request.Host + context.HttpContext.Request.Path + context.HttpContext.Request.QueryString; 
  60.             string method = context.HttpContext.Request.Method; 
  61.             string controller = context.Controller.ToString(); 
  62.             string action = context.ActionDescriptor.DisplayName; 
  63.             string token = ""; 
  64.             if (context.HttpContext.Request != null && context.HttpContext.Request.Headers != null && context.HttpContext.Request.Headers["Authorization"].Count > 0) 
  65.             { 
  66.                 token = context.HttpContext.Request.Headers["Authorization"]; 
  67.             } 
  68.             string qs = ActionArguments; 
  69.             dynamic result = context?.Result?.GetType()?.Name == "EmptyResult" ? new { Value = "无返回结果" } : context?.Result as dynamic; 
  70.  
  71.             string res = "在返回结果前发生了异常"; 
  72.             try 
  73.             { 
  74.                 if (result != null) 
  75.                 { 
  76.                     res = Newtonsoft.Json.JsonConvert.SerializeObject(result.Value); 
  77.                 } 
  78.             } 
  79.             catch (System.Exception) 
  80.             { 
  81.                 res = "日志未获取到结果,返回的数据无法序列化"; 
  82.             } 
  83.  
  84.             NLogger.Info( 
  85.                 $"地址:{url} \n " + 
  86.                   $"controller:{controller} \n " + 
  87.                     $"action:{action} \n " + 
  88.                       $"token:{token} \n " + 
  89.                 $"方式:{method} \n " + 
  90.                 $"请求体:{RequestBody} \n " + 
  91.                 $"参数:{qs}\n " + 
  92.                 $"结果:{res}\n " + 
  93.                 $"耗时:{Stopwatch.Elapsed.TotalMilliseconds} 毫秒(指控制器内对应方法执行完毕的时间)"); 
  94.         } 
  95.     } 

分享名称:巧用ActionFilterAttribute实现API日志的记录
文章来源:http://www.mswzjz.cn/qtweb/news6/476856.html

攀枝花网站建设、攀枝花网站运维推广公司-贝锐智能,是专注品牌与效果的网络营销公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 贝锐智能