平安校园
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

DatabaseLoggingWriter.cs 8.0 KiB

8 个月前
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. using Masuit.Tools;
  2. using Microsoft.Extensions.Logging;
  3. using NewLife.Serialization;
  4. using SqlSugar;
  5. using System.Globalization;
  6. using UAParser;
  7. namespace SafeCampus.Web.Core;
  8. /// <summary>
  9. /// 数据库写入器
  10. /// </summary>
  11. public class DatabaseLoggingWriter : IDatabaseLoggingWriter
  12. {
  13. private readonly ISearcher _searcher;
  14. private readonly SqlSugarScope _db;
  15. private readonly ILogger<DatabaseLoggingWriter> _logger;
  16. public DatabaseLoggingWriter(ILogger<DatabaseLoggingWriter> logger, ISearcher searcher)
  17. {
  18. _db = DbContext.DB;
  19. _logger = logger;
  20. _searcher = searcher;
  21. }
  22. public async Task WriteAsync(LogMessage logMsg, bool flush)
  23. {
  24. //获取请求json字符串
  25. var jsonString = logMsg.Context.Get("loggingMonitor").ToString();
  26. //转成实体
  27. var loggingMonitor = jsonString.ToJsonEntity<LoggingMonitorJson>();
  28. //日志时间赋值
  29. loggingMonitor.LogDateTime = logMsg.LogDateTime;
  30. // loggingMonitor.ReturnInformation.Value
  31. //验证失败和没有DisplayTitle之类的不记录日志
  32. if (loggingMonitor.Validation == null && loggingMonitor.DisplayTitle != null)
  33. {
  34. //获取操作名称
  35. var operation = loggingMonitor.DisplayTitle;
  36. var client = (ClientInfo)logMsg.Context.Get(LoggingConst.CLIENT);//获取客户端信息
  37. var path = logMsg.Context.Get(LoggingConst.PATH).ToString();//获取操作名称
  38. var method = logMsg.Context.Get(LoggingConst.METHOD).ToString();//获取方法
  39. //表示访问日志
  40. if (operation == EventSubscriberConst.LOGIN_B || operation == EventSubscriberConst.LOGIN_OUT_B)
  41. {
  42. //如果没有异常信息
  43. if (loggingMonitor.Exception == null)
  44. {
  45. await CreateVisitLog(operation, loggingMonitor, client);//添加到访问日志
  46. }
  47. else
  48. {
  49. //添加到异常日志
  50. await CreateOperationLog(operation, path, loggingMonitor, client);
  51. }
  52. }
  53. else
  54. {
  55. //只有定义了Title的POST方法才记录日志
  56. if (!operation.Contains("/") && method == "POST")
  57. {
  58. //添加到操作日志
  59. await CreateOperationLog(operation, path, loggingMonitor, client);
  60. }
  61. }
  62. }
  63. }
  64. /// <summary>
  65. /// 创建访问日志
  66. /// </summary>
  67. /// <param name="operation">访问类型</param>
  68. /// <param name="loggingMonitor">loggingMonitor</param>
  69. /// <param name="clientInfo">客户端信息</param>
  70. private async Task CreateVisitLog(string operation, LoggingMonitorJson loggingMonitor, ClientInfo clientInfo)
  71. {
  72. string name;//用户姓名
  73. string opAccount;//用户账号
  74. if (operation == EventSubscriberConst.LOGIN_B)
  75. {
  76. //如果是登录,用户信息就从返回值里拿
  77. var result = loggingMonitor.ReturnInformation.Value.ToJson();//返回值转json
  78. var userInfo = result.ToJsonEntity<SafeCampusResult<SysUser>>();//格式化成user表
  79. name = userInfo.Data.Name;//赋值姓名
  80. opAccount = userInfo.Data.Account;//赋值账号
  81. }
  82. else
  83. {
  84. //如果是登录出,用户信息就从AuthorizationClaims里拿
  85. name = loggingMonitor.AuthorizationClaims.Where(it => it.Type == ClaimConst.NAME).Select(it => it.Value).FirstOrDefault();
  86. opAccount = loggingMonitor.AuthorizationClaims.Where(it => it.Type == ClaimConst.ACCOUNT).Select(it => it.Value).FirstOrDefault();
  87. }
  88. //日志表实体
  89. var devLogVisit = new SysLogVisit
  90. {
  91. Name = operation,
  92. Category = operation == EventSubscriberConst.LOGIN_B ? CateGoryConst.LOG_LOGIN : CateGoryConst.LOG_LOGOUT,
  93. ExeStatus = SysLogConst.SUCCESS,
  94. OpAddress = GetLoginAddress(loggingMonitor.RemoteIPv4),
  95. OpIp = loggingMonitor.RemoteIPv4,
  96. OpBrowser = clientInfo.UA.Family + clientInfo.UA.Major,
  97. OpOs = clientInfo.OS.Family + clientInfo.OS.Major,
  98. OpTime = loggingMonitor.LogDateTime,
  99. OpUser = name,
  100. OpAccount = opAccount
  101. };
  102. await _db.CopyNew().InsertableWithAttr(devLogVisit).IgnoreColumns(true).SplitTable().ExecuteCommandAsync();//入库
  103. }
  104. /// <summary>
  105. /// 创建操作日志
  106. /// </summary>
  107. /// <param name="operation">操作名称</param>
  108. /// <param name="path">请求地址</param>
  109. /// <param name="loggingMonitor">loggingMonitor</param>
  110. /// <param name="clientInfo">客户端信息</param>
  111. /// <returns></returns>
  112. private async Task CreateOperationLog(string operation, string path, LoggingMonitorJson loggingMonitor,
  113. ClientInfo clientInfo)
  114. {
  115. //用户名称
  116. var name = loggingMonitor.AuthorizationClaims?.Where(it => it.Type == ClaimConst.NAME).Select(it => it.Value).FirstOrDefault();
  117. //账号
  118. var opAccount = loggingMonitor.AuthorizationClaims?.Where(it => it.Type == ClaimConst.ACCOUNT).Select(it => it.Value).FirstOrDefault();
  119. //获取参数json字符串,
  120. var paramJson = loggingMonitor.Parameters == null || loggingMonitor.Parameters.Count == 0
  121. ? null
  122. : loggingMonitor.Parameters[0].Value.ToJsonString();
  123. //获取结果json字符串
  124. var resultJson = string.Empty;
  125. if (loggingMonitor.ReturnInformation != null)//如果有返回值
  126. {
  127. if (loggingMonitor.ReturnInformation.Value != null)//如果返回值不为空
  128. {
  129. var time = loggingMonitor.ReturnInformation.Value.Time != null
  130. ? DateTime.Parse(loggingMonitor.ReturnInformation.Value.Time)
  131. : DateTime.Now;//转成时间
  132. loggingMonitor.ReturnInformation.Value.Time = time.ToString(CultureInfo.CurrentCulture);//转成字符串
  133. resultJson = loggingMonitor.ReturnInformation.Value.ToJsonString();
  134. }
  135. }
  136. //操作日志表实体
  137. var devLogOperate = new SysLogOperate
  138. {
  139. Name = operation,
  140. Category = CateGoryConst.LOG_OPERATE,
  141. ExeStatus = SysLogConst.SUCCESS,
  142. OpAddress = GetLoginAddress(loggingMonitor.RemoteIPv4),
  143. OpIp = loggingMonitor.RemoteIPv4,
  144. OpBrowser = clientInfo.UA.Family + clientInfo.UA.Major,
  145. OpOs = clientInfo.OS.Family + clientInfo.OS.Major,
  146. OpTime = loggingMonitor.LogDateTime,
  147. OpUser = name,
  148. OpAccount = opAccount,
  149. ReqMethod = loggingMonitor.HttpMethod,
  150. ReqUrl = path,
  151. ResultJson = resultJson,
  152. ClassName = loggingMonitor.DisplayName,
  153. MethodName = loggingMonitor.ActionName,
  154. ParamJson = paramJson
  155. };
  156. //如果异常不为空
  157. if (loggingMonitor.Exception != null)
  158. {
  159. devLogOperate.Category = CateGoryConst.LOG_EXCEPTION;//操作类型为异常
  160. devLogOperate.ExeStatus = SysLogConst.FAIL;//操作状态为失败
  161. devLogOperate.ExeMessage = loggingMonitor.Exception.Type + ":" + loggingMonitor.Exception.Message + "\n"
  162. + loggingMonitor.Exception.StackTrace;
  163. }
  164. await _db.CopyNew().InsertableWithAttr(devLogOperate).IgnoreColumns(true).SplitTable().ExecuteCommandAsync();//入库
  165. }
  166. /// <summary>
  167. /// 解析IP地址
  168. /// </summary>
  169. /// <param name="ip"></param>
  170. /// <returns></returns>
  171. private string GetLoginAddress(string ip)
  172. {
  173. var loginAddress = "未知";
  174. try
  175. {
  176. var ipInfo = _searcher.Search(ip);//解析ip
  177. loginAddress = ipInfo?.Replace("0|", "");//去掉前面的0|
  178. }
  179. catch (global::System.Exception ex)
  180. {
  181. _logger.LogError("IP解析错误" + ex, ex);
  182. }
  183. return loginAddress;
  184. }
  185. }