巧用ActionFilterAttribute实现API日志的记录

开发 前端
使用Func 委托实现API日志的记录,这次我们使用另外一种方式,Filter来记录输入输出日志。

[[381754]]

 本文转载自微信公众号「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.         /// <summary> 
  4.         /// 请求体中的所有值 
  5.         /// </summary> 
  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. /// <summary> 
  2.     /// 
  3.     /// </summary> 
  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.     /// <summary> 
  7.     /// 拦截器 
  8.     /// </summary> 
  9.     public class LogAttribute : ActionFilterAttribute 
  10.     { 
  11.         private string ActionArguments { get; set; } 
  12.  
  13.         /// <summary> 
  14.         /// 请求体中的所有值 
  15.         /// </summary> 
  16.         private string RequestBody { get; set; } 
  17.  
  18.         private Stopwatch Stopwatch { get; set; } 
  19.  
  20.         /// <summary> 
  21.         /// 
  22.         /// </summary> 
  23.         /// <param name="context"></param> 
  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.         /// <summary> 
  51.         /// 
  52.         /// </summary> 
  53.         /// <param name="context"></param> 
  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.     } 

 

责任编辑:武晓燕 来源: UP技术控
相关推荐

2021-02-03 05:24:44

API日志

2021-02-01 00:04:13

Dictionary数据批量

2023-03-10 08:59:30

2021-11-17 09:00:00

Kubernetes集群容器

2010-09-25 16:17:25

SQL语句

2022-02-08 17:07:54

Spring BooSpring Aop日志记录

2011-09-23 10:53:09

2011-08-04 13:31:50

数据库记录更改日志触发器

2023-05-26 07:08:05

CSS模糊实现文字

2021-11-03 17:10:37

CSS sticky前端代码

2011-11-21 15:44:46

管理服务器日志

2010-11-18 13:40:48

mysql分页查询

2021-05-10 23:39:31

Python日志记录

2020-12-24 08:37:41

Css前端加载动画

2022-01-28 09:01:49

架构

2021-08-30 06:20:39

CSS 技巧3D 效果

2021-03-01 23:26:41

日志Spring BootAOP

2011-03-02 17:54:45

Pure-ftpd

2023-06-27 09:33:15

Loading 动画CSS

2022-03-31 07:46:17

CSS动画技巧
点赞
收藏

51CTO技术栈公众号