RequestAuditFilter.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. using Microsoft.AspNetCore.Mvc.Controllers;
  2. using Microsoft.AspNetCore.Mvc.Filters;
  3. using System.Security.Claims;
  4. using System;
  5. using System.Threading.Tasks;
  6. using TEAMModelOS.SDK.Extension;
  7. using Microsoft.Extensions.Logging;
  8. using TEAMModelOS.SDK;
  9. using System.IdentityModel.Tokens.Jwt;
  10. using System.Linq;
  11. using Azure.Core;
  12. using TEAMModelOS.SDK.DI;
  13. using Microsoft.Extensions.Primitives;
  14. using System.Net.Http;
  15. using System.ServiceModel.Channels;
  16. using System.Net;
  17. using System.Net.Http.Json;
  18. using Microsoft.Extensions.Options;
  19. using TEAMModelOS.Models;
  20. namespace TEAMModelOS.Filter
  21. {
  22. public class RequestAuditFilter : IAsyncActionFilter
  23. {
  24. //private readonly ILogger _logger;
  25. private readonly IHttpClientFactory _httpClient;
  26. private readonly DingDing _dingding;
  27. private readonly Option _option; private string p = "bi";
  28. public RequestAuditFilter(/*ILoggerFactory loggerFactory*/IHttpClientFactory httpClient, DingDing dingding, IOptionsSnapshot<Option> option)
  29. {
  30. // _logger = loggerFactory.CreateLogger<RequestAuditFilter>();
  31. _httpClient = httpClient;
  32. _dingding=dingding;
  33. _option = option?.Value;
  34. }
  35. public async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
  36. {
  37. try
  38. {
  39. string id = string.Empty, name = string.Empty, picture = string.Empty, school = string.Empty, scope = string.Empty, roles = string.Empty;
  40. //============== 这里是执行方法之前获取数据 ====================
  41. // 获取控制器、路由信息
  42. //var actionDescriptor = context.ActionDescriptor as ControllerActionDescriptor;
  43. // 获取请求的方法
  44. //var method = actionDescriptor.MethodInfo;
  45. // 获取 HttpContext 和 HttpRequest 对象
  46. var httpContext = context.HttpContext;
  47. string ua = httpContext.GetUserAgent();
  48. var httpRequest = httpContext.Request;
  49. // 获取客户端 Ipv4 地址
  50. var remoteIPv4 = httpContext.GetRemoteIpAddressToIPv4();
  51. // 获取请求的 Url 地址
  52. // var requestUrl = httpRequest.GetRequestUrlAddress();
  53. // 获取来源 Url 地址
  54. var refererUrl = httpRequest.GetRefererUrlAddress();
  55. // 获取请求参数(写入日志,需序列化成字符串后存储)
  56. var parameters = context.ActionArguments;
  57. // 获取操作人(必须授权访问才有值)"userId" 为你存储的 claims type,jwt 授权对应的是 payload 中存储的键名
  58. //var userId = httpContext.User?.FindFirstValue("userId");
  59. var authtoken = context.HttpContext.GetXAuth("AuthToken");
  60. string tokenSha = string.Empty, client = string.Empty;
  61. if (context.HttpContext.Request.Headers.TryGetValue("Authorization", out StringValues Authorization) && Authorization.Any())
  62. {
  63. try
  64. {
  65. var jwt = new JwtSecurityTokenHandler().ReadJwtToken(Authorization.ToString().Replace("Bearer ", ""));
  66. client= roles = jwt.Claims.FirstOrDefault(claim => claim.Type.Equals("roles"))?.Value;
  67. tokenSha= ShaHashHelper.GetSHA1(Authorization.ToString());
  68. }
  69. catch (Exception ex)
  70. {
  71. // await _dingding.SendBotMsg($"{ex.Message}\n{ex.StackTrace}\n{Authorization}\n{httpRequest.PathBase}{httpRequest.Path}", GroupNames.成都开发測試群組);
  72. }
  73. }
  74. if (context.HttpContext.Request.Headers.TryGetValue("X-Auth-IdToken", out StringValues XAuthIdToken) && XAuthIdToken.Any())
  75. {
  76. try
  77. {
  78. var jwt = new JwtSecurityTokenHandler().ReadJwtToken(XAuthIdToken.ToString());
  79. id = jwt.Payload.Sub;
  80. name = jwt.Claims.FirstOrDefault(claim => claim.Type.Equals("name"))?.Value;
  81. if (string.IsNullOrEmpty(tokenSha))
  82. {
  83. tokenSha= ShaHashHelper.GetSHA1(XAuthIdToken.ToString());
  84. }
  85. scope="teacher";
  86. }
  87. catch (Exception ex)
  88. {
  89. await _dingding.SendBotMsg($"{ex.Message}\n{ex.StackTrace}\n{XAuthIdToken}\n{httpRequest.PathBase}{httpRequest.Path}", GroupNames.成都开发測試群組);
  90. }
  91. }
  92. if (context.HttpContext.Request.Headers.TryGetValue("X-Auth-School", out StringValues XAuthSchool) && XAuthSchool.Any())
  93. {
  94. try
  95. {
  96. school = XAuthSchool.ToString();
  97. }
  98. catch (Exception ex) { }
  99. }
  100. if (context.HttpContext.Request.Headers.TryGetValue("X-Auth-ApiToken", out StringValues XAutApiToken) && XAutApiToken.Any())
  101. {
  102. try
  103. {
  104. var jwt = new JwtSecurityTokenHandler().ReadJwtToken(XAutApiToken);
  105. id = jwt.Payload.Sub;
  106. if (string.IsNullOrEmpty(tokenSha))
  107. {
  108. tokenSha= jwt.Payload.Jti;
  109. }
  110. client="Open";
  111. }
  112. catch (Exception ex)
  113. {
  114. await _dingding.SendBotMsg($"{ex.Message}\n{ex.StackTrace}\n{XAutApiToken}\n{httpRequest.PathBase}{httpRequest.Path}", GroupNames.成都开发測試群組);
  115. }
  116. }
  117. if (!string.IsNullOrWhiteSpace(authtoken))
  118. {
  119. try
  120. {
  121. var jwt = new JwtSecurityTokenHandler().ReadJwtToken(authtoken);
  122. id = jwt.Payload.Sub;
  123. school = $"{jwt.Payload.Azp}".Equals("true")|| $"{jwt.Payload.Azp}".Equals("false") ? "" : $"{jwt.Payload.Azp}";
  124. name = jwt.Claims.FirstOrDefault(claim => claim.Type.Equals("name"))?.Value;
  125. scope = jwt.Claims.FirstOrDefault(claim => claim.Type.Equals("scope"))?.Value;
  126. if (string.IsNullOrEmpty(tokenSha))
  127. {
  128. tokenSha= ShaHashHelper.GetSHA1(authtoken);
  129. }
  130. }
  131. catch (Exception ex)
  132. {
  133. await _dingding.SendBotMsg($"{ex.Message}\n{ex.StackTrace}\n{authtoken}\n{httpRequest.PathBase}{httpRequest.Path}", GroupNames.成都开发測試群組);
  134. }
  135. }
  136. string secChUaPlatform = string.Empty;
  137. if (httpContext.Request.Headers.TryGetValue("Sec-Ch-Ua-Platform", out var values))
  138. {
  139. secChUaPlatform = values.FirstOrDefault();
  140. }
  141. if (string.IsNullOrEmpty(tokenSha))
  142. {
  143. tokenSha= ShaHashHelper.GetSHA1($"{ua}{remoteIPv4}{httpRequest.Host}{secChUaPlatform}");
  144. }
  145. // 请求时间
  146. var requestedTime = DateTimeOffset.Now.GetGMTTime(8).ToUnixTimeMilliseconds();
  147. //============== 这里是执行方法之后获取数据 ====================
  148. var actionContext = await next();
  149. // 获取返回的结果
  150. // var returnResult = actionContext.Result;
  151. // 判断是否请求成功,没有异常就是请求成功
  152. // var isRequestSucceed = actionContext.Exception == null;
  153. // 获取调用堆栈信息,提供更加简单明了的调用和异常堆栈
  154. // var stackTrace = EnhancedStackTrace.Current();
  155. // string region = await _searcher.SearchIpAsync(remoteIPv4);
  156. //同一个账号,同一IP,同一接口,UA标识(UA标识随意切换则表示可能会存在DDOS),时间段
  157. //_logger.LogInformation(new{ ua=httpContext.GetUserAgent(), ip=remoteIPv4,time=requestedTime,path =$"{httpRequest.PathBase}{httpRequest.Path}",host= $"{httpRequest.Host}", param=parameters,id ,name ,school,succeed =isRequestSucceed }.ToJsonString());
  158. var data = new
  159. {
  160. //ua =ua,
  161. ip = remoteIPv4,
  162. time = requestedTime,
  163. path = $"{httpRequest.PathBase}{httpRequest.Path}",
  164. host = $"{httpRequest.Host}",
  165. param = parameters,
  166. id = id,
  167. name = name,
  168. school = school,
  169. client = client,
  170. tid = tokenSha,
  171. scope = scope,
  172. // referer = refererUrl,
  173. //platform = secChUaPlatform,
  174. p = p,
  175. l = _option.Location.Contains("China", StringComparison.OrdinalIgnoreCase) ? "China" : "Global"
  176. //idToken=XAuthIdToken
  177. };
  178. var httpclient = _httpClient.CreateClient();
  179. httpclient.Timeout= TimeSpan.FromSeconds(10);
  180. #if DEBUG
  181. var response = await httpclient.PostAsJsonAsync("http://cdhabook.teammodel.cn:8806/api/http-log", data);
  182. //if (response.StatusCode==HttpStatusCode.OK)
  183. //{
  184. // string result = await response.Content.ReadAsStringAsync();
  185. //}
  186. #else
  187. _= httpclient.PostAsJsonAsync("http://cdhabook.teammodel.cn:8806/api/http-log",data);
  188. #endif
  189. // _ = _httpTrigger.RequestHttpTrigger(data, "China", "http-log");
  190. }
  191. catch (Exception ex)
  192. {
  193. await _dingding.SendBotMsg($"HTTP日志访问错误:{ex.Message}\n{ex.StackTrace}", GroupNames.成都开发測試群組);
  194. var actionContext = await next();
  195. }
  196. }
  197. }
  198. }