Compare commits
	
		
			33 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 1b758aa41a | ||
|   | 43bdc70899 | ||
|   | fadda000a6 | ||
|   | 45a8c91a5a | ||
|   | 8e938f18be | ||
|   | ab1b364c54 | ||
|   | 5ec65b2fb0 | ||
|   | 926eced724 | ||
|   | f7f8802272 | ||
|   | c6910dff02 | ||
|   | ad299d0dbb | ||
|   | 8b124d1050 | ||
|   | ff41080dbd | ||
|   | 0e28606e3d | ||
|   | 6a025ceee5 | ||
|   | 6b2e53d6dc | ||
|   | b989aa5561 | ||
|   | f5b0b7ebd2 | ||
|   | 16881ae076 | ||
|   | af04112656 | ||
|   | a2863112dc | ||
|   | f531e4dfc5 | ||
|   | 8db9b32ba7 | ||
|   | dd5691cbef | ||
|   | de48b32af3 | ||
|   | 600b5042a1 | ||
|   | aac77029da | ||
|   | e50205f557 | ||
|   | e227411d1f | ||
|   | 2de0ed793f | ||
|   | cb0276f273 | ||
|   | 562b3f17c9 | ||
|   | 0f78f81c1c | 
| @@ -63,6 +63,8 @@ public sealed class OperDescAttribute : MoAttribute | |||||||
|     public Type? LocalizerType { get; } |     public Type? LocalizerType { get; } | ||||||
|  |  | ||||||
|     public override void OnException(MethodContext context) |     public override void OnException(MethodContext context) | ||||||
|  |     { | ||||||
|  |         if (App.HttpContext.Request.Path.StartsWithSegments("/_blazor")) | ||||||
|         { |         { | ||||||
|             //插入异常日志 |             //插入异常日志 | ||||||
|             SysOperateLog log = GetOperLog(LocalizerType, context); |             SysOperateLog log = GetOperLog(LocalizerType, context); | ||||||
| @@ -76,13 +78,18 @@ public sealed class OperDescAttribute : MoAttribute | |||||||
|  |  | ||||||
|             OperDescAttribute.WriteToQueue(log); |             OperDescAttribute.WriteToQueue(log); | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public override void OnSuccess(MethodContext context) |     public override void OnSuccess(MethodContext context) | ||||||
|     { |     { | ||||||
|  |         if (App.HttpContext.Request.Path.StartsWithSegments("/_blazor")) | ||||||
|  |         { | ||||||
|  |  | ||||||
|             //插入操作日志 |             //插入操作日志 | ||||||
|             SysOperateLog log = GetOperLog(LocalizerType, context); |             SysOperateLog log = GetOperLog(LocalizerType, context); | ||||||
|             OperDescAttribute.WriteToQueue(log); |             OperDescAttribute.WriteToQueue(log); | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 将日志消息写入数据库中 |     /// 将日志消息写入数据库中 | ||||||
| @@ -115,7 +122,7 @@ public sealed class OperDescAttribute : MoAttribute | |||||||
|     private SysOperateLog GetOperLog(Type? localizerType, MethodContext context) |     private SysOperateLog GetOperLog(Type? localizerType, MethodContext context) | ||||||
|     { |     { | ||||||
|         var methodBase = context.Method; |         var methodBase = context.Method; | ||||||
|         var clientInfo = AppService.ClientInfo; |         var userAgent = AppService.UserAgent; | ||||||
|         string? paramJson = null; |         string? paramJson = null; | ||||||
|         if (IsRecordPar) |         if (IsRecordPar) | ||||||
|         { |         { | ||||||
| @@ -127,10 +134,10 @@ public sealed class OperDescAttribute : MoAttribute | |||||||
|             { |             { | ||||||
|                 parametersDict[parametersInfo[i].Name!] = args[i]; |                 parametersDict[parametersInfo[i].Name!] = args[i]; | ||||||
|             } |             } | ||||||
|             paramJson = parametersDict.ToJsonNetString(); |             paramJson = parametersDict.ToSystemTextJsonString(); | ||||||
|         } |         } | ||||||
|         var result = context.ReturnValue; |         var result = context.ReturnValue; | ||||||
|         var resultJson = IsRecordPar ? result?.ToJsonNetString() : null; |         var resultJson = IsRecordPar ? result?.ToSystemTextJsonString() : null; | ||||||
|         //操作日志表实体 |         //操作日志表实体 | ||||||
|         var log = new SysOperateLog |         var log = new SysOperateLog | ||||||
|         { |         { | ||||||
| @@ -138,8 +145,8 @@ public sealed class OperDescAttribute : MoAttribute | |||||||
|             Category = LogCateGoryEnum.Operate, |             Category = LogCateGoryEnum.Operate, | ||||||
|             ExeStatus = true, |             ExeStatus = true, | ||||||
|             OpIp = AppService?.RemoteIpAddress ?? string.Empty, |             OpIp = AppService?.RemoteIpAddress ?? string.Empty, | ||||||
|             OpBrowser = clientInfo?.UA?.Family + clientInfo?.UA?.Major, |             OpBrowser = userAgent?.Browser, | ||||||
|             OpOs = clientInfo?.OS?.Family + clientInfo?.OS?.Major, |             OpOs = userAgent?.Platform, | ||||||
|             OpTime = DateTime.Now, |             OpTime = DateTime.Now, | ||||||
|             OpAccount = UserManager.UserAccount, |             OpAccount = UserManager.UserAccount, | ||||||
|             ReqUrl = null, |             ReqUrl = null, | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ namespace ThingsGateway.Admin.Application; | |||||||
|  |  | ||||||
| [ApiDescriptionSettings(false)] | [ApiDescriptionSettings(false)] | ||||||
| [Route("api/auth")] | [Route("api/auth")] | ||||||
| [LoggingMonitor] | [RequestAudit] | ||||||
| public class AuthController : ControllerBase | public class AuthController : ControllerBase | ||||||
| { | { | ||||||
|     private readonly IAuthService _authService; |     private readonly IAuthService _authService; | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ namespace ThingsGateway.Admin.Application; | |||||||
| [Description("登录")] | [Description("登录")] | ||||||
| [Route("openapi/auth")] | [Route("openapi/auth")] | ||||||
| [Authorize(AuthenticationSchemes = "Bearer")] | [Authorize(AuthenticationSchemes = "Bearer")] | ||||||
| [LoggingMonitor] | [RequestAudit] | ||||||
| [ApiController] | [ApiController] | ||||||
| public class OpenApiController : ControllerBase | public class OpenApiController : ControllerBase | ||||||
| { | { | ||||||
|   | |||||||
| @@ -0,0 +1,15 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | namespace System.Logging; | ||||||
|  |  | ||||||
|  | public class RequestAudit | ||||||
|  | { | ||||||
|  | } | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | // ------------------------------------------------------------------------ | ||||||
|  | // 版权信息 | ||||||
|  | // 版权归百小僧及百签科技(广东)有限公司所有。 | ||||||
|  | // 所有权利保留。 | ||||||
|  | // 官方网站:https://baiqian.com | ||||||
|  | // | ||||||
|  | // 许可证信息 | ||||||
|  | // 项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。 | ||||||
|  | // 许可证的完整文本可以在源代码树根目录中的 LICENSE-APACHE 和 LICENSE-MIT 文件中找到。 | ||||||
|  | // ------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | using ThingsGateway.DependencyInjection; | ||||||
|  |  | ||||||
|  | namespace System; | ||||||
|  |  | ||||||
|  | [SuppressSniffer, AttributeUsage(AttributeTargets.Method | AttributeTargets.Class, Inherited = true, AllowMultiple = false)] | ||||||
|  | public sealed class RequestAuditAttribute : Attribute | ||||||
|  | { | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,98 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
|  | public class RequestAuditData | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// 分类 | ||||||
|  |     /// </summary> | ||||||
|  |     public string CateGory { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 客户端信息 | ||||||
|  |     /// </summary> | ||||||
|  |     public UserAgent Client { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 请求方法:POST/GET | ||||||
|  |     /// </summary> | ||||||
|  |     public string Method { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 操作名称 | ||||||
|  |     /// </summary> | ||||||
|  |     public string Operation { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 请求地址 | ||||||
|  |     /// </summary> | ||||||
|  |     public string Path { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 方法名称 | ||||||
|  |     /// </summary> | ||||||
|  |     public string ActionName { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 认证信息 | ||||||
|  |     /// </summary> | ||||||
|  |     public List<AuthorizationClaims> AuthorizationClaims { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 控制器名 | ||||||
|  |     /// </summary> | ||||||
|  |     public string ControllerName { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 异常信息 | ||||||
|  |     /// </summary> | ||||||
|  |     public LogException Exception { get; set; } | ||||||
|  |  | ||||||
|  |     public long TimeOperationElapsedMilliseconds { get; set; } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 服务端 | ||||||
|  |     /// </summary> | ||||||
|  |     public string LocalIPv4 { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 日志时间 | ||||||
|  |     /// </summary> | ||||||
|  |     public DateTimeOffset LogDateTime { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 参数列表 | ||||||
|  |     /// </summary> | ||||||
|  |     public List<Parameters> Parameters { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 客户端IPV4地址 | ||||||
|  |     /// </summary> | ||||||
|  |     public string RemoteIPv4 { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 请求地址 | ||||||
|  |     /// </summary> | ||||||
|  |     public string RequestUrl { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 返回信息 | ||||||
|  |     /// </summary> | ||||||
|  |     public object ReturnInformation { get; set; } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 验证错误信息 | ||||||
|  |     /// </summary> | ||||||
|  |     public Validation Validation { get; set; } | ||||||
|  | } | ||||||
|  |  | ||||||
| @@ -0,0 +1,301 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | using Microsoft.AspNetCore.Http; | ||||||
|  | using Microsoft.AspNetCore.Mvc; | ||||||
|  | using Microsoft.AspNetCore.Mvc.Controllers; | ||||||
|  | using Microsoft.AspNetCore.Mvc.Filters; | ||||||
|  | using Microsoft.Extensions.DependencyInjection; | ||||||
|  | using Microsoft.Extensions.Logging; | ||||||
|  |  | ||||||
|  | using System.Diagnostics; | ||||||
|  | using System.Logging; | ||||||
|  |  | ||||||
|  | using ThingsGateway.FriendlyException; | ||||||
|  | using ThingsGateway.Logging; | ||||||
|  | using ThingsGateway.NewLife.Json.Extension; | ||||||
|  | using ThingsGateway.UnifyResult; | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
|  | public class RequestAuditFilter : IAsyncActionFilter, IOrderedFilter | ||||||
|  | { | ||||||
|  |     private const int FilterOrder = -3000; | ||||||
|  |     public int Order => FilterOrder; | ||||||
|  |  | ||||||
|  |     public async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next) | ||||||
|  |     { | ||||||
|  |         var timeOperation = Stopwatch.StartNew(); | ||||||
|  |         var resultContext = await next().ConfigureAwait(false); | ||||||
|  |         // 计算接口执行时间 | ||||||
|  |         timeOperation.Stop(); | ||||||
|  |  | ||||||
|  |         var controllerActionDescriptor = (context.ActionDescriptor as ControllerActionDescriptor); | ||||||
|  |         // 获取动作方法描述器 | ||||||
|  |         var actionMethod = controllerActionDescriptor?.MethodInfo; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // 处理 Blazor Server | ||||||
|  |         if (actionMethod == null) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 排除 WebSocket 请求处理 | ||||||
|  |         if (context.HttpContext.IsWebSocketRequest()) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // 如果贴了 [SuppressMonitor] 特性则跳过 | ||||||
|  |         if (actionMethod.IsDefined(typeof(SuppressRequestAuditAttribute), true) | ||||||
|  |             || actionMethod.DeclaringType.IsDefined(typeof(SuppressRequestAuditAttribute), true)) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // 只有方法贴有特性才进行审计 | ||||||
|  |         if ( | ||||||
|  |             !actionMethod.DeclaringType.IsDefined(typeof(RequestAuditAttribute), true) | ||||||
|  |             && | ||||||
|  |             !actionMethod.IsDefined(typeof(RequestAuditAttribute), true)) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         var logData = new RequestAuditData(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         logData.TimeOperationElapsedMilliseconds = timeOperation.ElapsedMilliseconds; | ||||||
|  |  | ||||||
|  |         var resultHttpContext = (resultContext as FilterContext).HttpContext; | ||||||
|  |  | ||||||
|  |         // 获取 HttpContext 和 HttpRequest 对象 | ||||||
|  |         var httpContext = context.HttpContext; | ||||||
|  |         var httpRequest = httpContext.Request; | ||||||
|  |  | ||||||
|  |         // 获取客户端 Ipv4 地址 | ||||||
|  |         var remoteIPv4 = httpContext.GetRemoteIpAddressToIPv4(); | ||||||
|  |         logData.RemoteIPv4 = remoteIPv4; | ||||||
|  |         var requestUrl = Uri.UnescapeDataString(httpRequest.GetRequestUrlAddress()); | ||||||
|  |         logData.RequestUrl = requestUrl; | ||||||
|  |  | ||||||
|  |         object returnValue = null; | ||||||
|  |         Type finalReturnType; | ||||||
|  |         var result = resultContext.Result as IActionResult; | ||||||
|  |         // 解析返回值 | ||||||
|  |         if (UnifyContext.CheckVaildResult(result, out var data)) | ||||||
|  |         { | ||||||
|  |             returnValue = data; | ||||||
|  |             finalReturnType = data?.GetType(); | ||||||
|  |         } | ||||||
|  |         // 处理文件类型 | ||||||
|  |         else if (result is FileResult fresult) | ||||||
|  |         { | ||||||
|  |             returnValue = new | ||||||
|  |             { | ||||||
|  |                 FileName = fresult.FileDownloadName, | ||||||
|  |                 fresult.ContentType, | ||||||
|  |                 Length = fresult is FileContentResult cresult ? (object)cresult.FileContents.Length : null | ||||||
|  |             }; | ||||||
|  |             finalReturnType = fresult?.GetType(); | ||||||
|  |         } | ||||||
|  |         else finalReturnType = result?.GetType(); | ||||||
|  |  | ||||||
|  |         logData.ReturnInformation = returnValue; | ||||||
|  |  | ||||||
|  |         //获取客户端信息 | ||||||
|  |         var client = App.GetService<IAppService>().UserAgent; | ||||||
|  |         //操作名称默认是控制器名加方法名,自定义操作名称要在action上加Description特性 | ||||||
|  |         var option = $"{controllerActionDescriptor.ControllerName}/{controllerActionDescriptor.ActionName}"; | ||||||
|  |  | ||||||
|  |         var desc = App.CreateLocalizerByType(controllerActionDescriptor.ControllerTypeInfo.AsType())[actionMethod.Name]; | ||||||
|  |         //获取特性 | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         logData.CateGory = desc.Value;//传操作名称 | ||||||
|  |         logData.Operation = desc.Value;//传操作名称 | ||||||
|  |         logData.Client = client; | ||||||
|  |         logData.Path = httpContext.Request.Path.Value;//请求地址 | ||||||
|  |         logData.Method = httpContext.Request.Method;//请求方法 | ||||||
|  |  | ||||||
|  |         logData.ControllerName = controllerActionDescriptor.ControllerName; | ||||||
|  |         logData.ActionName = controllerActionDescriptor.ActionName; | ||||||
|  |  | ||||||
|  |         logData.AuthorizationClaims = new(); | ||||||
|  |         // 获取授权用户 | ||||||
|  |         var user = httpContext.User; | ||||||
|  |         foreach (var claim in user.Claims) | ||||||
|  |         { | ||||||
|  |             logData.AuthorizationClaims.Add(new AuthorizationClaims | ||||||
|  |             { | ||||||
|  |                 Type = claim.Type, | ||||||
|  |                 Value = claim.Value, | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         logData.LocalIPv4 = httpContext.GetLocalIpAddressToIPv4(); | ||||||
|  |         logData.LogDateTime = DateTimeOffset.Now; | ||||||
|  |         var parameterValues = context.ActionArguments; | ||||||
|  |  | ||||||
|  |         logData.Parameters = new(); | ||||||
|  |         var parameters = actionMethod.GetParameters(); | ||||||
|  |  | ||||||
|  |         foreach (var parameter in parameters) | ||||||
|  |         { | ||||||
|  |             // 判断是否禁用记录特定参数 | ||||||
|  |             if (parameter.IsDefined(typeof(SuppressRequestAuditAttribute), false)) continue; | ||||||
|  |  | ||||||
|  |             // 排除标记 [FromServices] 的解析 | ||||||
|  |             if (parameter.IsDefined(typeof(FromServicesAttribute), false)) continue; | ||||||
|  |  | ||||||
|  |             var name = parameter.Name; | ||||||
|  |             var parameterType = parameter.ParameterType; | ||||||
|  |  | ||||||
|  |             _ = parameterValues.TryGetValue(name, out var value); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |             var par = new Parameters() | ||||||
|  |             { | ||||||
|  |                 Name = name, | ||||||
|  |             }; | ||||||
|  |             logData.Parameters.Add(par); | ||||||
|  |  | ||||||
|  |             object rawValue = default; | ||||||
|  |  | ||||||
|  |             // 文件类型参数 | ||||||
|  |             if (value is IFormFile || value is List<IFormFile>) | ||||||
|  |             { | ||||||
|  |                 // 单文件 | ||||||
|  |                 if (value is IFormFile formFile) | ||||||
|  |                 { | ||||||
|  |                     var fileSize = Math.Round(formFile.Length / 1024D); | ||||||
|  |                     rawValue = new | ||||||
|  |                     { | ||||||
|  |                         name = formFile.Name, | ||||||
|  |                         fileName = formFile.FileName, | ||||||
|  |                         length = formFile.Length, | ||||||
|  |                         contentType = formFile.ContentType | ||||||
|  |                     }; | ||||||
|  |                 } | ||||||
|  |                 // 多文件 | ||||||
|  |                 else if (value is List<IFormFile> formFiles) | ||||||
|  |                 { | ||||||
|  |                     var rawValues1 = new List<object>(); | ||||||
|  |                     for (var i = 0; i < formFiles.Count; i++) | ||||||
|  |                     { | ||||||
|  |                         var file = formFiles[i]; | ||||||
|  |                         var size = Math.Round(file.Length / 1024D); | ||||||
|  |                         var rawValue1 = new | ||||||
|  |                         { | ||||||
|  |                             name = file.Name, | ||||||
|  |                             fileName = file.FileName, | ||||||
|  |                             length = file.Length, | ||||||
|  |                             contentType = file.ContentType | ||||||
|  |                         }; | ||||||
|  |                         rawValues1.Add(rawValue1); | ||||||
|  |                     } | ||||||
|  |                     rawValue = rawValues1; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             // 处理 byte[] 参数类型 | ||||||
|  |             else if (value is byte[] byteArray) | ||||||
|  |             { | ||||||
|  |                 rawValue = new | ||||||
|  |                 { | ||||||
|  |                     length = byteArray.Length, | ||||||
|  |                 }; | ||||||
|  |             } | ||||||
|  |             // 处理基元类型,字符串类型和空值 | ||||||
|  |             else if (parameterType.IsPrimitive || value is string || value == null) | ||||||
|  |             { | ||||||
|  |                 rawValue = value; | ||||||
|  |             } | ||||||
|  |             // 其他类型统一进行序列化 | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 rawValue = value; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             par.Value = rawValue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // 获取异常对象情况 | ||||||
|  |         Exception exception = resultContext.Exception; | ||||||
|  |         if (exception is AppFriendlyException friendlyException) | ||||||
|  |         { | ||||||
|  |             logData.Validation = new(); | ||||||
|  |             logData.Validation.Message = friendlyException.Message; | ||||||
|  |         } | ||||||
|  |         else if (exception != null) | ||||||
|  |         { | ||||||
|  |             logData.Exception = new(); | ||||||
|  |             logData.Exception.Message = exception.Message; | ||||||
|  |             logData.Exception.StackTrace = exception.StackTrace; | ||||||
|  |             logData.Exception.Type = HandleGenericType(exception.GetType()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // 创建日志记录器 | ||||||
|  |         var logger = httpContext.RequestServices.GetRequiredService<ILogger<RequestAudit>>(); | ||||||
|  |  | ||||||
|  |         var logContext = new LogContext(); | ||||||
|  |  | ||||||
|  |         logContext.Set(nameof(RequestAuditData), logData); | ||||||
|  |  | ||||||
|  |         // 设置日志上下文 | ||||||
|  |         using var scope = logger.ScopeContext(logContext); | ||||||
|  |  | ||||||
|  |         if (exception == null) | ||||||
|  |         { | ||||||
|  |             logger.Log(LogLevel.Information, $"{logData.Method}:{logData.Path}-{logData.Operation}"); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             logger.Log(LogLevel.Warning, $"{logData.Method}:{logData.Path}-{logData.Operation}{Environment.NewLine}{logData.Exception.ToSystemTextJsonString()}"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 处理泛型类型转字符串打印问题 | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="type"></param> | ||||||
|  |     /// <returns></returns> | ||||||
|  |     private static string HandleGenericType(Type type) | ||||||
|  |     { | ||||||
|  |         if (type == null) return string.Empty; | ||||||
|  |  | ||||||
|  |         var typeName = type.FullName ?? (!string.IsNullOrEmpty(type.Namespace) ? type.Namespace + "." : string.Empty) + type.Name; | ||||||
|  |  | ||||||
|  |         // 处理泛型类型问题 | ||||||
|  |         if (type.IsConstructedGenericType) | ||||||
|  |         { | ||||||
|  |             var prefix = type.GetGenericArguments() | ||||||
|  |                 .Select(genericArg => HandleGenericType(genericArg)) | ||||||
|  |                 .Aggregate((previous, current) => previous + ", " + current); | ||||||
|  |  | ||||||
|  |             typeName = typeName.Split('`').First() + "<" + prefix + ">"; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return typeName; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | // ------------------------------------------------------------------------ | ||||||
|  | // 版权信息 | ||||||
|  | // 版权归百小僧及百签科技(广东)有限公司所有。 | ||||||
|  | // 所有权利保留。 | ||||||
|  | // 官方网站:https://baiqian.com | ||||||
|  | // | ||||||
|  | // 许可证信息 | ||||||
|  | // 项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。 | ||||||
|  | // 许可证的完整文本可以在源代码树根目录中的 LICENSE-APACHE 和 LICENSE-MIT 文件中找到。 | ||||||
|  | // ------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | using ThingsGateway.DependencyInjection; | ||||||
|  |  | ||||||
|  | namespace System; | ||||||
|  |  | ||||||
|  | [SuppressSniffer, AttributeUsage(AttributeTargets.Method | AttributeTargets.Class, Inherited = true, AllowMultiple = false)] | ||||||
|  | public sealed class SuppressRequestAuditAttribute : Attribute | ||||||
|  | { | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -51,7 +51,7 @@ public class HardwareInfo | |||||||
|     /// 进程占用内存 |     /// 进程占用内存 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     [AutoGenerateColumn(Ignore = true)] |     [AutoGenerateColumn(Ignore = true)] | ||||||
|     public string WorkingSet { get; set; } |     public int WorkingSet { get; set; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 更新时间 |     /// 更新时间 | ||||||
|   | |||||||
| @@ -17,6 +17,7 @@ using System.Runtime.InteropServices; | |||||||
|  |  | ||||||
| using ThingsGateway.Extension; | using ThingsGateway.Extension; | ||||||
| using ThingsGateway.NewLife; | using ThingsGateway.NewLife; | ||||||
|  | using ThingsGateway.NewLife.Caching; | ||||||
| using ThingsGateway.NewLife.Threading; | using ThingsGateway.NewLife.Threading; | ||||||
| using ThingsGateway.Schedule; | using ThingsGateway.Schedule; | ||||||
|  |  | ||||||
| @@ -51,11 +52,20 @@ public class HardwareJob : IJob, IHardwareJob | |||||||
|  |  | ||||||
|     #endregion 属性 |     #endregion 属性 | ||||||
|  |  | ||||||
|  |     private MemoryCache MemoryCache = new() { }; | ||||||
|  |     private const string CacheKey = "HistoryHardwareInfo"; | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     public async Task<List<HistoryHardwareInfo>> GetHistoryHardwareInfos() |     public async Task<List<HistoryHardwareInfo>> GetHistoryHardwareInfos() | ||||||
|  |     { | ||||||
|  |         var historyHardwareInfos = MemoryCache.Get<List<HistoryHardwareInfo>>(CacheKey); | ||||||
|  |         if (historyHardwareInfos == null) | ||||||
|         { |         { | ||||||
|             using var db = DbContext.Db.GetConnectionScopeWithAttr<HistoryHardwareInfo>().CopyNew(); |             using var db = DbContext.Db.GetConnectionScopeWithAttr<HistoryHardwareInfo>().CopyNew(); | ||||||
|         return await db.Queryable<HistoryHardwareInfo>().ToListAsync().ConfigureAwait(false); |             historyHardwareInfos = await db.Queryable<HistoryHardwareInfo>().Where(a => a.Date > DateTime.Now.AddDays(-3)).ToListAsync().ConfigureAwait(false); | ||||||
|  |  | ||||||
|  |             MemoryCache.Set(CacheKey, historyHardwareInfos); | ||||||
|  |         } | ||||||
|  |         return historyHardwareInfos; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private bool error = false; |     private bool error = false; | ||||||
| @@ -94,7 +104,7 @@ public class HardwareJob : IJob, IHardwareJob | |||||||
|             { |             { | ||||||
|                 HardwareInfo.MachineInfo.Refresh(); |                 HardwareInfo.MachineInfo.Refresh(); | ||||||
|                 HardwareInfo.UpdateTime = TimerX.Now.ToDefaultDateTimeFormat(); |                 HardwareInfo.UpdateTime = TimerX.Now.ToDefaultDateTimeFormat(); | ||||||
|                 HardwareInfo.WorkingSet = (Environment.WorkingSet / 1024.0 / 1024.0).ToString("F2"); |                 HardwareInfo.WorkingSet = (Environment.WorkingSet / 1024.0 / 1024.0).ToInt(); | ||||||
|                 error = false; |                 error = false; | ||||||
|             } |             } | ||||||
|             catch (Exception ex) |             catch (Exception ex) | ||||||
| @@ -116,17 +126,22 @@ public class HardwareJob : IJob, IHardwareJob | |||||||
|                             var his = new HistoryHardwareInfo() |                             var his = new HistoryHardwareInfo() | ||||||
|                             { |                             { | ||||||
|                                 Date = TimerX.Now, |                                 Date = TimerX.Now, | ||||||
|                                 DriveUsage = (100 - (HardwareInfo.DriveInfo.TotalFreeSpace * 100.00 / HardwareInfo.DriveInfo.TotalSize)).ToString("F2"), |                                 DriveUsage = (100 - (HardwareInfo.DriveInfo.TotalFreeSpace * 100.00 / HardwareInfo.DriveInfo.TotalSize)).ToInt(), | ||||||
|                                 Battery = (HardwareInfo.MachineInfo.Battery * 100).ToString("F2"), |                                 Battery = (HardwareInfo.MachineInfo.Battery * 100).ToInt(), | ||||||
|                                 MemoryUsage = (HardwareInfo.WorkingSet), |                                 MemoryUsage = (HardwareInfo.WorkingSet), | ||||||
|                                 CpuUsage = (HardwareInfo.MachineInfo.CpuRate * 100).ToString("F2"), |                                 CpuUsage = (HardwareInfo.MachineInfo.CpuRate * 100).ToInt(), | ||||||
|                                 Temperature = (HardwareInfo.MachineInfo.Temperature).ToString("F2"), |                                 Temperature = (HardwareInfo.MachineInfo.Temperature).ToInt(), | ||||||
|                             }; |                             }; | ||||||
|                             await db.Insertable(his).ExecuteCommandAsync(stoppingToken).ConfigureAwait(false); |                             await db.Insertable(his).ExecuteCommandAsync(stoppingToken).ConfigureAwait(false); | ||||||
|  |                             MemoryCache.Remove(CacheKey); | ||||||
|                         } |                         } | ||||||
|                         var sevenDaysAgo = TimerX.Now.AddDays(-HardwareInfoOptions.DaysAgo); |                         var sevenDaysAgo = TimerX.Now.AddDays(-HardwareInfoOptions.DaysAgo); | ||||||
|                         //删除特定信息 |                         //删除特定信息 | ||||||
|                         await db.Deleteable<HistoryHardwareInfo>(a => a.Date <= sevenDaysAgo).ExecuteCommandAsync(stoppingToken).ConfigureAwait(false); |                         var result = await db.Deleteable<HistoryHardwareInfo>(a => a.Date <= sevenDaysAgo).ExecuteCommandAsync(stoppingToken).ConfigureAwait(false); | ||||||
|  |                         if (result > 0) | ||||||
|  |                         { | ||||||
|  |                             MemoryCache.Remove(CacheKey); | ||||||
|  |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 error = false; |                 error = false; | ||||||
|   | |||||||
| @@ -19,23 +19,23 @@ public class HistoryHardwareInfo | |||||||
| { | { | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     [SugarColumn(ColumnDescription = "磁盘使用率")] |     [SugarColumn(ColumnDescription = "磁盘使用率")] | ||||||
|     public string DriveUsage { get; set; } |     public int DriveUsage { get; set; } | ||||||
|  |  | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     [SugarColumn(ColumnDescription = "内存")] |     [SugarColumn(ColumnDescription = "内存")] | ||||||
|     public string MemoryUsage { get; set; } |     public int MemoryUsage { get; set; } | ||||||
|  |  | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     [SugarColumn(ColumnDescription = "CPU使用率")] |     [SugarColumn(ColumnDescription = "CPU使用率")] | ||||||
|     public string CpuUsage { get; set; } |     public int CpuUsage { get; set; } | ||||||
|  |  | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     [SugarColumn(ColumnDescription = "温度")] |     [SugarColumn(ColumnDescription = "温度")] | ||||||
|     public string Temperature { get; set; } |     public int Temperature { get; set; } | ||||||
|  |  | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     [SugarColumn(ColumnDescription = "电池")] |     [SugarColumn(ColumnDescription = "电池")] | ||||||
|     public string Battery { get; set; } |     public int Battery { get; set; } | ||||||
|  |  | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     [SugarColumn(ColumnDescription = "时间")] |     [SugarColumn(ColumnDescription = "时间")] | ||||||
|   | |||||||
| @@ -16,9 +16,6 @@ using ThingsGateway.Extension; | |||||||
| using ThingsGateway.FriendlyException; | using ThingsGateway.FriendlyException; | ||||||
| using ThingsGateway.Logging; | using ThingsGateway.Logging; | ||||||
| using ThingsGateway.NewLife.Json.Extension; | using ThingsGateway.NewLife.Json.Extension; | ||||||
| using ThingsGateway.Razor; |  | ||||||
|  |  | ||||||
| using UAParser; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.Admin.Application; | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
| @@ -41,33 +38,31 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | |||||||
|     /// <param name="flush"></param> |     /// <param name="flush"></param> | ||||||
|     public async Task WriteAsync(LogMessage logMsg, bool flush) |     public async Task WriteAsync(LogMessage logMsg, bool flush) | ||||||
|     { |     { | ||||||
|         //获取请求json字符串 |  | ||||||
|         var jsonString = logMsg.Context.Get("loggingMonitor").ToString(); |  | ||||||
|         //转成实体 |         //转成实体 | ||||||
|         var loggingMonitor = jsonString.FromJsonNetString<LoggingMonitorJson>(); |         var requestAuditData = logMsg.Context.Get(nameof(RequestAuditData)) as RequestAuditData; | ||||||
|         //日志时间赋值 |         //日志时间赋值 | ||||||
|         loggingMonitor.LogDateTime = logMsg.LogDateTime; |         requestAuditData.LogDateTime = logMsg.LogDateTime; | ||||||
|         // loggingMonitor.ReturnInformation.Value |         // requestAuditData.ReturnInformation.Value | ||||||
|         //验证失败不记录日志 |         //验证失败不记录日志 | ||||||
|         bool save = false; |         bool save = false; | ||||||
|         if (loggingMonitor.Validation == null) |         if (requestAuditData.Validation == null) | ||||||
|         { |         { | ||||||
|             var operation = logMsg.Context.Get(LoggingConst.Operation).ToString();//获取操作名称 |             var operation = requestAuditData.Operation;//获取操作名称 | ||||||
|             var client = (ClientInfo)logMsg.Context.Get(LoggingConst.Client);//获取客户端信息 |             var client = requestAuditData.Client;//获取客户端信息 | ||||||
|             var path = logMsg.Context.Get(LoggingConst.Path).ToString();//获取操作名称 |             var path = requestAuditData.Path;//获取操作名称 | ||||||
|             var method = logMsg.Context.Get(LoggingConst.Method).ToString();//获取方法 |             var method = requestAuditData.Method;//获取方法 | ||||||
|             //表示访问日志 |             //表示访问日志 | ||||||
|             if (path == "/api/auth/login" || path == "/api/auth/logout") |             if (path == "/api/auth/login" || path == "/api/auth/logout") | ||||||
|             { |             { | ||||||
|                 //如果没有异常信息 |                 //如果没有异常信息 | ||||||
|                 if (loggingMonitor.Exception == null) |                 if (requestAuditData.Exception == null) | ||||||
|                 { |                 { | ||||||
|                     save = await CreateVisitLog(operation, path, loggingMonitor, client, flush).ConfigureAwait(false);//添加到访问日志 |                     save = await CreateVisitLog(operation, path, requestAuditData, client, flush).ConfigureAwait(false);//添加到访问日志 | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     //添加到异常日志 |                     //添加到异常日志 | ||||||
|                     save = await CreateOperationLog(operation, path, loggingMonitor, client, flush).ConfigureAwait(false); |                     save = await CreateOperationLog(operation, path, requestAuditData, client, flush).ConfigureAwait(false); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             else |             else | ||||||
| @@ -76,7 +71,7 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | |||||||
|                 if (!operation.IsNullOrWhiteSpace() && method == "POST") |                 if (!operation.IsNullOrWhiteSpace() && method == "POST") | ||||||
|                 { |                 { | ||||||
|                     //添加到操作日志 |                     //添加到操作日志 | ||||||
|                     save = await CreateOperationLog(operation, path, loggingMonitor, client, flush).ConfigureAwait(false); |                     save = await CreateOperationLog(operation, path, requestAuditData, client, flush).ConfigureAwait(false); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -91,27 +86,21 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <param name="operation">操作名称</param> |     /// <param name="operation">操作名称</param> | ||||||
|     /// <param name="path">请求地址</param> |     /// <param name="path">请求地址</param> | ||||||
|     /// <param name="loggingMonitor">loggingMonitor</param> |     /// <param name="requestAuditData">requestAuditData</param> | ||||||
|     /// <param name="clientInfo">客户端信息</param> |     /// <param name="userAgent">客户端信息</param> | ||||||
|     /// <param name="flush"></param> |     /// <param name="flush"></param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     private async Task<bool> CreateOperationLog(string operation, string path, LoggingMonitorJson loggingMonitor, ClientInfo clientInfo, bool flush) |     private async Task<bool> CreateOperationLog(string operation, string path, RequestAuditData requestAuditData, UserAgent userAgent, bool flush) | ||||||
|     { |     { | ||||||
|         //账号 |         //账号 | ||||||
|         var opAccount = loggingMonitor.AuthorizationClaims?.Where(it => it.Type == ClaimConst.Account).Select(it => it.Value).FirstOrDefault(); |         var opAccount = requestAuditData.AuthorizationClaims?.Where(it => it.Type == ClaimConst.Account).Select(it => it.Value).FirstOrDefault(); | ||||||
|  |  | ||||||
|         //获取参数json字符串, |         //获取参数json字符串, | ||||||
|         var paramJson = loggingMonitor.Parameters == null || loggingMonitor.Parameters.Count == 0 ? null : loggingMonitor.Parameters[0].Value.ToJsonNetString(); |         var paramJson = requestAuditData.Parameters == null || requestAuditData.Parameters.Count == 0 ? null : requestAuditData.Parameters.ToSystemTextJsonString(); | ||||||
|  |  | ||||||
|         //获取结果json字符串 |         //获取结果json字符串 | ||||||
|         var resultJson = string.Empty; |         var resultJson = requestAuditData.ReturnInformation?.ToSystemTextJsonString(); | ||||||
|         if (loggingMonitor.ReturnInformation != null)//如果有返回值 |  | ||||||
|         { |  | ||||||
|             if (loggingMonitor.ReturnInformation.Value != null)//如果返回值不为空 |  | ||||||
|             { |  | ||||||
|                 resultJson = loggingMonitor.ReturnInformation.Value.ToJsonNetString(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         //操作日志表实体 |         //操作日志表实体 | ||||||
|         var sysLogOperate = new SysOperateLog |         var sysLogOperate = new SysOperateLog | ||||||
| @@ -119,29 +108,29 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | |||||||
|             Name = operation, |             Name = operation, | ||||||
|             Category = LogCateGoryEnum.Operate, |             Category = LogCateGoryEnum.Operate, | ||||||
|             ExeStatus = true, |             ExeStatus = true, | ||||||
|             OpIp = loggingMonitor.RemoteIPv4, |             OpIp = requestAuditData.RemoteIPv4, | ||||||
|             OpBrowser = clientInfo?.UA?.Family + clientInfo?.UA?.Major, |             OpBrowser = userAgent?.Browser, | ||||||
|             OpOs = clientInfo?.OS?.Family + clientInfo?.OS?.Major, |             OpOs = userAgent?.Platform, | ||||||
|             OpTime = loggingMonitor.LogDateTime.LocalDateTime, |             OpTime = requestAuditData.LogDateTime.LocalDateTime, | ||||||
|             OpAccount = opAccount, |             OpAccount = opAccount, | ||||||
|             ReqMethod = loggingMonitor.HttpMethod, |             ReqMethod = requestAuditData.Method, | ||||||
|             ReqUrl = path, |             ReqUrl = path, | ||||||
|             ResultJson = resultJson, |             ResultJson = resultJson, | ||||||
|             ClassName = loggingMonitor.DisplayName, |             ClassName = requestAuditData.ControllerName, | ||||||
|             MethodName = loggingMonitor.ActionName, |             MethodName = requestAuditData.ActionName, | ||||||
|             ParamJson = paramJson, |             ParamJson = paramJson, | ||||||
|             VerificatId = UserManager.VerificatId, |             VerificatId = UserManager.VerificatId, | ||||||
|         }; |         }; | ||||||
|         //如果异常不为空 |         //如果异常不为空 | ||||||
|         if (loggingMonitor.Exception != null) |         if (requestAuditData.Exception != null) | ||||||
|         { |         { | ||||||
|             sysLogOperate.Category = LogCateGoryEnum.Exception;//操作类型为异常 |             sysLogOperate.Category = LogCateGoryEnum.Exception;//操作类型为异常 | ||||||
|             sysLogOperate.ExeStatus = false;//操作状态为失败 |             sysLogOperate.ExeStatus = false;//操作状态为失败 | ||||||
|  |  | ||||||
|             if (loggingMonitor.Exception.Type == typeof(AppFriendlyException).ToString()) |             if (requestAuditData.Exception.Type == typeof(AppFriendlyException).ToString()) | ||||||
|                 sysLogOperate.ExeMessage = loggingMonitor?.Exception.Message; |                 sysLogOperate.ExeMessage = requestAuditData?.Exception.Message; | ||||||
|             else |             else | ||||||
|                 sysLogOperate.ExeMessage = $"{loggingMonitor.Exception.Type}:{loggingMonitor.Exception.Message}{Environment.NewLine}{loggingMonitor.Exception.StackTrace}"; |                 sysLogOperate.ExeMessage = $"{requestAuditData.Exception.Type}:{requestAuditData.Exception.Message}{Environment.NewLine}{requestAuditData.Exception.StackTrace}"; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         _operateLogMessageQueue.Enqueue(sysLogOperate); |         _operateLogMessageQueue.Enqueue(sysLogOperate); | ||||||
| @@ -160,26 +149,25 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <param name="operation">访问类型</param> |     /// <param name="operation">访问类型</param> | ||||||
|     /// <param name="path"></param> |     /// <param name="path"></param> | ||||||
|     /// <param name="loggingMonitor">loggingMonitor</param> |     /// <param name="requestAuditData">requestAuditData</param> | ||||||
|     /// <param name="clientInfo">客户端信息</param> |     /// <param name="userAgent">客户端信息</param> | ||||||
|     /// <param name="flush"></param> |     /// <param name="flush"></param> | ||||||
|     private async Task<bool> CreateVisitLog(string operation, string path, LoggingMonitorJson loggingMonitor, ClientInfo clientInfo, bool flush) |     private async Task<bool> CreateVisitLog(string operation, string path, RequestAuditData requestAuditData, UserAgent userAgent, bool flush) | ||||||
|     { |     { | ||||||
|         long verificatId = 0;//验证Id |         long verificatId = 0;//验证Id | ||||||
|         var opAccount = "";//用户账号 |         var opAccount = "";//用户账号 | ||||||
|         if (path == "/api/auth/login") |         if (path == "/api/auth/login") | ||||||
|         { |         { | ||||||
|             //如果是登录,用户信息就从返回值里拿 |             //如果是登录,用户信息就从返回值里拿 | ||||||
|             var result = loggingMonitor.ReturnInformation?.Value?.ToJsonNetString();//返回值转json |             dynamic userInfo = requestAuditData.ReturnInformation; | ||||||
|             var userInfo = result.FromJsonNetString<UnifyResult<LoginOutput>>();//格式化成user表 |  | ||||||
|             opAccount = userInfo.Data.Account;//赋值账号 |             opAccount = userInfo.Data.Account;//赋值账号 | ||||||
|             verificatId = userInfo.Data.VerificatId; |             verificatId = userInfo.Data.VerificatId; | ||||||
|         } |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             //如果是登录出,用户信息就从AuthorizationClaims里拿 |             //如果是登录出,用户信息就从AuthorizationClaims里拿 | ||||||
|             opAccount = loggingMonitor.AuthorizationClaims.Where(it => it.Type == ClaimConst.Account).Select(it => it.Value).FirstOrDefault(); |             opAccount = requestAuditData.AuthorizationClaims.Where(it => it.Type == ClaimConst.Account).Select(it => it.Value).FirstOrDefault(); | ||||||
|             verificatId = loggingMonitor.AuthorizationClaims.Where(it => it.Type == ClaimConst.VerificatId).Select(it => it.Value).FirstOrDefault().ToLong(); |             verificatId = requestAuditData.AuthorizationClaims.Where(it => it.Type == ClaimConst.VerificatId).Select(it => it.Value).FirstOrDefault().ToLong(); | ||||||
|         } |         } | ||||||
|         //日志表实体 |         //日志表实体 | ||||||
|         var sysLogVisit = new SysOperateLog |         var sysLogVisit = new SysOperateLog | ||||||
| @@ -187,19 +175,19 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | |||||||
|             Name = operation, |             Name = operation, | ||||||
|             Category = path == "/api/auth/login" ? LogCateGoryEnum.Login : LogCateGoryEnum.Logout, |             Category = path == "/api/auth/login" ? LogCateGoryEnum.Login : LogCateGoryEnum.Logout, | ||||||
|             ExeStatus = true, |             ExeStatus = true, | ||||||
|             OpIp = loggingMonitor.RemoteIPv4, |             OpIp = requestAuditData.RemoteIPv4, | ||||||
|             OpBrowser = clientInfo?.UA?.Family + clientInfo?.UA?.Major, |             OpBrowser = userAgent?.Browser, | ||||||
|             OpOs = clientInfo?.OS?.Family + clientInfo?.OS?.Major, |             OpOs = userAgent?.Platform, | ||||||
|             OpTime = loggingMonitor.LogDateTime.LocalDateTime, |             OpTime = requestAuditData.LogDateTime.LocalDateTime, | ||||||
|             VerificatId = verificatId, |             VerificatId = verificatId, | ||||||
|             OpAccount = opAccount, |             OpAccount = opAccount, | ||||||
|  |  | ||||||
|             ReqMethod = loggingMonitor.HttpMethod, |             ReqMethod = requestAuditData.Method, | ||||||
|             ReqUrl = path, |             ReqUrl = path, | ||||||
|             ResultJson = loggingMonitor.ReturnInformation?.Value?.ToJsonNetString(), |             ResultJson = requestAuditData.ReturnInformation?.ToSystemTextJsonString(), | ||||||
|             ClassName = loggingMonitor.DisplayName, |             ClassName = requestAuditData.ControllerName, | ||||||
|             MethodName = loggingMonitor.ActionName, |             MethodName = requestAuditData.ActionName, | ||||||
|             ParamJson = loggingMonitor.Parameters?.ToJsonNetString(), |             ParamJson = requestAuditData.Parameters?.ToSystemTextJsonString(), | ||||||
|         }; |         }; | ||||||
|         _operateLogMessageQueue.Enqueue(sysLogVisit); |         _operateLogMessageQueue.Enqueue(sysLogVisit); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,12 +15,17 @@ using Microsoft.AspNetCore.WebUtilities; | |||||||
|  |  | ||||||
| using System.Security.Claims; | using System.Security.Claims; | ||||||
|  |  | ||||||
| using UAParser; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.Admin.Application; | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
| public class AppService : IAppService | public class AppService : IAppService | ||||||
| { | { | ||||||
|  |     private readonly IUserAgentService UserAgentService; | ||||||
|  |     private readonly IClaimsPrincipalService ClaimsPrincipalService; | ||||||
|  |     public AppService(IUserAgentService userAgentService, IClaimsPrincipalService claimsPrincipalService) | ||||||
|  |     { | ||||||
|  |         UserAgentService = userAgentService; | ||||||
|  |         ClaimsPrincipalService = claimsPrincipalService; | ||||||
|  |     } | ||||||
|     public string GetReturnUrl(string returnUrl) |     public string GetReturnUrl(string returnUrl) | ||||||
|     { |     { | ||||||
|         var url = QueryHelpers.AddQueryString(CookieAuthenticationDefaults.LoginPath, new Dictionary<string, string?> |         var url = QueryHelpers.AddQueryString(CookieAuthenticationDefaults.LoginPath, new Dictionary<string, string?> | ||||||
| @@ -41,18 +46,16 @@ public class AppService : IAppService | |||||||
|         { |         { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     public Parser Parser = Parser.GetDefault(); |     public UserAgent? UserAgent | ||||||
|     public ClientInfo? ClientInfo |  | ||||||
|     { |     { | ||||||
|         get |         get | ||||||
|         { |         { | ||||||
|             var str = App.HttpContext?.Request?.Headers?.UserAgent; |             var str = App.HttpContext?.Request?.Headers?.UserAgent; | ||||||
|             ClientInfo? clientInfo = null; |  | ||||||
|             if (!string.IsNullOrEmpty(str)) |             if (!string.IsNullOrEmpty(str)) | ||||||
|             { |             { | ||||||
|                 clientInfo = Parser.Parse(str); |                 return UserAgentService.Parse(str); | ||||||
|             } |             } | ||||||
|             return clientInfo; |             return null; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -69,7 +72,7 @@ public class AppService : IAppService | |||||||
|             ExpiresUtc = diffTime, |             ExpiresUtc = diffTime, | ||||||
|         }).ConfigureAwait(false); |         }).ConfigureAwait(false); | ||||||
|     } |     } | ||||||
|     public ClaimsPrincipal? User => App.User; |     public ClaimsPrincipal? User => ClaimsPrincipalService.User; | ||||||
|  |  | ||||||
|     public string? RemoteIpAddress => App.HttpContext?.GetRemoteIpAddressToIPv4(); |     public string? RemoteIpAddress => App.HttpContext?.GetRemoteIpAddressToIPv4(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,19 +13,17 @@ using Microsoft.Extensions.DependencyInjection; | |||||||
|  |  | ||||||
| using System.Security.Claims; | using System.Security.Claims; | ||||||
|  |  | ||||||
| using UAParser; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.Admin.Application; | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
| public class HybridAppService : IAppService | public class HybridAppService : IAppService | ||||||
| { | { | ||||||
|     public HybridAppService() |     public HybridAppService(IUserAgentService userAgentService) | ||||||
|     { |     { | ||||||
|         var str = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/127.0.0.0 Safari/537.36 Edg/127.0.0.0"; |         var str = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/127.0.0.0 Safari/537.36 Edg/127.0.0.0"; | ||||||
|         ClientInfo = Parser.GetDefault().Parse(str); |         UserAgent = userAgentService.Parse(str); | ||||||
|         RemoteIpAddress = "127.0.0.1"; |         RemoteIpAddress = "127.0.0.1"; | ||||||
|     } |     } | ||||||
|     public ClientInfo? ClientInfo { get; } |     public UserAgent? UserAgent { get; } | ||||||
|  |  | ||||||
|     private static BlazorHybridAuthenticationStateProvider _authenticationStateProvider; |     private static BlazorHybridAuthenticationStateProvider _authenticationStateProvider; | ||||||
|     private static BlazorHybridAuthenticationStateProvider AuthenticationStateProvider |     private static BlazorHybridAuthenticationStateProvider AuthenticationStateProvider | ||||||
|   | |||||||
| @@ -11,8 +11,6 @@ | |||||||
|  |  | ||||||
| using System.Security.Claims; | using System.Security.Claims; | ||||||
|  |  | ||||||
| using UAParser; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.Admin.Application; | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
| public interface IAppService | public interface IAppService | ||||||
| @@ -20,7 +18,7 @@ public interface IAppService | |||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// ClientInfo |     /// ClientInfo | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public ClientInfo? ClientInfo { get; } |     public UserAgent? UserAgent { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// ClaimsPrincipal |     /// ClaimsPrincipal | ||||||
|   | |||||||
| @@ -237,7 +237,7 @@ public class AuthService : IAuthService | |||||||
|         var logingEvent = new LoginEvent |         var logingEvent = new LoginEvent | ||||||
|         { |         { | ||||||
|             Ip = _appService.RemoteIpAddress, |             Ip = _appService.RemoteIpAddress, | ||||||
|             Device = App.GetService<IAppService>().ClientInfo?.OS?.ToString(), |             Device = App.GetService<IAppService>().UserAgent?.Platform, | ||||||
|             Expire = expire, |             Expire = expire, | ||||||
|             SysUser = sysUser, |             SysUser = sysUser, | ||||||
|             VerificatId = verificatId |             VerificatId = verificatId | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ internal sealed class SysDictService : BaseService<SysDict>, ISysDictService | |||||||
|         //更新数据 |         //更新数据 | ||||||
|         List<SysDict> dicts = new List<SysDict>() |         List<SysDict> dicts = new List<SysDict>() | ||||||
|         { |         { | ||||||
|             new SysDict() { DictType = DictTypeEnum.System, Category = nameof(PagePolicy), Name = nameof(PagePolicy.Shortcuts), Code = input.Shortcuts.ToJsonNetString() }, |             new SysDict() { DictType = DictTypeEnum.System, Category = nameof(PagePolicy), Name = nameof(PagePolicy.Shortcuts), Code = input.Shortcuts.ToSystemTextJsonString() }, | ||||||
|     }; |     }; | ||||||
|         var storageable = await db.Storageable(dicts).WhereColumns(it => new { it.DictType, it.Category, it.Name }).ToStorageAsync().ConfigureAwait(false); |         var storageable = await db.Storageable(dicts).WhereColumns(it => new { it.DictType, it.Category, it.Name }).ToStorageAsync().ConfigureAwait(false); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -16,9 +16,9 @@ namespace ThingsGateway.Admin.Application; | |||||||
| /// 内存推送事件服务 | /// 内存推送事件服务 | ||||||
| /// </summary> | /// </summary> | ||||||
| /// <typeparam name="TEntry"></typeparam> | /// <typeparam name="TEntry"></typeparam> | ||||||
| public class EventService<TEntry> : IEventService<TEntry> | public class EventService<TEntry> : IEventService<TEntry>, IDisposable | ||||||
| { | { | ||||||
|     private ConcurrentDictionary<string, Func<TEntry, Task>> Cache { get; } = new(); |     private ConcurrentDictionary<string, Func<TEntry, Task>> Cache = new(); | ||||||
|  |  | ||||||
|     public void Dispose() |     public void Dispose() | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -11,8 +11,6 @@ | |||||||
| using Microsoft.AspNetCore.Http.Connections.Features; | using Microsoft.AspNetCore.Http.Connections.Features; | ||||||
| using Microsoft.AspNetCore.SignalR; | using Microsoft.AspNetCore.SignalR; | ||||||
|  |  | ||||||
| using Yitter.IdGenerator; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.Admin.Application; | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| @@ -28,7 +26,7 @@ public class UserIdProvider : IUserIdProvider | |||||||
|  |  | ||||||
|         if (UserId > 0) |         if (UserId > 0) | ||||||
|         { |         { | ||||||
|             return $"{UserId}{SysHub.Separate}{YitIdHelper.NextId()}";//返回用户ID |             return $"{UserId}{SysHub.Separate}{CommonUtils.GetSingleId()}";//返回用户ID | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return connection.ConnectionId; |         return connection.ConnectionId; | ||||||
|   | |||||||
| @@ -277,7 +277,7 @@ internal sealed class SysRoleService : BaseService<SysRole>, ISysRoleService | |||||||
|         if (isSuperAdmin) |         if (isSuperAdmin) | ||||||
|             throw Oops.Bah(Localizer["CanotGrantAdmin"]); |             throw Oops.Bah(Localizer["CanotGrantAdmin"]); | ||||||
|         var menuIds = input.GrantInfoList.Select(it => it.MenuId).ToList();//菜单ID |         var menuIds = input.GrantInfoList.Select(it => it.MenuId).ToList();//菜单ID | ||||||
|         var extJsons = input.GrantInfoList.Select(it => it.ToJsonNetString()).ToList();//拓展信息 |         var extJsons = input.GrantInfoList.Select(it => it.ToSystemTextJsonString()).ToList();//拓展信息 | ||||||
|         var relationRoles = new List<SysRelation>();//要添加的角色资源和授权关系表 |         var relationRoles = new List<SysRelation>();//要添加的角色资源和授权关系表 | ||||||
|         var sysRole = (await GetAllAsync().ConfigureAwait(false)).FirstOrDefault(it => it.Id == input.Id);//获取角色 |         var sysRole = (await GetAllAsync().ConfigureAwait(false)).FirstOrDefault(it => it.Id == input.Id);//获取角色 | ||||||
|  |  | ||||||
| @@ -338,7 +338,7 @@ internal sealed class SysRoleService : BaseService<SysRole>, ISysRoleService | |||||||
|                     ExtJson = new RelationPermission |                     ExtJson = new RelationPermission | ||||||
|                     { |                     { | ||||||
|                         ApiUrl = it.ApiRoute, |                         ApiUrl = it.ApiRoute, | ||||||
|                     }.ToJsonNetString() |                     }.ToSystemTextJsonString() | ||||||
|                 }); |                 }); | ||||||
|                 relationRoles.AddRange(relationRolePer);//合并列表 |                 relationRoles.AddRange(relationRolePer);//合并列表 | ||||||
|             } |             } | ||||||
| @@ -410,7 +410,7 @@ internal sealed class SysRoleService : BaseService<SysRole>, ISysRoleService | |||||||
|         if (sysRole != null) |         if (sysRole != null) | ||||||
|         { |         { | ||||||
|             await _relationService.SaveRelationBatchAsync(RelationCategoryEnum.RoleHasOpenApiPermission, input.Id, |             await _relationService.SaveRelationBatchAsync(RelationCategoryEnum.RoleHasOpenApiPermission, input.Id, | ||||||
|                  input.GrantInfoList.Select(a => (a.ApiUrl, a.ToJsonNetString())) |                  input.GrantInfoList.Select(a => (a.ApiUrl, a.ToSystemTextJsonString())) | ||||||
|                 , true).ConfigureAwait(false);//添加到数据库 |                 , true).ConfigureAwait(false);//添加到数据库 | ||||||
|             await ClearTokenUtil.DeleteUserCacheByRoleIds(new List<long> { input.Id }).ConfigureAwait(false);//清除角色下用户缓存 |             await ClearTokenUtil.DeleteUserCacheByRoleIds(new List<long> { input.Id }).ConfigureAwait(false);//清除角色下用户缓存 | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -435,7 +435,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService | |||||||
|         if (sysUser != null) |         if (sysUser != null) | ||||||
|         { |         { | ||||||
|             await _relationService.SaveRelationBatchAsync(RelationCategoryEnum.UserHasOpenApiPermission, input.Id, |             await _relationService.SaveRelationBatchAsync(RelationCategoryEnum.UserHasOpenApiPermission, input.Id, | ||||||
|                  input.GrantInfoList.Select(a => (a.ApiUrl, a.ToJsonNetString())), |                  input.GrantInfoList.Select(a => (a.ApiUrl, a.ToSystemTextJsonString())), | ||||||
|                 true).ConfigureAwait(false);//添加到数据库 |                 true).ConfigureAwait(false);//添加到数据库 | ||||||
|             DeleteUserFromCache(input.Id); |             DeleteUserFromCache(input.Id); | ||||||
|         } |         } | ||||||
| @@ -557,7 +557,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService | |||||||
|     public async Task GrantResourceAsync(GrantResourceData input) |     public async Task GrantResourceAsync(GrantResourceData input) | ||||||
|     { |     { | ||||||
|         var menuIds = input.GrantInfoList.Select(it => it.MenuId).ToList();//菜单ID |         var menuIds = input.GrantInfoList.Select(it => it.MenuId).ToList();//菜单ID | ||||||
|         var extJsons = input.GrantInfoList.Select(it => it.ToJsonNetString()).ToList();//拓展信息 |         var extJsons = input.GrantInfoList.Select(it => it.ToSystemTextJsonString()).ToList();//拓展信息 | ||||||
|         var relationUsers = new List<SysRelation>();//要添加的用户资源和授权关系表 |         var relationUsers = new List<SysRelation>();//要添加的用户资源和授权关系表 | ||||||
|         var sysUser = await GetUserByIdAsync(input.Id).ConfigureAwait(false);//获取用户 |         var sysUser = await GetUserByIdAsync(input.Id).ConfigureAwait(false);//获取用户 | ||||||
|         await CheckApiDataScopeAsync(sysUser.OrgId, sysUser.CreateUserId).ConfigureAwait(false); |         await CheckApiDataScopeAsync(sysUser.OrgId, sysUser.CreateUserId).ConfigureAwait(false); | ||||||
| @@ -613,7 +613,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService | |||||||
|                     TargetId = it.ApiRoute, |                     TargetId = it.ApiRoute, | ||||||
|                     Category = RelationCategoryEnum.UserHasPermission, |                     Category = RelationCategoryEnum.UserHasPermission, | ||||||
|                     ExtJson = new RelationPermission { ApiUrl = it.ApiRoute } |                     ExtJson = new RelationPermission { ApiUrl = it.ApiRoute } | ||||||
|                             .ToJsonNetString() |                             .ToSystemTextJsonString() | ||||||
|                 }); |                 }); | ||||||
|                 relationUsers.AddRange(relationUserPer);//合并列表 |                 relationUsers.AddRange(relationUserPer);//合并列表 | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -203,7 +203,7 @@ internal sealed class UserCenterService : BaseService<SysUser>, IUserCenterServi | |||||||
|     public async Task UpdateWorkbenchInfoAsync(WorkbenchInfo input) |     public async Task UpdateWorkbenchInfoAsync(WorkbenchInfo input) | ||||||
|     { |     { | ||||||
|         //关系表保存个人工作台 |         //关系表保存个人工作台 | ||||||
|         await _relationService.SaveRelationAsync(RelationCategoryEnum.UserWorkbenchData, input.Id, null, input.Shortcuts.ToJsonNetString(), |         await _relationService.SaveRelationAsync(RelationCategoryEnum.UserWorkbenchData, input.Id, null, input.Shortcuts.ToSystemTextJsonString(), | ||||||
|             true).ConfigureAwait(false); |             true).ConfigureAwait(false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,8 +13,6 @@ using BootstrapBlazor.Components; | |||||||
| using Microsoft.AspNetCore.Builder; | using Microsoft.AspNetCore.Builder; | ||||||
| using Microsoft.Extensions.DependencyInjection; | using Microsoft.Extensions.DependencyInjection; | ||||||
|  |  | ||||||
| using SqlSugar; |  | ||||||
|  |  | ||||||
| using System.Reflection; | using System.Reflection; | ||||||
|  |  | ||||||
| using ThingsGateway.UnifyResult; | using ThingsGateway.UnifyResult; | ||||||
| @@ -28,18 +26,12 @@ public class Startup : AppStartup | |||||||
|     { |     { | ||||||
|         Directory.CreateDirectory("DB"); |         Directory.CreateDirectory("DB"); | ||||||
|  |  | ||||||
|         services.AddConfigurableOptions<SqlSugarOptions>(); |  | ||||||
|         services.AddConfigurableOptions<AdminLogOptions>(); |         services.AddConfigurableOptions<AdminLogOptions>(); | ||||||
|         services.AddConfigurableOptions<TenantOptions>(); |         services.AddConfigurableOptions<TenantOptions>(); | ||||||
|  |  | ||||||
|         services.AddSingleton(typeof(IDataService<>), typeof(BaseService<>)); |         services.AddSingleton<IUserAgentService, UserAgentService>(); | ||||||
|         services.AddSingleton<ISugarAopService, SugarAopService>(); |  | ||||||
|         services.AddSingleton<ISugarConfigAopService, SugarConfigAopService>(); |  | ||||||
|  |  | ||||||
|         services.AddSingleton<IAppService, AppService>(); |         services.AddSingleton<IAppService, AppService>(); | ||||||
|  |  | ||||||
|         StaticConfig.EnableAllWhereIF = true; |  | ||||||
|  |  | ||||||
|         services.AddConfigurableOptions<EmailOptions>(); |         services.AddConfigurableOptions<EmailOptions>(); | ||||||
|         services.AddConfigurableOptions<HardwareInfoOptions>(); |         services.AddConfigurableOptions<HardwareInfoOptions>(); | ||||||
|  |  | ||||||
| @@ -56,7 +48,6 @@ public class Startup : AppStartup | |||||||
|  |  | ||||||
|         services.AddSingleton<IVerificatInfoService, VerificatInfoService>(); |         services.AddSingleton<IVerificatInfoService, VerificatInfoService>(); | ||||||
|         services.AddSingleton<IUserCenterService, UserCenterService>(); |         services.AddSingleton<IUserCenterService, UserCenterService>(); | ||||||
|         services.AddSingleton<ISugarAopService, SugarAopService>(); |  | ||||||
|         services.AddSingleton<ISysDictService, SysDictService>(); |         services.AddSingleton<ISysDictService, SysDictService>(); | ||||||
|         services.AddSingleton<ISysOperateLogService, SysOperateLogService>(); |         services.AddSingleton<ISysOperateLogService, SysOperateLogService>(); | ||||||
|         services.AddSingleton<IRelationService, RelationService>(); |         services.AddSingleton<IRelationService, RelationService>(); | ||||||
| @@ -97,6 +88,21 @@ public class Startup : AppStartup | |||||||
|         CodeFirstUtils.CodeFirst(fullName!);//CodeFirst |         CodeFirstUtils.CodeFirst(fullName!);//CodeFirst | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             using var db = DbContext.GetDB<SysOperateLog>(); | ||||||
|  |             if (db.CurrentConnectionConfig.DbType == SqlSugar.DbType.Sqlite) | ||||||
|  |             { | ||||||
|  |                 if (!db.DbMaintenance.IsAnyIndex("idx_operatelog_optime_date")) | ||||||
|  |                 { | ||||||
|  |                     var indexsql = "CREATE INDEX idx_operatelog_optime_date ON sys_operatelog(strftime('%Y-%m-%d', OpTime));"; | ||||||
|  |                     db.Ado.ExecuteCommand(indexsql); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         catch { } | ||||||
|  |  | ||||||
|  |  | ||||||
|         //删除在线用户统计 |         //删除在线用户统计 | ||||||
|         var verificatInfoService = App.RootServices.GetService<IVerificatInfoService>(); |         var verificatInfoService = App.RootServices.GetService<IVerificatInfoService>(); | ||||||
|         verificatInfoService.RemoveAllClientId(); |         verificatInfoService.RemoveAllClientId(); | ||||||
|   | |||||||
| @@ -18,10 +18,7 @@ | |||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
| 	 | 	 | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
| 		<PackageReference Include="BootstrapBlazor.TableExport" Version="9.2.4" /> |  | ||||||
| 		<PackageReference Include="UAParser" Version="3.1.47" /> |  | ||||||
| 		<PackageReference Include="Rougamo.Fody" Version="5.0.0" /> | 		<PackageReference Include="Rougamo.Fody" Version="5.0.0" /> | ||||||
| 		<PackageReference Include="SqlSugarCore" Version="5.1.4.193" /> |  | ||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
| 	<ItemGroup Condition=" '$(TargetFramework)' == 'net8.0' "> | 	<ItemGroup Condition=" '$(TargetFramework)' == 'net8.0' "> | ||||||
| 		<PackageReference Include="Microsoft.Extensions.Hosting.Abstractions" Version="8.0.1" /> | 		<PackageReference Include="Microsoft.Extensions.Hosting.Abstractions" Version="8.0.1" /> | ||||||
| @@ -50,6 +47,7 @@ | |||||||
|  |  | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
| 		<ProjectReference Include="..\ThingsGateway.Razor\ThingsGateway.Razor.csproj" /> | 		<ProjectReference Include="..\ThingsGateway.Razor\ThingsGateway.Razor.csproj" /> | ||||||
|  | 		<ProjectReference Include="..\ThingsGateway.SqlSugar\ThingsGateway.SqlSugar.csproj" /> | ||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
|   | |||||||
| @@ -0,0 +1,14 @@ | |||||||
|  | namespace ThingsGateway.Admin.Application | ||||||
|  | { | ||||||
|  |     /// <summary>Default interface for UserAgentService</summary> | ||||||
|  |     public interface IUserAgentService | ||||||
|  |     { | ||||||
|  |         /// <summary>Gets or sets the settings.</summary> | ||||||
|  |         public UserAgentSettings Settings { get; set; } | ||||||
|  |  | ||||||
|  |         /// <summary>Parses the specified user agent string.</summary> | ||||||
|  |         /// <param name="userAgentString">The user agent string.</param> | ||||||
|  |         /// <returns>An UserAgent object</returns> | ||||||
|  |         UserAgent? Parse(string userAgentString); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										145
									
								
								src/Admin/ThingsGateway.Admin.Application/UserAgent/UserAgent.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								src/Admin/ThingsGateway.Admin.Application/UserAgent/UserAgent.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | |||||||
|  | using System.Text.RegularExpressions; | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.Admin.Application | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Parsed UserAgent object | ||||||
|  |     /// </summary> | ||||||
|  |     public class UserAgent | ||||||
|  |     { | ||||||
|  |         private readonly UserAgentSettings settings; | ||||||
|  |  | ||||||
|  |         internal string Agent = ""; | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets a value indicating whether this UserAgent is a browser. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         ///   <c>true</c> if this UserAgent is a browser; otherwise, <c>false</c>. | ||||||
|  |         /// </value> | ||||||
|  |         public bool IsBrowser { get; set; } = false; | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets a value indicating whether this UserAgent is a robot. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         ///   <c>true</c> if this UserAgent is a robot; otherwise, <c>false</c>. | ||||||
|  |         /// </value> | ||||||
|  |         public bool IsRobot { get; set; } = false; | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets a value indicating whether this UserAgent is a mobile device. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         ///   <c>true</c> if this UserAgent is a mobile device; otherwise, <c>false</c>. | ||||||
|  |         /// </value> | ||||||
|  |         public bool IsMobile { get; set; } = false; | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the platform. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         /// The platform or operating system. | ||||||
|  |         /// </value> | ||||||
|  |         public string Platform { get; set; } = ""; | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the browser. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         /// The browser. | ||||||
|  |         /// </value> | ||||||
|  |         public string Browser { get; set; } = ""; | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the browser version. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         /// The browser version. | ||||||
|  |         /// </value> | ||||||
|  |         public string BrowserVersion { get; set; } = ""; | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the mobile device. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         /// The mobile device. | ||||||
|  |         /// </value> | ||||||
|  |         public string Mobile { get; set; } = ""; | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the robot. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value> | ||||||
|  |         /// The robot. | ||||||
|  |         /// </value> | ||||||
|  |         public string Robot { get; set; } = ""; | ||||||
|  |  | ||||||
|  |         internal UserAgent(UserAgentSettings settings, string? userAgentString = null) | ||||||
|  |         { | ||||||
|  |             this.settings = settings; | ||||||
|  |  | ||||||
|  |             if (userAgentString != null) | ||||||
|  |             { | ||||||
|  |                 Agent = userAgentString.Trim(); | ||||||
|  |                 SetPlatform(); | ||||||
|  |                 if (SetRobot()) return; | ||||||
|  |                 if (SetBrowser()) return; | ||||||
|  |                 if (SetMobile()) return; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         internal bool SetPlatform() | ||||||
|  |         { | ||||||
|  |             foreach (var item in settings.Platforms) | ||||||
|  |             { | ||||||
|  |                 if (Regex.IsMatch(Agent, $"{Regex.Escape(item.Key)}", RegexOptions.IgnoreCase)) | ||||||
|  |                 { | ||||||
|  |                     Platform = item.Value; | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             Platform = "Unknown Platform"; | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         internal bool SetBrowser() | ||||||
|  |         { | ||||||
|  |             foreach (var item in settings.Browsers) | ||||||
|  |             { | ||||||
|  |                 var match = Regex.Match(Agent, $@"{item.Key}.*?([0-9\.]+)", RegexOptions.IgnoreCase); | ||||||
|  |                 if (match.Success) | ||||||
|  |                 { | ||||||
|  |                     IsBrowser = true; | ||||||
|  |                     BrowserVersion = match.Groups[1].Value; | ||||||
|  |                     Browser = item.Value; | ||||||
|  |                     SetMobile(); | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         internal bool SetRobot() | ||||||
|  |         { | ||||||
|  |             foreach (var item in settings.Robots) | ||||||
|  |             { | ||||||
|  |                 if (Regex.IsMatch(Agent, $"{Regex.Escape(item.Key)}", RegexOptions.IgnoreCase)) | ||||||
|  |                 { | ||||||
|  |                     IsRobot = true; | ||||||
|  |                     Robot = item.Value; | ||||||
|  |                     SetMobile(); | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         internal bool SetMobile() | ||||||
|  |         { | ||||||
|  |             foreach (var item in settings.Mobiles) | ||||||
|  |             { | ||||||
|  |                 if (Agent?.IndexOf(item.Key, StringComparison.OrdinalIgnoreCase) != -1) | ||||||
|  |                 { | ||||||
|  |                     IsMobile = true; | ||||||
|  |                     Mobile = item.Value; | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | using ThingsGateway.NewLife.Caching; | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.Admin.Application | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// The UserAgent service | ||||||
|  |     /// </summary> | ||||||
|  |     /// <seealso cref="ThingsGateway.Admin.Application.IUserAgentService" /> | ||||||
|  |     public class UserAgentService : IUserAgentService | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the settings. | ||||||
|  |         /// </summary> | ||||||
|  |         public UserAgentSettings Settings { get; set; } | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="UserAgentService"/> class. | ||||||
|  |         /// </summary> | ||||||
|  |         public UserAgentService() | ||||||
|  |         { | ||||||
|  |             Settings = new UserAgentSettings(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private MemoryCache MemoryCache { get; set; } = new(); | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Parses the specified user agent string. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="userAgentString">The user agent string.</param> | ||||||
|  |         /// <returns> | ||||||
|  |         /// An UserAgent object | ||||||
|  |         /// </returns> | ||||||
|  |         public UserAgent? Parse(string? userAgentString) | ||||||
|  |         { | ||||||
|  |             userAgentString = ((userAgentString?.Length > Settings.UaStringSizeLimit) ? userAgentString?.Trim().Substring(0, Settings.UaStringSizeLimit) : userAgentString?.Trim()) ?? ""; | ||||||
|  |             return MemoryCache.GetOrAdd(userAgentString, entry => | ||||||
|  |             { | ||||||
|  |                 return new UserAgent(Settings, userAgentString); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,214 @@ | |||||||
|  | namespace ThingsGateway.Admin.Application | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// UserAgent settings container. | ||||||
|  |     /// </summary> | ||||||
|  |     public class UserAgentSettings | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the maximum size of the useragent string. Limiting the length of the useragent string protects from hackers sending in extremely long user agent strings. | ||||||
|  |         /// </summary> | ||||||
|  |         public int UaStringSizeLimit { get; set; } = 512; | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets a dictionary containing mappings for platforms. | ||||||
|  |         /// </summary> | ||||||
|  |         public Dictionary<string, string> Platforms { get; } = new() | ||||||
|  |         { | ||||||
|  |             {"windows nt 10.0", "Windows 10"}, | ||||||
|  |             {"windows nt 6.3", "Windows 8.1"}, | ||||||
|  |             {"windows nt 6.2", "Windows 8"}, | ||||||
|  |             {"windows nt 6.1", "Windows 7"}, | ||||||
|  |             {"windows nt 6.0", "Windows Vista"}, | ||||||
|  |             {"windows nt 5.2", "Windows 2003"}, | ||||||
|  |             {"windows nt 5.1", "Windows XP"}, | ||||||
|  |             {"windows nt 5.0", "Windows 2000"}, | ||||||
|  |             {"windows nt 4.0", "Windows NT 4.0"}, | ||||||
|  |             {"winnt4.0", "Windows NT 4.0"}, | ||||||
|  |             {"winnt 4.0", "Windows NT"}, | ||||||
|  |             {"winnt", "Windows NT"}, | ||||||
|  |             {"windows 98", "Windows 98"}, | ||||||
|  |             {"win98", "Windows 98"}, | ||||||
|  |             {"windows 95", "Windows 95"}, | ||||||
|  |             {"win95", "Windows 95"}, | ||||||
|  |             {"windows phone", "Windows Phone"}, | ||||||
|  |             {"windows", "Unknown Windows OS"}, | ||||||
|  |             {"android", "Android"}, | ||||||
|  |             {"blackberry", "BlackBerry"}, | ||||||
|  |             {"iphone", "iOS"}, | ||||||
|  |             {"ipad", "iOS"}, | ||||||
|  |             {"ipod", "iOS"}, | ||||||
|  |             {"os x", "Mac OS X"}, | ||||||
|  |             {"ppc mac", "Power PC Mac"}, | ||||||
|  |             {"freebsd", "FreeBSD"}, | ||||||
|  |             {"ppc", "Macintosh"}, | ||||||
|  |             {"linux", "Linux"}, | ||||||
|  |             {"debian", "Debian"}, | ||||||
|  |             {"sunos", "Sun Solaris"}, | ||||||
|  |             {"beos", "BeOS"}, | ||||||
|  |             {"apachebench", "ApacheBench"}, | ||||||
|  |             {"aix", "AIX"}, | ||||||
|  |             {"irix", "Irix"}, | ||||||
|  |             {"osf", "DEC OSF"}, | ||||||
|  |             {"hp-ux", "HP-UX"}, | ||||||
|  |             {"netbsd", "NetBSD"}, | ||||||
|  |             {"bsdi", "BSDi"}, | ||||||
|  |             {"openbsd", "OpenBSD"}, | ||||||
|  |             {"gnu", "GNU/Linux"}, | ||||||
|  |             {"unix", "Unknown Unix OS"}, | ||||||
|  |             {"symbian", "Symbian OS"}, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets a dictionary containing mappings for browsers. | ||||||
|  |         /// </summary> | ||||||
|  |         public Dictionary<string, string> Browsers { get; } = new() | ||||||
|  |         { | ||||||
|  |             {"Microsoft Outlook", "Microsoft Outlook"}, | ||||||
|  |             {"OPR", "Opera"}, | ||||||
|  |             {"Flock", "Flock"}, | ||||||
|  |             {"Edge", "Edge"}, | ||||||
|  |             {"Edg", "Edge"}, | ||||||
|  |             {"Chrome", "Chrome"}, | ||||||
|  |             {"Opera.*?Version", "Opera"}, | ||||||
|  |             {"Opera", "Opera"}, | ||||||
|  |             {"MSIE", "Internet Explorer"}, | ||||||
|  |             {"Internet Explorer", "Internet Explorer"}, | ||||||
|  |             {"Trident.* rv" , "Internet Explorer"}, | ||||||
|  |             {"Shiira", "Shiira"}, | ||||||
|  |             {"Firefox", "Firefox"}, | ||||||
|  |             {"Chimera", "Chimera"}, | ||||||
|  |             {"Phoenix", "Phoenix"}, | ||||||
|  |             {"Firebird", "Firebird"}, | ||||||
|  |             {"Camino", "Camino"}, | ||||||
|  |             {"Netscape", "Netscape"}, | ||||||
|  |             {"OmniWeb", "OmniWeb"}, | ||||||
|  |             {"Safari", "Safari"}, | ||||||
|  |             {"Mozilla", "Mozilla"}, | ||||||
|  |             {"Konqueror", "Konqueror"}, | ||||||
|  |             {"icab", "iCab"}, | ||||||
|  |             {"Lynx", "Lynx"}, | ||||||
|  |             {"Links", "Links"}, | ||||||
|  |             {"hotjava", "HotJava"}, | ||||||
|  |             {"amaya", "Amaya"}, | ||||||
|  |             {"IBrowse", "IBrowse"}, | ||||||
|  |             {"Maxthon", "Maxthon"}, | ||||||
|  |             {"Ubuntu", "Ubuntu Web Browser"}, | ||||||
|  |             {"Vivaldi", "Vivaldi"}, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets a dictionary containing mappings for mobiles. | ||||||
|  |         /// </summary> | ||||||
|  |         public Dictionary<string, string> Mobiles { get; } = new() | ||||||
|  |         { | ||||||
|  |             // Legacy | ||||||
|  |             {"mobileexplorer", "Mobile Explorer"}, | ||||||
|  |             {"palmsource", "Palm"}, | ||||||
|  |             {"palmscape", "Palmscape"}, | ||||||
|  |             // Phones and Manufacturers | ||||||
|  |             {"motorola", "Motorola"}, | ||||||
|  |             {"nokia", "Nokia"}, | ||||||
|  |             {"palm", "Palm"}, | ||||||
|  |             {"iphone", "Apple iPhone"}, | ||||||
|  |             {"ipad", "iPad"}, | ||||||
|  |             {"ipod", "Apple iPod Touch"}, | ||||||
|  |             {"sony", "Sony Ericsson"}, | ||||||
|  |             {"ericsson", "Sony Ericsson"}, | ||||||
|  |             {"blackberry", "BlackBerry"}, | ||||||
|  |             {"cocoon", "O2 Cocoon"}, | ||||||
|  |             {"blazer", "Treo"}, | ||||||
|  |             {"lg", "LG"}, | ||||||
|  |             {"amoi", "Amoi"}, | ||||||
|  |             {"xda", "XDA"}, | ||||||
|  |             {"mda", "MDA"}, | ||||||
|  |             {"vario", "Vario"}, | ||||||
|  |             {"htc", "HTC"}, | ||||||
|  |             {"samsung", "Samsung"}, | ||||||
|  |             {"sharp", "Sharp"}, | ||||||
|  |             {"sie-", "Siemens"}, | ||||||
|  |             {"alcatel", "Alcatel"}, | ||||||
|  |             {"benq", "BenQ"}, | ||||||
|  |             {"ipaq", "HP iPaq"}, | ||||||
|  |             {"mot-", "Motorola"}, | ||||||
|  |             {"playstation portable", "PlayStation Portable"}, | ||||||
|  |             {"playstation 3", "PlayStation 3"}, | ||||||
|  |             {"playstation vita", "PlayStation Vita"}, | ||||||
|  |             {"hiptop", "Danger Hiptop"}, | ||||||
|  |             {"nec-", "NEC"}, | ||||||
|  |             {"panasonic", "Panasonic"}, | ||||||
|  |             {"philips", "Philips"}, | ||||||
|  |             {"sagem", "Sagem"}, | ||||||
|  |             {"sanyo", "Sanyo"}, | ||||||
|  |             {"spv", "SPV"}, | ||||||
|  |             {"zte", "ZTE"}, | ||||||
|  |             {"sendo", "Sendo"}, | ||||||
|  |             {"nintendo dsi", "Nintendo DSi"}, | ||||||
|  |             {"nintendo ds", "Nintendo DS"}, | ||||||
|  |             {"nintendo 3ds", "Nintendo 3DS"}, | ||||||
|  |             {"wii", "Nintendo Wii"}, | ||||||
|  |             {"open web", "Open Web"}, | ||||||
|  |             {"openweb", "OpenWeb"}, | ||||||
|  |             // Operating Systems | ||||||
|  |             {"android", "Android"}, | ||||||
|  |             {"symbian", "Symbian"}, | ||||||
|  |             {"SymbianOS", "SymbianOS"}, | ||||||
|  |             {"elaine", "Palm"}, | ||||||
|  |             {"series60", "Symbian S60"}, | ||||||
|  |             {"windows ce", "Windows CE"}, | ||||||
|  |             // Browsers | ||||||
|  |             {"obigo", "Obigo"}, | ||||||
|  |             {"netfront", "Netfront Browser"}, | ||||||
|  |             {"openwave", "Openwave Browser"}, | ||||||
|  |             {"mobilexplorer", "Mobile Explorer"}, | ||||||
|  |             {"operamini", "Opera Mini"}, | ||||||
|  |             {"opera mini", "Opera Mini"}, | ||||||
|  |             {"opera mobi", "Opera Mobile"}, | ||||||
|  |             {"fennec", "Firefox Mobile"}, | ||||||
|  |             // Other | ||||||
|  |             {"digital paths", "Digital Paths"}, | ||||||
|  |             {"avantgo", "AvantGo"}, | ||||||
|  |             {"xiino", "Xiino"}, | ||||||
|  |             {"novarra", "Novarra Transcoder"}, | ||||||
|  |             {"vodafone", "Vodafone"}, | ||||||
|  |             {"docomo", "NTT DoCoMo"}, | ||||||
|  |             {"o2", "O2"}, | ||||||
|  |             // Fallback | ||||||
|  |             {"mobile", "Generic Mobile"}, | ||||||
|  |             {"wireless", "Generic Mobile"}, | ||||||
|  |             {"j2me", "Generic Mobile"}, | ||||||
|  |             {"midp", "Generic Mobile"}, | ||||||
|  |             {"cldc", "Generic Mobile"}, | ||||||
|  |             {"up.link", "Generic Mobile"}, | ||||||
|  |             {"up.browser", "Generic Mobile"}, | ||||||
|  |             {"smartphone", "Generic Mobile"}, | ||||||
|  |             {"cellphone", "Generic Mobile"}, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets a dictionary containing mappings for robots. | ||||||
|  |         /// </summary> | ||||||
|  |         public Dictionary<string, string> Robots { get; } = new() | ||||||
|  |         { | ||||||
|  |             {"googlebot", "Googlebot"}, | ||||||
|  |             {"msnbot", "MSNBot"}, | ||||||
|  |             {"baiduspider", "Baiduspider"}, | ||||||
|  |             {"bingbot", "Bing"}, | ||||||
|  |             {"slurp", "Inktomi Slurp"}, | ||||||
|  |             {"yahoo", "Yahoo"}, | ||||||
|  |             {"ask jeeves", "Ask Jeeves"}, | ||||||
|  |             {"fastcrawler", "FastCrawler"}, | ||||||
|  |             {"infoseek", "InfoSeek Robot 1.0"}, | ||||||
|  |             {"lycos", "Lycos"}, | ||||||
|  |             {"yandex", "YandexBot"}, | ||||||
|  |             {"mediapartners-google", "MediaPartners Google"}, | ||||||
|  |             {"CRAZYWEBCRAWLER", "Crazy Webcrawler"}, | ||||||
|  |             {"adsbot-google", "AdsBot Google"}, | ||||||
|  |             {"feedfetcher-google", "Feedfetcher Google"}, | ||||||
|  |             {"curious george", "Curious George"}, | ||||||
|  |             {"ia_archiver", "Alexa Crawler"}, | ||||||
|  |             {"MJ12bot", "Majestic-12"}, | ||||||
|  |             {"Uptimebot", "Uptimebot"}, | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -97,7 +97,7 @@ public class BlazorAppContext | |||||||
|             AllResource = sysResources; |             AllResource = sysResources; | ||||||
|             var ids = CurrentUser.ModuleList.Select(a => a.Id).ToHashSet(); |             var ids = CurrentUser.ModuleList.Select(a => a.Id).ToHashSet(); | ||||||
|             CurrentUser.ModuleList = AllResource.Where(a => ids.Contains(a.Id)).OrderBy(a => a.SortCode).ToList(); |             CurrentUser.ModuleList = AllResource.Where(a => ids.Contains(a.Id)).OrderBy(a => a.SortCode).ToList(); | ||||||
|             AllMenus = sysResources.Where(a => a.Category == ResourceCategoryEnum.Menu); |             AllMenus = AllResource.Where(a => a.Category == ResourceCategoryEnum.Menu); | ||||||
|  |  | ||||||
|             if (moduleId == null) |             if (moduleId == null) | ||||||
|             { |             { | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ | |||||||
|  |  | ||||||
|     // nuget动态加载的程序集 |     // nuget动态加载的程序集 | ||||||
|     "SupportPackageNamePrefixs": [ |     "SupportPackageNamePrefixs": [ | ||||||
|  |       "ThingsGateway.SqlSugar", | ||||||
|       "ThingsGateway.Admin.Application", |       "ThingsGateway.Admin.Application", | ||||||
|       "ThingsGateway.Admin.Razor", |       "ThingsGateway.Admin.Razor", | ||||||
|       "ThingsGateway.Razor" |       "ThingsGateway.Razor" | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ | |||||||
|  |  | ||||||
|     // nuget动态加载的程序集 |     // nuget动态加载的程序集 | ||||||
|     "SupportPackageNamePrefixs": [ |     "SupportPackageNamePrefixs": [ | ||||||
|  |       "ThingsGateway.SqlSugar", | ||||||
|       "ThingsGateway.Admin.Application", |       "ThingsGateway.Admin.Application", | ||||||
|       "ThingsGateway.Admin.Razor", |       "ThingsGateway.Admin.Razor", | ||||||
|       "ThingsGateway.Razor" |       "ThingsGateway.Razor" | ||||||
|   | |||||||
| @@ -39,19 +39,4 @@ | |||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| <div class="row g-2 mx-1 form-inline"> |  | ||||||
|     <div class="col-12  col-md-12"> |  | ||||||
|         <Card IsShadow=true class="m-2 flex-fill" Color="Color.Primary"> |  | ||||||
|             <HeaderTemplate> |  | ||||||
|                 @Localizer["HardwareInfoChart"] |  | ||||||
|             </HeaderTemplate> |  | ||||||
|  |  | ||||||
|             <BodyTemplate> |  | ||||||
|                 <Chart @ref=CPULineChart OnInitAsync="OnCPUInit" Height="var(--line-chart-height)" Width="100%" OnAfterInitAsync="()=>{chartInit=true;return Task.CompletedTask;}" /> |  | ||||||
|             </BodyTemplate> |  | ||||||
|         </Card> |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -18,8 +18,6 @@ using Microsoft.AspNetCore.Authorization; | |||||||
| using Microsoft.AspNetCore.Components; | using Microsoft.AspNetCore.Components; | ||||||
| using Microsoft.Extensions.Localization; | using Microsoft.Extensions.Localization; | ||||||
|  |  | ||||||
| using System.Diagnostics.CodeAnalysis; |  | ||||||
|  |  | ||||||
| using ThingsGateway.Admin.Application; | using ThingsGateway.Admin.Application; | ||||||
| using ThingsGateway.Admin.Razor; | using ThingsGateway.Admin.Razor; | ||||||
| using ThingsGateway.Extension; | using ThingsGateway.Extension; | ||||||
| @@ -31,118 +29,8 @@ namespace ThingsGateway.AdminServer; | |||||||
| [IgnoreRolePermission] | [IgnoreRolePermission] | ||||||
| [Route("/")] | [Route("/")] | ||||||
| [TabItemOption(Text = "Home", Icon = "fas fa-house")] | [TabItemOption(Text = "Home", Icon = "fas fa-house")] | ||||||
| public partial class AdminIndex : IDisposable | public partial class AdminIndex | ||||||
| { | { | ||||||
|     [Inject] |  | ||||||
|     private IHardwareJob HardwareJob { get; set; } |  | ||||||
|  |  | ||||||
|     protected override void OnInitialized() |  | ||||||
|     { |  | ||||||
|         _ = RunTimerAsync(); |  | ||||||
|         base.OnInitialized(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public bool Disposed { get; set; } |  | ||||||
|  |  | ||||||
|     public void Dispose() |  | ||||||
|     { |  | ||||||
|         Disposed = true; |  | ||||||
|         GC.SuppressFinalize(this); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async Task RunTimerAsync() |  | ||||||
|     { |  | ||||||
|         while (!Disposed) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 if (chartInit) |  | ||||||
|                     await CPULineChart.Update(ChartAction.Update); |  | ||||||
|  |  | ||||||
|                 await InvokeAsync(StateHasChanged); |  | ||||||
|                 await Task.Delay(30000); |  | ||||||
|             } |  | ||||||
|             catch (Exception ex) |  | ||||||
|             { |  | ||||||
|                 NewLife.Log.XTrace.WriteException(ex); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     #region 曲线 |  | ||||||
|  |  | ||||||
|     private bool chartInit { get; set; } |  | ||||||
|     private Chart CPULineChart { get; set; } |  | ||||||
|     private ChartDataSource? ChartDataSource { get; set; } |  | ||||||
|  |  | ||||||
|     [Inject] |  | ||||||
|     [NotNull] |  | ||||||
|     private IStringLocalizer<HistoryHardwareInfo> HistoryHardwareInfoLocalizer { get; set; } |  | ||||||
|  |  | ||||||
|     private async Task<ChartDataSource> OnCPUInit() |  | ||||||
|     { |  | ||||||
|         if (ChartDataSource == null) |  | ||||||
|         { |  | ||||||
|             var hisHardwareInfos = await HardwareJob.GetHistoryHardwareInfos(); |  | ||||||
|             ChartDataSource = new ChartDataSource(); |  | ||||||
|             ChartDataSource.Options.Title = Localizer[nameof(HistoryHardwareInfo)]; |  | ||||||
|             ChartDataSource.Options.X.Title = Localizer["DateTime"]; |  | ||||||
|             ChartDataSource.Options.Y.Title = Localizer["Data"]; |  | ||||||
|             ChartDataSource.Labels = hisHardwareInfos.Select(a => a.Date.ToString("dd HH:mm zz")); |  | ||||||
|             ChartDataSource.Data.Add(new ChartDataset() |  | ||||||
|             { |  | ||||||
|                 Tension = 0.4f, |  | ||||||
|                 PointRadius = 1, |  | ||||||
|                 Label = HistoryHardwareInfoLocalizer[nameof(HistoryHardwareInfo.CpuUsage)], |  | ||||||
|                 Data = hisHardwareInfos.Select(a => (object)a.CpuUsage), |  | ||||||
|             }); |  | ||||||
|             ChartDataSource.Data.Add(new ChartDataset() |  | ||||||
|             { |  | ||||||
|                 Tension = 0.4f, |  | ||||||
|                 PointRadius = 1, |  | ||||||
|                 Label = HistoryHardwareInfoLocalizer[nameof(HistoryHardwareInfo.MemoryUsage)], |  | ||||||
|                 Data = hisHardwareInfos.Select(a => (object)a.MemoryUsage), |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             ChartDataSource.Data.Add(new ChartDataset() |  | ||||||
|             { |  | ||||||
|                 Tension = 0.4f, |  | ||||||
|                 PointRadius = 1, |  | ||||||
|                 Label = HistoryHardwareInfoLocalizer[nameof(HistoryHardwareInfo.DriveUsage)], |  | ||||||
|                 Data = hisHardwareInfos.Select(a => (object)a.DriveUsage), |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             ChartDataSource.Data.Add(new ChartDataset() |  | ||||||
|             { |  | ||||||
|                 ShowPointStyle = false, |  | ||||||
|                 Tension = 0.4f, |  | ||||||
|                 PointRadius = 1, |  | ||||||
|                 Label = HistoryHardwareInfoLocalizer[nameof(HistoryHardwareInfo.Temperature)], |  | ||||||
|                 Data = hisHardwareInfos.Select(a => (object)a.Temperature), |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             ChartDataSource.Data.Add(new ChartDataset() |  | ||||||
|             { |  | ||||||
|                 Tension = 0.4f, |  | ||||||
|                 PointRadius = 1, |  | ||||||
|                 Label = HistoryHardwareInfoLocalizer[nameof(HistoryHardwareInfo.Battery)], |  | ||||||
|                 Data = hisHardwareInfos.Select(a => (object)a.Battery), |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             var hisHardwareInfos = await HardwareJob.GetHistoryHardwareInfos(); |  | ||||||
|             ChartDataSource.Labels = hisHardwareInfos.Select(a => a.Date.ToString("dd HH:mm zz")); |  | ||||||
|             ChartDataSource.Data[0].Data = hisHardwareInfos.Select(a => (object)a.CpuUsage); |  | ||||||
|             ChartDataSource.Data[1].Data = hisHardwareInfos.Select(a => (object)a.MemoryUsage); |  | ||||||
|             ChartDataSource.Data[2].Data = hisHardwareInfos.Select(a => (object)a.DriveUsage); |  | ||||||
|             ChartDataSource.Data[3].Data = hisHardwareInfos.Select(a => (object)a.Temperature); |  | ||||||
|             ChartDataSource.Data[4].Data = hisHardwareInfos.Select(a => (object)a.Battery); |  | ||||||
|         } |  | ||||||
|         return ChartDataSource; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     #endregion 曲线 |  | ||||||
|  |  | ||||||
|     [Inject] |     [Inject] | ||||||
|     private BlazorAppContext AppContext { get; set; } |     private BlazorAppContext AppContext { get; set; } | ||||||
|   | |||||||
| @@ -40,7 +40,8 @@ public class SingleFilePublish : ISingleFilePublish | |||||||
|             "ThingsGateway.NewLife.X", |             "ThingsGateway.NewLife.X", | ||||||
|             "ThingsGateway.Razor", |             "ThingsGateway.Razor", | ||||||
|             "ThingsGateway.Admin.Razor"   , |             "ThingsGateway.Admin.Razor"   , | ||||||
|             "ThingsGateway.Admin.Application" |             "ThingsGateway.Admin.Application", | ||||||
|  |             "ThingsGateway.SqlSugar", | ||||||
|         ]; |         ]; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -14,7 +14,6 @@ using Microsoft.AspNetCore.DataProtection; | |||||||
| using Microsoft.AspNetCore.DataProtection.AuthenticatedEncryption; | using Microsoft.AspNetCore.DataProtection.AuthenticatedEncryption; | ||||||
| using Microsoft.AspNetCore.DataProtection.AuthenticatedEncryption.ConfigurationModel; | using Microsoft.AspNetCore.DataProtection.AuthenticatedEncryption.ConfigurationModel; | ||||||
| using Microsoft.AspNetCore.HttpOverrides; | using Microsoft.AspNetCore.HttpOverrides; | ||||||
| using Microsoft.AspNetCore.Mvc.Controllers; |  | ||||||
| using Microsoft.AspNetCore.StaticFiles; | using Microsoft.AspNetCore.StaticFiles; | ||||||
| using Microsoft.Extensions.Localization; | using Microsoft.Extensions.Localization; | ||||||
| using Microsoft.Extensions.Options; | using Microsoft.Extensions.Options; | ||||||
| @@ -29,7 +28,6 @@ using System.Text.Unicode; | |||||||
| using ThingsGateway.Admin.Application; | using ThingsGateway.Admin.Application; | ||||||
| using ThingsGateway.Admin.Razor; | using ThingsGateway.Admin.Razor; | ||||||
| using ThingsGateway.Extension; | using ThingsGateway.Extension; | ||||||
| using ThingsGateway.Logging; |  | ||||||
| using ThingsGateway.NewLife.Caching; | using ThingsGateway.NewLife.Caching; | ||||||
|  |  | ||||||
| namespace ThingsGateway.AdminServer; | namespace ThingsGateway.AdminServer; | ||||||
| @@ -89,6 +87,7 @@ public class Startup : AppStartup | |||||||
|         } |         } | ||||||
|         ; |         ; | ||||||
|  |  | ||||||
|  |         services.AddMvcFilter<RequestAuditFilter>(); | ||||||
|         services.AddControllers() |         services.AddControllers() | ||||||
|             .AddNewtonsoftJson(options => SetNewtonsoftJsonSetting(options.SerializerSettings)) |             .AddNewtonsoftJson(options => SetNewtonsoftJsonSetting(options.SerializerSettings)) | ||||||
|             //.AddXmlSerializerFormatters() |             //.AddXmlSerializerFormatters() | ||||||
| @@ -161,7 +160,9 @@ public class Startup : AppStartup | |||||||
|         { |         { | ||||||
|             options.WriteFilter = (logMsg) => |             options.WriteFilter = (logMsg) => | ||||||
|             { |             { | ||||||
|                 return true; |                 if (App.HostApplicationLifetime.ApplicationStopping.IsCancellationRequested && logMsg.LogLevel >= LogLevel.Warning) return false; | ||||||
|  |                 if (string.IsNullOrEmpty(logMsg.Message)) return false; | ||||||
|  |                 else return true; | ||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             options.MessageFormat = (logMsg) => |             options.MessageFormat = (logMsg) => | ||||||
| @@ -211,39 +212,39 @@ public class Startup : AppStartup | |||||||
|         #region api日志 |         #region api日志 | ||||||
|  |  | ||||||
|         //Monitor日志配置 |         //Monitor日志配置 | ||||||
|         services.AddMonitorLogging(options => |         //services.AddMonitorLogging(options => | ||||||
|         { |         //{ | ||||||
|             options.JsonIndented = true;// 是否美化 JSON |         //    options.JsonIndented = true;// 是否美化 JSON | ||||||
|             options.GlobalEnabled = false;//全局启用 |         //    options.GlobalEnabled = false;//全局启用 | ||||||
|             options.ConfigureLogger((logger, logContext, context) => |         //    options.ConfigureLogger((logger, logContext, context) => | ||||||
|             { |         //    { | ||||||
|                 var httpContext = context.HttpContext;//获取httpContext |         //        var httpContext = context.HttpContext;//获取httpContext | ||||||
|  |  | ||||||
|                 //获取客户端信息 |         //        //获取客户端信息 | ||||||
|                 var client = App.GetService<IAppService>().ClientInfo; |         //        var client = App.GetService<IAppService>().UserAgent; | ||||||
|                 // 获取控制器/操作描述器 |         //        // 获取控制器/操作描述器 | ||||||
|                 var controllerActionDescriptor = context.ActionDescriptor as ControllerActionDescriptor; |         //        var controllerActionDescriptor = context.ActionDescriptor as ControllerActionDescriptor; | ||||||
|                 //操作名称默认是控制器名加方法名,自定义操作名称要在action上加Description特性 |         //        //操作名称默认是控制器名加方法名,自定义操作名称要在action上加Description特性 | ||||||
|                 var option = $"{controllerActionDescriptor.ControllerName}/{controllerActionDescriptor.ActionName}"; |         //        var option = $"{controllerActionDescriptor.ControllerName}/{controllerActionDescriptor.ActionName}"; | ||||||
|  |  | ||||||
|                 var desc = App.CreateLocalizerByType(controllerActionDescriptor.ControllerTypeInfo.AsType())[controllerActionDescriptor.MethodInfo.Name]; |         //        var desc = App.CreateLocalizerByType(controllerActionDescriptor.ControllerTypeInfo.AsType())[controllerActionDescriptor.MethodInfo.Name]; | ||||||
|                 //获取特性 |         //        //获取特性 | ||||||
|                 option = desc.Value;//则将操作名称赋值为控制器上写的title |         //        option = desc.Value;//则将操作名称赋值为控制器上写的title | ||||||
|  |  | ||||||
|                 logContext.Set(LoggingConst.CateGory, option);//传操作名称 |         //        logContext.Set(LoggingConst.CateGory, option);//传操作名称 | ||||||
|                 logContext.Set(LoggingConst.Operation, option);//传操作名称 |         //        logContext.Set(LoggingConst.Operation, option);//传操作名称 | ||||||
|                 logContext.Set(LoggingConst.Client, client);//客户端信息 |         //        logContext.Set(LoggingConst.Client, client);//客户端信息 | ||||||
|                 logContext.Set(LoggingConst.Path, httpContext.Request.Path.Value);//请求地址 |         //        logContext.Set(LoggingConst.Path, httpContext.Request.Path.Value);//请求地址 | ||||||
|                 logContext.Set(LoggingConst.Method, httpContext.Request.Method);//请求方法 |         //        logContext.Set(LoggingConst.Method, httpContext.Request.Method);//请求方法 | ||||||
|             }); |         //    }); | ||||||
|         }); |         //}); | ||||||
|  |  | ||||||
|         //日志写入数据库配置 |         //日志写入数据库配置 | ||||||
|         services.AddDatabaseLogging<DatabaseLoggingWriter>(options => |         services.AddDatabaseLogging<DatabaseLoggingWriter>(options => | ||||||
|         { |         { | ||||||
|             options.WriteFilter = (logMsg) => |             options.WriteFilter = (logMsg) => | ||||||
|             { |             { | ||||||
|                 return logMsg.LogName == "System.Logging.LoggingMonitor";//只写入LoggingMonitor日志 |                 return logMsg.LogName == "System.Logging.RequestAudit"; | ||||||
|             }; |             }; | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
| @@ -302,7 +303,7 @@ public class Startup : AppStartup | |||||||
|         var certificate = new X509Certificate2("ThingsGateway.pfx", "ThingsGateway", X509KeyStorageFlags.EphemeralKeySet); |         var certificate = new X509Certificate2("ThingsGateway.pfx", "ThingsGateway", X509KeyStorageFlags.EphemeralKeySet); | ||||||
| #endif | #endif | ||||||
|         services.AddDataProtection() |         services.AddDataProtection() | ||||||
|             .PersistKeysToFileSystem(new DirectoryInfo("../keys")) |             .PersistKeysToFileSystem(new DirectoryInfo("keys")) | ||||||
|             .ProtectKeysWithCertificate(certificate) |             .ProtectKeysWithCertificate(certificate) | ||||||
|             .UseCryptographicAlgorithms(new AuthenticatedEncryptorConfiguration |             .UseCryptographicAlgorithms(new AuthenticatedEncryptorConfiguration | ||||||
|             { |             { | ||||||
|   | |||||||
| @@ -71,13 +71,25 @@ public static class App | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     public static IServiceProvider RootServices => InternalApp.RootServices; |     public static IServiceProvider RootServices => InternalApp.RootServices; | ||||||
|  |  | ||||||
|  |     private static IHostApplicationLifetime hostApplicationLifetime; | ||||||
|  |     public static IHostApplicationLifetime HostApplicationLifetime | ||||||
|  |     { | ||||||
|  |         get | ||||||
|  |         { | ||||||
|  |             if ((hostApplicationLifetime == null)) | ||||||
|  |             { | ||||||
|  |                 hostApplicationLifetime = RootServices?.GetService<IHostApplicationLifetime>(); | ||||||
|  |             } | ||||||
|  |             return hostApplicationLifetime; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private static IStringLocalizerFactory? stringLocalizerFactory; |     private static IStringLocalizerFactory? stringLocalizerFactory; | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 本地化服务工厂 |     /// 本地化服务工厂 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public static IStringLocalizerFactory? StringLocalizerFactory |     public static IStringLocalizerFactory? StringLocalizerFactory | ||||||
|  |  | ||||||
|     { |     { | ||||||
|         get |         get | ||||||
|         { |         { | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ public static class ILoggerExtensions | |||||||
|     /// <param name="logger"></param> |     /// <param name="logger"></param> | ||||||
|     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> |     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public static IDisposable ScopeContext(this ILogger logger, IDictionary<object, object> properties) |     public static IDisposable ScopeContext(this ILogger logger, IDictionary<string, object> properties) | ||||||
|     { |     { | ||||||
|         if (logger == null) throw new ArgumentNullException(nameof(logger)); |         if (logger == null) throw new ArgumentNullException(nameof(logger)); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -26,11 +26,11 @@ public static class LogContextExtensions | |||||||
|     /// <param name="key">键</param> |     /// <param name="key">键</param> | ||||||
|     /// <param name="value">值</param> |     /// <param name="value">值</param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public static LogContext Set(this LogContext logContext, object key, object value) |     public static LogContext Set(this LogContext logContext, string key, object value) | ||||||
|     { |     { | ||||||
|         if (logContext == null || key == null) return logContext; |         if (logContext == null || key == null) return logContext; | ||||||
|  |  | ||||||
|         logContext.Properties ??= new Dictionary<object, object>(); |         logContext.Properties ??= new Dictionary<string, object>(); | ||||||
|  |  | ||||||
|         logContext.Properties.Remove(key); |         logContext.Properties.Remove(key); | ||||||
|         logContext.Properties.Add(key, value); |         logContext.Properties.Add(key, value); | ||||||
| @@ -43,7 +43,7 @@ public static class LogContextExtensions | |||||||
|     /// <param name="logContext"></param> |     /// <param name="logContext"></param> | ||||||
|     /// <param name="properties"></param> |     /// <param name="properties"></param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public static LogContext SetRange(this LogContext logContext, IDictionary<object, object> properties) |     public static LogContext SetRange(this LogContext logContext, IDictionary<string, object> properties) | ||||||
|     { |     { | ||||||
|         if (logContext == null |         if (logContext == null | ||||||
|             || properties == null |             || properties == null | ||||||
| @@ -63,7 +63,7 @@ public static class LogContextExtensions | |||||||
|     /// <param name="logContext"></param> |     /// <param name="logContext"></param> | ||||||
|     /// <param name="key">键</param> |     /// <param name="key">键</param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public static object Get(this LogContext logContext, object key) |     public static object Get(this LogContext logContext, string key) | ||||||
|     { |     { | ||||||
|         if (logContext == null |         if (logContext == null | ||||||
|             || key == null |             || key == null | ||||||
| @@ -80,7 +80,7 @@ public static class LogContextExtensions | |||||||
|     /// <param name="logContext"></param> |     /// <param name="logContext"></param> | ||||||
|     /// <param name="key">键</param> |     /// <param name="key">键</param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public static T Get<T>(this LogContext logContext, object key) |     public static T Get<T>(this LogContext logContext, string key) | ||||||
|     { |     { | ||||||
|         var value = logContext.Get(key); |         var value = logContext.Get(key); | ||||||
|         return value.ChangeType<T>(); |         return value.ChangeType<T>(); | ||||||
|   | |||||||
| @@ -84,7 +84,7 @@ public static class StringLoggingExtensions | |||||||
|     /// <param name="message"></param> |     /// <param name="message"></param> | ||||||
|     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> |     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public static StringLoggingPart ScopeContext(this string message, IDictionary<object, object> properties) |     public static StringLoggingPart ScopeContext(this string message, IDictionary<string, object> properties) | ||||||
|     { |     { | ||||||
|         return StringLoggingPart.Default().SetMessage(message).ScopeContext(properties); |         return StringLoggingPart.Default().SetMessage(message).ScopeContext(properties); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ namespace ThingsGateway.Logging; | |||||||
| /// </summary> | /// </summary> | ||||||
| /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | ||||||
| [SuppressSniffer] | [SuppressSniffer] | ||||||
| public sealed class DatabaseLogger : ILogger | public sealed class DatabaseLogger : ILogger, IDisposable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 记录器类别名称 |     /// 记录器类别名称 | ||||||
| @@ -60,6 +60,11 @@ public sealed class DatabaseLogger : ILogger | |||||||
|         return _databaseLoggerProvider.ScopeProvider?.Push(state); |         return _databaseLoggerProvider.ScopeProvider?.Push(state); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public void Dispose() | ||||||
|  |     { | ||||||
|  |         _databaseLoggerProvider.RemoveCache(_logName); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 检查是否已启用给定日志级别 |     /// 检查是否已启用给定日志级别 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|   | |||||||
| @@ -14,6 +14,8 @@ using Microsoft.Extensions.Logging; | |||||||
|  |  | ||||||
| using System.Collections.Concurrent; | using System.Collections.Concurrent; | ||||||
|  |  | ||||||
|  | using ThingsGateway.Extension.Generic; | ||||||
|  |  | ||||||
| namespace ThingsGateway.Logging; | namespace ThingsGateway.Logging; | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| @@ -54,6 +56,8 @@ public sealed class DatabaseLoggerProvider : ILoggerProvider, ISupportExternalSc | |||||||
|     /// <remarks>实现不间断写入</remarks> |     /// <remarks>实现不间断写入</remarks> | ||||||
|     private Task _processQueueTask; |     private Task _processQueueTask; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 构造函数 |     /// 构造函数 | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @@ -82,7 +86,10 @@ public sealed class DatabaseLoggerProvider : ILoggerProvider, ISupportExternalSc | |||||||
|     { |     { | ||||||
|         return _databaseLoggers.GetOrAdd(categoryName, name => new DatabaseLogger(name, this)); |         return _databaseLoggers.GetOrAdd(categoryName, name => new DatabaseLogger(name, this)); | ||||||
|     } |     } | ||||||
|  |     public void RemoveCache(string categoryName) | ||||||
|  |     { | ||||||
|  |         _databaseLoggers.Remove(categoryName); | ||||||
|  |     } | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 设置作用域提供器 |     /// 设置作用域提供器 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|   | |||||||
| @@ -18,8 +18,17 @@ namespace ThingsGateway.Logging; | |||||||
| /// </summary> | /// </summary> | ||||||
| /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | ||||||
| [SuppressSniffer] | [SuppressSniffer] | ||||||
| public sealed class EmptyLogger : ILogger | public sealed class EmptyLogger : ILogger, IDisposable | ||||||
| { | { | ||||||
|  |     public EmptyLogger(string categoryName, EmptyLoggerProvider emptyLoggerProvider) | ||||||
|  |     { | ||||||
|  |         _logName = categoryName; | ||||||
|  |         _emptyLoggerProvider = emptyLoggerProvider; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private string _logName { get; } | ||||||
|  |     private EmptyLoggerProvider _emptyLoggerProvider { get; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 开始逻辑操作范围 |     /// 开始逻辑操作范围 | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @@ -31,6 +40,11 @@ public sealed class EmptyLogger : ILogger | |||||||
|         return default; |         return default; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public void Dispose() | ||||||
|  |     { | ||||||
|  |         _emptyLoggerProvider.RemoveCache(_logName); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 检查是否已启用给定日志级别 |     /// 检查是否已启用给定日志级别 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|   | |||||||
| @@ -13,6 +13,8 @@ using Microsoft.Extensions.Logging; | |||||||
|  |  | ||||||
| using System.Collections.Concurrent; | using System.Collections.Concurrent; | ||||||
|  |  | ||||||
|  | using ThingsGateway.Extension.Generic; | ||||||
|  |  | ||||||
| namespace ThingsGateway.Logging; | namespace ThingsGateway.Logging; | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| @@ -34,9 +36,12 @@ public sealed class EmptyLoggerProvider : ILoggerProvider | |||||||
|     /// <returns><see cref="ILogger"/></returns> |     /// <returns><see cref="ILogger"/></returns> | ||||||
|     public ILogger CreateLogger(string categoryName) |     public ILogger CreateLogger(string categoryName) | ||||||
|     { |     { | ||||||
|         return _emptyLoggers.GetOrAdd(categoryName, name => new EmptyLogger()); |         return _emptyLoggers.GetOrAdd(categoryName, name => new EmptyLogger(categoryName, this)); | ||||||
|  |     } | ||||||
|  |     public void RemoveCache(string categoryName) | ||||||
|  |     { | ||||||
|  |         _emptyLoggers.Remove(categoryName); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 释放非托管资源 |     /// 释放非托管资源 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ namespace ThingsGateway.Logging; | |||||||
| /// </summary> | /// </summary> | ||||||
| /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | ||||||
| [SuppressSniffer] | [SuppressSniffer] | ||||||
| public sealed class FileLogger : ILogger | public sealed class FileLogger : ILogger, IDisposable | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 记录器类别名称 |     /// 记录器类别名称 | ||||||
| @@ -58,6 +58,11 @@ public sealed class FileLogger : ILogger | |||||||
|         return _fileLoggerProvider.ScopeProvider?.Push(state); |         return _fileLoggerProvider.ScopeProvider?.Push(state); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public void Dispose() | ||||||
|  |     { | ||||||
|  |         _fileLoggerProvider.RemoveCache(_logName); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 检查是否已启用给定日志级别 |     /// 检查是否已启用给定日志级别 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|   | |||||||
| @@ -13,6 +13,8 @@ using Microsoft.Extensions.Logging; | |||||||
|  |  | ||||||
| using System.Collections.Concurrent; | using System.Collections.Concurrent; | ||||||
|  |  | ||||||
|  | using ThingsGateway.Extension.Generic; | ||||||
|  |  | ||||||
| namespace ThingsGateway.Logging; | namespace ThingsGateway.Logging; | ||||||
|  |  | ||||||
| /// <summary> | /// <summary> | ||||||
| @@ -116,6 +118,10 @@ public sealed class FileLoggerProvider : ILoggerProvider, ISupportExternalScope | |||||||
|     { |     { | ||||||
|         return _fileLoggers.GetOrAdd(categoryName, name => new FileLogger(name, this)); |         return _fileLoggers.GetOrAdd(categoryName, name => new FileLogger(name, this)); | ||||||
|     } |     } | ||||||
|  |     public void RemoveCache(string categoryName) | ||||||
|  |     { | ||||||
|  |         _fileLoggers.Remove(categoryName); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 设置作用域提供器 |     /// 设置作用域提供器 | ||||||
|   | |||||||
| @@ -17,11 +17,10 @@ namespace ThingsGateway.Logging; | |||||||
| [SuppressSniffer] | [SuppressSniffer] | ||||||
| public sealed class LogContext : IDisposable | public sealed class LogContext : IDisposable | ||||||
| { | { | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 日志上下文数据 |     /// 日志上下文数据 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public IDictionary<object, object> Properties { get; set; } |     public IDictionary<string, object> Properties { get; set; } | ||||||
|  |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 原生日志上下文数据 |     /// 原生日志上下文数据 | ||||||
|   | |||||||
| @@ -96,7 +96,7 @@ public sealed partial class StringLoggingPart | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> |     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public StringLoggingPart ScopeContext(IDictionary<object, object> properties) |     public StringLoggingPart ScopeContext(IDictionary<string, object> properties) | ||||||
|     { |     { | ||||||
|         if (properties == null) return this; |         if (properties == null) return this; | ||||||
|         LogContext = new LogContext { Properties = properties }; |         LogContext = new LogContext { Properties = properties }; | ||||||
|   | |||||||
| @@ -59,7 +59,7 @@ public static class Log | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> |     /// <param name="properties">建议使用 ConcurrentDictionary 类型</param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public static (ILogger logger, IDisposable scope) ScopeContext(IDictionary<object, object> properties) |     public static (ILogger logger, IDisposable scope) ScopeContext(IDictionary<string, object> properties) | ||||||
|     { |     { | ||||||
|         return GetLogger(StringLoggingPart.Default().ScopeContext(properties)); |         return GetLogger(StringLoggingPart.Default().ScopeContext(properties)); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ | |||||||
| 		<PackageReference Include="System.Text.RegularExpressions" Version="4.3.1" /> | 		<PackageReference Include="System.Text.RegularExpressions" Version="4.3.1" /> | ||||||
| 		<PackageReference Include="Mapster" Version="7.4.0" /> | 		<PackageReference Include="Mapster" Version="7.4.0" /> | ||||||
| 		<PackageReference Include="MiniProfiler.AspNetCore.Mvc" Version="4.5.4" /> | 		<PackageReference Include="MiniProfiler.AspNetCore.Mvc" Version="4.5.4" /> | ||||||
| 		<PackageReference Include="Swashbuckle.AspNetCore" Version="8.1.1" /> | 		<PackageReference Include="Swashbuckle.AspNetCore" Version="8.1.2" /> | ||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
|  |  | ||||||
| 	<ItemGroup Condition=" '$(TargetFramework)' == 'net8.0' "> | 	<ItemGroup Condition=" '$(TargetFramework)' == 'net8.0' "> | ||||||
|   | |||||||
| @@ -349,7 +349,7 @@ public static class UnifyContext | |||||||
|     /// <param name="result"></param> |     /// <param name="result"></param> | ||||||
|     /// <param name="data"></param> |     /// <param name="data"></param> | ||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     internal static bool CheckVaildResult(IActionResult result, out object data) |     public static bool CheckVaildResult(IActionResult result, out object data) | ||||||
|     { |     { | ||||||
|         data = default; |         data = default; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| using System.Buffers; | using System.Runtime.CompilerServices; | ||||||
| using System.Runtime.CompilerServices; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.NewLife.Buffers; | namespace ThingsGateway.NewLife.Buffers; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,4 @@ | |||||||
| using System.Buffers; | #if NETFRAMEWORK || NETSTANDARD2_0 | ||||||
|  |  | ||||||
| #if NETFRAMEWORK || NETSTANDARD2_0 |  | ||||||
| using ValueTask = System.Threading.Tasks.Task; | using ValueTask = System.Threading.Tasks.Task; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| using System.Buffers; | using System.Runtime.InteropServices; | ||||||
| using System.Runtime.InteropServices; |  | ||||||
| using System.Text; | using System.Text; | ||||||
|  |  | ||||||
| using ThingsGateway.NewLife.Collections; | using ThingsGateway.NewLife.Collections; | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| using System.Buffers; | using System.Text; | ||||||
| using System.Text; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.NewLife.Collections; | namespace ThingsGateway.NewLife.Collections; | ||||||
|  |  | ||||||
| @@ -95,7 +94,7 @@ public static class Pool | |||||||
|     { |     { | ||||||
|         //if (ms == null) return null; |         //if (ms == null) return null; | ||||||
|  |  | ||||||
|         var buf = returnResult ? ms.ToArray() : Empty; |         var buf = returnResult ? ms.ToArray() : Array.Empty<byte>(); | ||||||
|  |  | ||||||
|         Pool.MemoryStream.Return(ms); |         Pool.MemoryStream.Return(ms); | ||||||
|  |  | ||||||
| @@ -133,11 +132,5 @@ public static class Pool | |||||||
|     } |     } | ||||||
|     #endregion |     #endregion | ||||||
|  |  | ||||||
|     #region ByteArray |  | ||||||
|     /// <summary>字节数组共享存储</summary> |  | ||||||
|     public static ArrayPool<Byte> Shared { get; set; } = ArrayPool<Byte>.Shared; |  | ||||||
|  |  | ||||||
|     /// <summary>空数组</summary> |  | ||||||
|     public static Byte[] Empty { get; } = []; |  | ||||||
|     #endregion |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,8 +2,6 @@ | |||||||
| using System.Globalization; | using System.Globalization; | ||||||
| using System.Reflection; | using System.Reflection; | ||||||
|  |  | ||||||
| using ThingsGateway.NewLife.Collections; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.NewLife.Extension; | namespace ThingsGateway.NewLife.Extension; | ||||||
|  |  | ||||||
| /// <summary>工具类</summary> | /// <summary>工具类</summary> | ||||||
| @@ -452,12 +450,12 @@ public class DefaultConvert | |||||||
|                     // 凑够8字节 |                     // 凑够8字节 | ||||||
|                     if (buf.Length < 8) |                     if (buf.Length < 8) | ||||||
|                     { |                     { | ||||||
|                         var bts = Pool.Shared.Rent(8); |                         var bts = ArrayPool<Byte>.Shared.Rent(8); | ||||||
|                         Buffer.BlockCopy(buf, 0, bts, 0, buf.Length); |                         Buffer.BlockCopy(buf, 0, bts, 0, buf.Length); | ||||||
|  |  | ||||||
|                         var dec = BitConverter.ToDouble(bts, 0).ToDecimal(); |                         var dec = BitConverter.ToDouble(bts, 0).ToDecimal(); | ||||||
|  |  | ||||||
|                         Pool.Shared.Return(bts); |                         ArrayPool<Byte>.Shared.Return(bts); | ||||||
|  |  | ||||||
|                         return dec; |                         return dec; | ||||||
|                     } |                     } | ||||||
|   | |||||||
| @@ -1,57 +0,0 @@ | |||||||
| //------------------------------------------------------------------------------ |  | ||||||
| //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 |  | ||||||
| //  此代码版权(除特别声明外的代码)归作者本人Diego所有 |  | ||||||
| //  源代码使用协议遵循本仓库的开源协议及附加协议 |  | ||||||
| //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway |  | ||||||
| //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway |  | ||||||
| //  使用文档:https://thingsgateway.cn/ |  | ||||||
| //  QQ群:605534569 |  | ||||||
| //------------------------------------------------------------------------------ |  | ||||||
|  |  | ||||||
| using Newtonsoft.Json; |  | ||||||
|  |  | ||||||
| namespace ThingsGateway.NewLife.Extension; |  | ||||||
|  |  | ||||||
| public class ByteArrayToNumberArrayConverter : JsonConverter<byte[]> |  | ||||||
| { |  | ||||||
|     public override void WriteJson(JsonWriter writer, byte[]? value, JsonSerializer serializer) |  | ||||||
|     { |  | ||||||
|         if (value == null) |  | ||||||
|         { |  | ||||||
|             writer.WriteNull(); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         // 将 byte[] 转换为数值数组 |  | ||||||
|         writer.WriteStartArray(); |  | ||||||
|         foreach (var b in value) |  | ||||||
|         { |  | ||||||
|             writer.WriteValue(b); |  | ||||||
|         } |  | ||||||
|         writer.WriteEndArray(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public override byte[] ReadJson(JsonReader reader, Type objectType, byte[]? existingValue, bool hasExistingValue, JsonSerializer serializer) |  | ||||||
|     { |  | ||||||
|         // 从数值数组读取 byte[] |  | ||||||
|         if (reader.TokenType == JsonToken.StartArray) |  | ||||||
|         { |  | ||||||
|             var byteList = new System.Collections.Generic.List<byte>(); |  | ||||||
|             while (reader.Read()) |  | ||||||
|             { |  | ||||||
|                 if (reader.TokenType == JsonToken.EndArray) |  | ||||||
|                 { |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if (reader.TokenType == JsonToken.Integer) |  | ||||||
|                 { |  | ||||||
|                     byteList.Add(Convert.ToByte(reader.Value)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             return byteList.ToArray(); |  | ||||||
|         } |  | ||||||
|         throw new JsonSerializationException("Invalid JSON format for byte array."); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public override bool CanRead => true; |  | ||||||
| } |  | ||||||
| @@ -15,14 +15,14 @@ namespace ThingsGateway.NewLife.Json.Extension; | |||||||
| /// <summary> | /// <summary> | ||||||
| /// json扩展 | /// json扩展 | ||||||
| /// </summary> | /// </summary> | ||||||
| public static class JsonExtensions | public static class JsonExtension | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 默认Json规则 |     /// 默认Json规则 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public static JsonSerializerSettings IndentedOptions; |     public static JsonSerializerSettings IndentedOptions; | ||||||
|     public static JsonSerializerSettings NoneIndentedOptions; |     public static JsonSerializerSettings NoneIndentedOptions; | ||||||
|     static JsonExtensions() |     static JsonExtension() | ||||||
|     { |     { | ||||||
|         IndentedOptions = new JsonSerializerSettings |         IndentedOptions = new JsonSerializerSettings | ||||||
|         { |         { | ||||||
| @@ -81,4 +81,52 @@ public static class JsonExtensions | |||||||
|     { |     { | ||||||
|         return Newtonsoft.Json.JsonConvert.SerializeObject(item, indented == false ? NoneIndentedOptions : IndentedOptions); |         return Newtonsoft.Json.JsonConvert.SerializeObject(item, indented == false ? NoneIndentedOptions : IndentedOptions); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | public class ByteArrayToNumberArrayConverter : JsonConverter<byte[]> | ||||||
|  | { | ||||||
|  |     public override void WriteJson(JsonWriter writer, byte[]? value, JsonSerializer serializer) | ||||||
|  |     { | ||||||
|  |         if (value == null) | ||||||
|  |         { | ||||||
|  |             writer.WriteNull(); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         // 将 byte[] 转换为数值数组 | ||||||
|  |         writer.WriteStartArray(); | ||||||
|  |         foreach (var b in value) | ||||||
|  |         { | ||||||
|  |             writer.WriteValue(b); | ||||||
|  |         } | ||||||
|  |         writer.WriteEndArray(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public override byte[] ReadJson(JsonReader reader, Type objectType, byte[]? existingValue, bool hasExistingValue, JsonSerializer serializer) | ||||||
|  |     { | ||||||
|  |         // 从数值数组读取 byte[] | ||||||
|  |         if (reader.TokenType == JsonToken.StartArray) | ||||||
|  |         { | ||||||
|  |             var byteList = new System.Collections.Generic.List<byte>(); | ||||||
|  |             while (reader.Read()) | ||||||
|  |             { | ||||||
|  |                 if (reader.TokenType == JsonToken.EndArray) | ||||||
|  |                 { | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 if (reader.TokenType == JsonToken.Integer) | ||||||
|  |                 { | ||||||
|  |                     byteList.Add(Convert.ToByte(reader.Value)); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return byteList.ToArray(); | ||||||
|  |         } | ||||||
|  |         throw new JsonSerializationException("Invalid JSON format for byte array."); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public override bool CanRead => true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| @@ -0,0 +1,694 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | #if NET6_0_OR_GREATER | ||||||
|  | using Newtonsoft.Json.Linq; | ||||||
|  |  | ||||||
|  | using System.Text.Encodings.Web; | ||||||
|  | using System.Text.Json; | ||||||
|  | using System.Text.Json.Serialization; | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.NewLife.Json.Extension; | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// System.Text.Json 扩展 | ||||||
|  | /// </summary> | ||||||
|  | public static class SystemTextJsonExtension | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// 默认Json规则(带缩进) | ||||||
|  |     /// </summary> | ||||||
|  |     public static JsonSerializerOptions IndentedOptions; | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 默认Json规则(无缩进) | ||||||
|  |     /// </summary> | ||||||
|  |     public static JsonSerializerOptions NoneIndentedOptions; | ||||||
|  |  | ||||||
|  |     static SystemTextJsonExtension() | ||||||
|  |     { | ||||||
|  |         IndentedOptions = new JsonSerializerOptions | ||||||
|  |         { | ||||||
|  |             Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, | ||||||
|  |             WriteIndented = true, // 缩进 | ||||||
|  |             DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull, // 忽略 null | ||||||
|  |             NumberHandling = JsonNumberHandling.AllowNamedFloatingPointLiterals, | ||||||
|  |         }; | ||||||
|  |         // 如有自定义Converter,这里添加 | ||||||
|  |         // IndentedOptions.Converters.Add(new ByteArrayJsonConverter()); | ||||||
|  |         IndentedOptions.Converters.Add(new ByteArrayToNumberArrayConverterSystemTextJson()); | ||||||
|  |         IndentedOptions.Converters.Add(new JTokenSystemTextJsonConverter()); | ||||||
|  |         IndentedOptions.Converters.Add(new JValueSystemTextJsonConverter()); | ||||||
|  |         IndentedOptions.Converters.Add(new JObjectSystemTextJsonConverter()); | ||||||
|  |         IndentedOptions.Converters.Add(new JArraySystemTextJsonConverter()); | ||||||
|  |         NoneIndentedOptions = new JsonSerializerOptions | ||||||
|  |         { | ||||||
|  |             Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, | ||||||
|  |             WriteIndented = false, // 不缩进 | ||||||
|  |             DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull, | ||||||
|  |             NumberHandling = JsonNumberHandling.AllowNamedFloatingPointLiterals, | ||||||
|  |         }; | ||||||
|  |         NoneIndentedOptions.Converters.Add(new ByteArrayToNumberArrayConverterSystemTextJson()); | ||||||
|  |         NoneIndentedOptions.Converters.Add(new JTokenSystemTextJsonConverter()); | ||||||
|  |         NoneIndentedOptions.Converters.Add(new JValueSystemTextJsonConverter()); | ||||||
|  |         NoneIndentedOptions.Converters.Add(new JObjectSystemTextJsonConverter()); | ||||||
|  |         NoneIndentedOptions.Converters.Add(new JArraySystemTextJsonConverter()); | ||||||
|  |         // NoneIndentedOptions.Converters.Add(new ByteArrayJsonConverter()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 反序列化 | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="json"></param> | ||||||
|  |     /// <param name="type"></param> | ||||||
|  |     /// <param name="options"></param> | ||||||
|  |     /// <returns></returns> | ||||||
|  |     public static object? FromSystemTextJsonString(this string json, Type type, JsonSerializerOptions? options = null) | ||||||
|  |     { | ||||||
|  |         return JsonSerializer.Deserialize(json, type, options ?? IndentedOptions); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 反序列化 | ||||||
|  |     /// </summary> | ||||||
|  |     public static T? FromSystemTextJsonString<T>(this string json, JsonSerializerOptions? options = null) | ||||||
|  |     { | ||||||
|  |         return JsonSerializer.Deserialize<T>(json, options ?? IndentedOptions); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 序列化 | ||||||
|  |     /// </summary> | ||||||
|  |     /// <param name="item"></param> | ||||||
|  |     /// <param name="options"></param> | ||||||
|  |     /// <returns></returns> | ||||||
|  |     public static string ToSystemTextJsonString(this object item, JsonSerializerOptions? options) | ||||||
|  |     { | ||||||
|  |         return JsonSerializer.Serialize(item, item?.GetType() ?? typeof(object), options ?? IndentedOptions); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 序列化 | ||||||
|  |     /// </summary> | ||||||
|  |     public static string ToSystemTextJsonString(this object item, bool indented = true) | ||||||
|  |     { | ||||||
|  |         return JsonSerializer.Serialize(item, item?.GetType() ?? typeof(object), indented ? IndentedOptions : NoneIndentedOptions); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// <summary> | ||||||
|  |     /// 序列化 | ||||||
|  |     /// </summary> | ||||||
|  |     public static byte[] ToSystemTextJsonUtf8Bytes(this object item, bool indented = true) | ||||||
|  |     { | ||||||
|  |         return JsonSerializer.SerializeToUtf8Bytes(item, item.GetType(), indented ? IndentedOptions : NoneIndentedOptions); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// 将 byte[] 序列化为数值数组,反序列化数值数组为 byte[] | ||||||
|  | /// </summary> | ||||||
|  | public class ByteArrayToNumberArrayConverterSystemTextJson : JsonConverter<byte[]> | ||||||
|  | { | ||||||
|  |     public override byte[]? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         if (reader.TokenType != JsonTokenType.StartArray) | ||||||
|  |         { | ||||||
|  |             throw new JsonException("Expected StartArray token."); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         var bytes = new List<byte>(); | ||||||
|  |         while (reader.Read()) | ||||||
|  |         { | ||||||
|  |             if (reader.TokenType == JsonTokenType.EndArray) | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             if (reader.TokenType == JsonTokenType.Number) | ||||||
|  |             { | ||||||
|  |                 if (reader.TryGetByte(out byte value)) | ||||||
|  |                 { | ||||||
|  |                     bytes.Add(value); | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     throw new JsonException("Invalid number value for byte array."); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 throw new JsonException($"Unexpected token {reader.TokenType} in byte array."); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return bytes.ToArray(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public override void Write(Utf8JsonWriter writer, byte[] value, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         if (value == null) | ||||||
|  |         { | ||||||
|  |             writer.WriteNullValue(); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         writer.WriteStartArray(); | ||||||
|  |         foreach (var b in value) | ||||||
|  |         { | ||||||
|  |             writer.WriteNumberValue(b); | ||||||
|  |         } | ||||||
|  |         writer.WriteEndArray(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// System.Text.Json → JToken / JObject / JArray 转换器 | ||||||
|  | /// </summary> | ||||||
|  | public class JTokenSystemTextJsonConverter : JsonConverter<JToken> | ||||||
|  | { | ||||||
|  |     public override JToken? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         return ReadToken(ref reader); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static JToken ReadToken(ref Utf8JsonReader reader) | ||||||
|  |     { | ||||||
|  |         switch (reader.TokenType) | ||||||
|  |         { | ||||||
|  |             case JsonTokenType.StartObject: | ||||||
|  |                 var obj = new JObject(); | ||||||
|  |                 while (reader.Read()) | ||||||
|  |                 { | ||||||
|  |                     if (reader.TokenType == JsonTokenType.EndObject) | ||||||
|  |                         return obj; | ||||||
|  |  | ||||||
|  |                     var propertyName = reader.GetString(); | ||||||
|  |                     reader.Read(); | ||||||
|  |                     var value = ReadToken(ref reader); | ||||||
|  |                     obj[propertyName!] = value; | ||||||
|  |                 } | ||||||
|  |                 throw new JsonException(); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.StartArray: | ||||||
|  |                 var array = new JArray(); | ||||||
|  |                 while (reader.Read()) | ||||||
|  |                 { | ||||||
|  |                     if (reader.TokenType == JsonTokenType.EndArray) | ||||||
|  |                         return array; | ||||||
|  |  | ||||||
|  |                     array.Add(ReadToken(ref reader)); | ||||||
|  |                 } | ||||||
|  |                 throw new JsonException(); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.String: | ||||||
|  |                 if (reader.TryGetDateTime(out var date)) | ||||||
|  |                     return new JValue(date); | ||||||
|  |                 return new JValue(reader.GetString()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Number: | ||||||
|  |                 if (reader.TryGetInt64(out var l)) | ||||||
|  |                     return new JValue(l); | ||||||
|  |                 return new JValue(reader.GetDouble()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.True: | ||||||
|  |                 return new JValue(true); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.False: | ||||||
|  |                 return new JValue(false); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Null: | ||||||
|  |                 return JValue.CreateNull(); | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 throw new JsonException($"Unsupported token type {reader.TokenType}"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public override void Write(Utf8JsonWriter writer, JToken value, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         switch (value.Type) | ||||||
|  |         { | ||||||
|  |             case JTokenType.Object: | ||||||
|  |                 writer.WriteStartObject(); | ||||||
|  |                 foreach (var prop in (JObject)value) | ||||||
|  |                 { | ||||||
|  |                     writer.WritePropertyName(prop.Key); | ||||||
|  |                     Write(writer, prop.Value!, options); | ||||||
|  |                 } | ||||||
|  |                 writer.WriteEndObject(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Array: | ||||||
|  |                 writer.WriteStartArray(); | ||||||
|  |                 foreach (var item in (JArray)value) | ||||||
|  |                 { | ||||||
|  |                     Write(writer, item!, options); | ||||||
|  |                 } | ||||||
|  |                 writer.WriteEndArray(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Null: | ||||||
|  |                 writer.WriteNullValue(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Boolean: | ||||||
|  |                 writer.WriteBooleanValue(value.Value<bool>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Integer: | ||||||
|  |                 writer.WriteNumberValue(value.Value<long>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Float: | ||||||
|  |                 writer.WriteNumberValue(value.Value<double>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.String: | ||||||
|  |                 writer.WriteStringValue(value.Value<string>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Date: | ||||||
|  |                 writer.WriteStringValue(value.Value<DateTime>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Guid: | ||||||
|  |                 writer.WriteStringValue(value.Value<Guid>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Uri: | ||||||
|  |                 writer.WriteStringValue(value.Value<Uri>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.TimeSpan: | ||||||
|  |                 writer.WriteStringValue(value.Value<TimeSpan>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 // fallback — 转字符串 | ||||||
|  |                 writer.WriteStringValue(value.ToString()); | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// System.Text.Json → JToken / JObject / JArray 转换器 | ||||||
|  | /// </summary> | ||||||
|  | public class JObjectSystemTextJsonConverter : JsonConverter<JObject> | ||||||
|  | { | ||||||
|  |     public override JObject? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         var obj = new JObject(); | ||||||
|  |         while (reader.Read()) | ||||||
|  |         { | ||||||
|  |             if (reader.TokenType == JsonTokenType.EndObject) | ||||||
|  |                 return obj; | ||||||
|  |  | ||||||
|  |             var propertyName = reader.GetString(); | ||||||
|  |             reader.Read(); | ||||||
|  |             var value = ReadJToken(ref reader); | ||||||
|  |             obj[propertyName!] = value; | ||||||
|  |         } | ||||||
|  |         throw new JsonException(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static JToken ReadJToken(ref Utf8JsonReader reader) | ||||||
|  |     { | ||||||
|  |         switch (reader.TokenType) | ||||||
|  |         { | ||||||
|  |             case JsonTokenType.StartObject: | ||||||
|  |                 var obj = new JObject(); | ||||||
|  |                 while (reader.Read()) | ||||||
|  |                 { | ||||||
|  |                     if (reader.TokenType == JsonTokenType.EndObject) | ||||||
|  |                         return obj; | ||||||
|  |  | ||||||
|  |                     var propertyName = reader.GetString(); | ||||||
|  |                     reader.Read(); | ||||||
|  |                     var value = ReadJToken(ref reader); | ||||||
|  |                     obj[propertyName!] = value; | ||||||
|  |                 } | ||||||
|  |                 throw new JsonException(); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.StartArray: | ||||||
|  |                 var array = new JArray(); | ||||||
|  |                 while (reader.Read()) | ||||||
|  |                 { | ||||||
|  |                     if (reader.TokenType == JsonTokenType.EndArray) | ||||||
|  |                         return array; | ||||||
|  |  | ||||||
|  |                     array.Add(ReadJToken(ref reader)); | ||||||
|  |                 } | ||||||
|  |                 throw new JsonException(); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.String: | ||||||
|  |                 if (reader.TryGetDateTime(out var date)) | ||||||
|  |                     return new JValue(date); | ||||||
|  |                 return new JValue(reader.GetString()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Number: | ||||||
|  |                 if (reader.TryGetInt64(out var l)) | ||||||
|  |                     return new JValue(l); | ||||||
|  |                 return new JValue(reader.GetDouble()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.True: | ||||||
|  |                 return new JValue(true); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.False: | ||||||
|  |                 return new JValue(false); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Null: | ||||||
|  |                 return JValue.CreateNull(); | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 throw new JsonException($"Unsupported token type {reader.TokenType}"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public override void Write(Utf8JsonWriter writer, JObject value, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |  | ||||||
|  |         writer.WriteStartObject(); | ||||||
|  |         foreach (var prop in (JObject)value) | ||||||
|  |         { | ||||||
|  |             writer.WritePropertyName(prop.Key); | ||||||
|  |             Write(writer, prop.Value!, options); | ||||||
|  |         } | ||||||
|  |         writer.WriteEndObject(); | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     private static void Write(Utf8JsonWriter writer, JToken value, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         switch (value.Type) | ||||||
|  |         { | ||||||
|  |             case JTokenType.Object: | ||||||
|  |                 writer.WriteStartObject(); | ||||||
|  |                 foreach (var prop in (JObject)value) | ||||||
|  |                 { | ||||||
|  |                     writer.WritePropertyName(prop.Key); | ||||||
|  |                     Write(writer, prop.Value!, options); | ||||||
|  |                 } | ||||||
|  |                 writer.WriteEndObject(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Array: | ||||||
|  |                 writer.WriteStartArray(); | ||||||
|  |                 foreach (var item in (JArray)value) | ||||||
|  |                 { | ||||||
|  |                     Write(writer, item!, options); | ||||||
|  |                 } | ||||||
|  |                 writer.WriteEndArray(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Null: | ||||||
|  |                 writer.WriteNullValue(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Boolean: | ||||||
|  |                 writer.WriteBooleanValue(value.Value<bool>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Integer: | ||||||
|  |                 writer.WriteNumberValue(value.Value<long>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Float: | ||||||
|  |                 writer.WriteNumberValue(value.Value<double>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.String: | ||||||
|  |                 writer.WriteStringValue(value.Value<string>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Date: | ||||||
|  |                 writer.WriteStringValue(value.Value<DateTime>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Guid: | ||||||
|  |                 writer.WriteStringValue(value.Value<Guid>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Uri: | ||||||
|  |                 writer.WriteStringValue(value.Value<Uri>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.TimeSpan: | ||||||
|  |                 writer.WriteStringValue(value.Value<TimeSpan>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 // fallback — 转字符串 | ||||||
|  |                 writer.WriteStringValue(value.ToString()); | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// System.Text.Json → JToken / JObject / JArray 转换器 | ||||||
|  | /// </summary> | ||||||
|  | public class JArraySystemTextJsonConverter : JsonConverter<JArray> | ||||||
|  | { | ||||||
|  |     public override JArray? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         var array = new JArray(); | ||||||
|  |         while (reader.Read()) | ||||||
|  |         { | ||||||
|  |             if (reader.TokenType == JsonTokenType.EndArray) | ||||||
|  |                 return array; | ||||||
|  |  | ||||||
|  |             array.Add(ReadToken(ref reader)); | ||||||
|  |         } | ||||||
|  |         throw new JsonException(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static JToken ReadToken(ref Utf8JsonReader reader) | ||||||
|  |     { | ||||||
|  |         switch (reader.TokenType) | ||||||
|  |         { | ||||||
|  |             case JsonTokenType.StartObject: | ||||||
|  |                 var obj = new JObject(); | ||||||
|  |                 while (reader.Read()) | ||||||
|  |                 { | ||||||
|  |                     if (reader.TokenType == JsonTokenType.EndObject) | ||||||
|  |                         return obj; | ||||||
|  |  | ||||||
|  |                     var propertyName = reader.GetString(); | ||||||
|  |                     reader.Read(); | ||||||
|  |                     var value = ReadToken(ref reader); | ||||||
|  |                     obj[propertyName!] = value; | ||||||
|  |                 } | ||||||
|  |                 throw new JsonException(); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.StartArray: | ||||||
|  |                 var array = new JArray(); | ||||||
|  |                 while (reader.Read()) | ||||||
|  |                 { | ||||||
|  |                     if (reader.TokenType == JsonTokenType.EndArray) | ||||||
|  |                         return array; | ||||||
|  |  | ||||||
|  |                     array.Add(ReadToken(ref reader)); | ||||||
|  |                 } | ||||||
|  |                 throw new JsonException(); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.String: | ||||||
|  |                 if (reader.TryGetDateTime(out var date)) | ||||||
|  |                     return new JValue(date); | ||||||
|  |                 return new JValue(reader.GetString()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Number: | ||||||
|  |                 if (reader.TryGetInt64(out var l)) | ||||||
|  |                     return new JValue(l); | ||||||
|  |                 return new JValue(reader.GetDouble()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.True: | ||||||
|  |                 return new JValue(true); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.False: | ||||||
|  |                 return new JValue(false); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Null: | ||||||
|  |                 return JValue.CreateNull(); | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 throw new JsonException($"Unsupported token type {reader.TokenType}"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public override void Write(Utf8JsonWriter writer, JArray value, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |  | ||||||
|  |         writer.WriteStartArray(); | ||||||
|  |         foreach (var item in (JArray)value) | ||||||
|  |         { | ||||||
|  |             Write(writer, item!, options); | ||||||
|  |         } | ||||||
|  |         writer.WriteEndArray(); | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void Write(Utf8JsonWriter writer, JToken value, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         switch (value.Type) | ||||||
|  |         { | ||||||
|  |             case JTokenType.Object: | ||||||
|  |                 writer.WriteStartObject(); | ||||||
|  |                 foreach (var prop in (JObject)value) | ||||||
|  |                 { | ||||||
|  |                     writer.WritePropertyName(prop.Key); | ||||||
|  |                     Write(writer, prop.Value!, options); | ||||||
|  |                 } | ||||||
|  |                 writer.WriteEndObject(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Array: | ||||||
|  |                 writer.WriteStartArray(); | ||||||
|  |                 foreach (var item in (JArray)value) | ||||||
|  |                 { | ||||||
|  |                     Write(writer, item!, options); | ||||||
|  |                 } | ||||||
|  |                 writer.WriteEndArray(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Null: | ||||||
|  |                 writer.WriteNullValue(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Boolean: | ||||||
|  |                 writer.WriteBooleanValue(value.Value<bool>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Integer: | ||||||
|  |                 writer.WriteNumberValue(value.Value<long>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Float: | ||||||
|  |                 writer.WriteNumberValue(value.Value<double>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.String: | ||||||
|  |                 writer.WriteStringValue(value.Value<string>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Date: | ||||||
|  |                 writer.WriteStringValue(value.Value<DateTime>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Guid: | ||||||
|  |                 writer.WriteStringValue(value.Value<Guid>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Uri: | ||||||
|  |                 writer.WriteStringValue(value.Value<Uri>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.TimeSpan: | ||||||
|  |                 writer.WriteStringValue(value.Value<TimeSpan>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 // fallback — 转字符串 | ||||||
|  |                 writer.WriteStringValue(value.ToString()); | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /// <summary> | ||||||
|  | /// System.Text.Json → JToken / JObject / JArray 转换器 | ||||||
|  | /// </summary> | ||||||
|  | public class JValueSystemTextJsonConverter : JsonConverter<JValue> | ||||||
|  | { | ||||||
|  |     public override JValue? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         return ReadJValue(ref reader); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static JValue ReadJValue(ref Utf8JsonReader reader) | ||||||
|  |     { | ||||||
|  |         switch (reader.TokenType) | ||||||
|  |         { | ||||||
|  |  | ||||||
|  |             case JsonTokenType.String: | ||||||
|  |                 if (reader.TryGetDateTime(out var date)) | ||||||
|  |                     return new JValue(date); | ||||||
|  |                 return new JValue(reader.GetString()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Number: | ||||||
|  |                 if (reader.TryGetInt64(out var l)) | ||||||
|  |                     return new JValue(l); | ||||||
|  |                 return new JValue(reader.GetDouble()); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.True: | ||||||
|  |                 return new JValue(true); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.False: | ||||||
|  |                 return new JValue(false); | ||||||
|  |  | ||||||
|  |             case JsonTokenType.Null: | ||||||
|  |                 return JValue.CreateNull(); | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 throw new JsonException($"Unsupported token type {reader.TokenType}"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public override void Write(Utf8JsonWriter writer, JValue value, JsonSerializerOptions options) | ||||||
|  |     { | ||||||
|  |         switch (value.Type) | ||||||
|  |         { | ||||||
|  |  | ||||||
|  |             case JTokenType.Null: | ||||||
|  |                 writer.WriteNullValue(); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Boolean: | ||||||
|  |                 writer.WriteBooleanValue(value.Value<bool>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Integer: | ||||||
|  |                 writer.WriteNumberValue(value.Value<long>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Float: | ||||||
|  |                 writer.WriteNumberValue(value.Value<double>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.String: | ||||||
|  |                 writer.WriteStringValue(value.Value<string>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Date: | ||||||
|  |                 writer.WriteStringValue(value.Value<DateTime>()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Guid: | ||||||
|  |                 writer.WriteStringValue(value.Value<Guid>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.Uri: | ||||||
|  |                 writer.WriteStringValue(value.Value<Uri>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             case JTokenType.TimeSpan: | ||||||
|  |                 writer.WriteStringValue(value.Value<TimeSpan>().ToString()); | ||||||
|  |                 break; | ||||||
|  |  | ||||||
|  |             default: | ||||||
|  |                 // fallback — 转字符串 | ||||||
|  |                 writer.WriteStringValue(value.ToString()); | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | #endif | ||||||
| @@ -8,4 +8,6 @@ | |||||||
| //  QQ群:605534569 | //  QQ群:605534569 | ||||||
| //------------------------------------------------------------------------------ | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | global using System.Buffers; | ||||||
|  |  | ||||||
| global using ThingsGateway.NewLife.Extension; | global using ThingsGateway.NewLife.Extension; | ||||||
|   | |||||||
| @@ -5,7 +5,6 @@ using System.Net.Sockets; | |||||||
| using System.Runtime.InteropServices; | using System.Runtime.InteropServices; | ||||||
|  |  | ||||||
| using ThingsGateway.NewLife.Caching; | using ThingsGateway.NewLife.Caching; | ||||||
| using ThingsGateway.NewLife.Collections; |  | ||||||
| using ThingsGateway.NewLife.Net; | using ThingsGateway.NewLife.Net; | ||||||
|  |  | ||||||
| namespace ThingsGateway.NewLife; | namespace ThingsGateway.NewLife; | ||||||
| @@ -52,7 +51,7 @@ public static class NetHelper | |||||||
| #endif | #endif | ||||||
|         { |         { | ||||||
|             UInt32 dummy = 0; |             UInt32 dummy = 0; | ||||||
|             var inOptionValues = Pool.Shared.Rent(Marshal.SizeOf(dummy) * 3); |             var inOptionValues = ArrayPool<Byte>.Shared.Rent(Marshal.SizeOf(dummy) * 3); | ||||||
|  |  | ||||||
|             // 是否启用Keep-Alive |             // 是否启用Keep-Alive | ||||||
|             BitConverter.GetBytes((UInt32)(isKeepAlive ? 1 : 0)).CopyTo(inOptionValues, 0); |             BitConverter.GetBytes((UInt32)(isKeepAlive ? 1 : 0)).CopyTo(inOptionValues, 0); | ||||||
| @@ -63,7 +62,7 @@ public static class NetHelper | |||||||
|  |  | ||||||
|             socket.IOControl(IOControlCode.KeepAliveValues, inOptionValues, null); |             socket.IOControl(IOControlCode.KeepAliveValues, inOptionValues, null); | ||||||
|  |  | ||||||
|             Pool.Shared.Return(inOptionValues); |             ArrayPool<Byte>.Shared.Return(inOptionValues); | ||||||
|  |  | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| @@ -538,11 +537,11 @@ public static class NetHelper | |||||||
|     private static void Wake(String mac) |     private static void Wake(String mac) | ||||||
|     { |     { | ||||||
|         mac = mac.Replace("-", null).Replace(":", null); |         mac = mac.Replace("-", null).Replace(":", null); | ||||||
|         var buffer = Pool.Shared.Rent(mac.Length / 2); |         var buffer = ArrayPool<Byte>.Shared.Rent(mac.Length / 2); | ||||||
|         for (var i = 0; i < buffer.Length; i++) |         for (var i = 0; i < buffer.Length; i++) | ||||||
|             buffer[i] = Byte.Parse(mac.Substring(i * 2, 2), NumberStyles.HexNumber); |             buffer[i] = Byte.Parse(mac.Substring(i * 2, 2), NumberStyles.HexNumber); | ||||||
|  |  | ||||||
|         var bts = Pool.Shared.Rent(6 + 16 * buffer.Length); |         var bts = ArrayPool<Byte>.Shared.Rent(6 + 16 * buffer.Length); | ||||||
|         for (var i = 0; i < 6; i++) |         for (var i = 0; i < 6; i++) | ||||||
|             bts[i] = 0xFF; |             bts[i] = 0xFF; | ||||||
|         for (Int32 i = 6, k = 0; i < bts.Length; i++, k++) |         for (Int32 i = 6, k = 0; i < bts.Length; i++, k++) | ||||||
| @@ -560,8 +559,8 @@ public static class NetHelper | |||||||
|         client.Close(); |         client.Close(); | ||||||
|         //client.SendAsync(bts, bts.Length, new IPEndPoint(IPAddress.Broadcast, 7)); |         //client.SendAsync(bts, bts.Length, new IPEndPoint(IPAddress.Broadcast, 7)); | ||||||
|  |  | ||||||
|         Pool.Shared.Return(bts); |         ArrayPool<Byte>.Shared.Return(bts); | ||||||
|         Pool.Shared.Return(buffer); |         ArrayPool<Byte>.Shared.Return(buffer); | ||||||
|     } |     } | ||||||
|     #endregion |     #endregion | ||||||
|  |  | ||||||
|   | |||||||
| @@ -627,6 +627,8 @@ public class DefaultReflect : IReflect | |||||||
|     /// <returns></returns> |     /// <returns></returns> | ||||||
|     public virtual Type? GetElementType(Type type) |     public virtual Type? GetElementType(Type type) | ||||||
|     { |     { | ||||||
|  |         if (type == null) return null; | ||||||
|  |  | ||||||
|         if (type.HasElementType) return type.GetElementType(); |         if (type.HasElementType) return type.GetElementType(); | ||||||
|  |  | ||||||
|         if (type.As<IEnumerable>()) |         if (type.As<IEnumerable>()) | ||||||
|   | |||||||
| @@ -1,8 +1,6 @@ | |||||||
| using System.Buffers; | using System.Reflection; | ||||||
| using System.Reflection; |  | ||||||
| using System.Runtime.InteropServices; | using System.Runtime.InteropServices; | ||||||
|  |  | ||||||
| using ThingsGateway.NewLife.Collections; |  | ||||||
| using ThingsGateway.NewLife.Reflection; | using ThingsGateway.NewLife.Reflection; | ||||||
|  |  | ||||||
| namespace ThingsGateway.NewLife.Serialization; | namespace ThingsGateway.NewLife.Serialization; | ||||||
| @@ -495,7 +493,7 @@ public class Binary : FormatterBase, IBinary | |||||||
|     /// <param name="max"></param> |     /// <param name="max"></param> | ||||||
|     public void WriteBCD(String value, Int32 max) |     public void WriteBCD(String value, Int32 max) | ||||||
|     { |     { | ||||||
|         var buf = Pool.Shared.Rent(max); |         var buf = ArrayPool<Byte>.Shared.Rent(max); | ||||||
|         for (Int32 i = 0, j = 0; i < max && j + 1 < value.Length; i++, j += 2) |         for (Int32 i = 0, j = 0; i < max && j + 1 < value.Length; i++, j += 2) | ||||||
|         { |         { | ||||||
|             var a = (Byte)(value[j] - '0'); |             var a = (Byte)(value[j] - '0'); | ||||||
| @@ -504,7 +502,7 @@ public class Binary : FormatterBase, IBinary | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         Write(buf, 0, max); |         Write(buf, 0, max); | ||||||
|         Pool.Shared.Return(buf); |         ArrayPool<Byte>.Shared.Return(buf); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// <summary>写入定长字符串。多余截取,少则补零</summary> |     /// <summary>写入定长字符串。多余截取,少则补零</summary> | ||||||
| @@ -512,11 +510,11 @@ public class Binary : FormatterBase, IBinary | |||||||
|     /// <param name="max"></param> |     /// <param name="max"></param> | ||||||
|     public void WriteFixedString(String? value, Int32 max) |     public void WriteFixedString(String? value, Int32 max) | ||||||
|     { |     { | ||||||
|         var buf = Pool.Shared.Rent(max); |         var buf = ArrayPool<Byte>.Shared.Rent(max); | ||||||
|         if (!value.IsNullOrEmpty()) Encoding.GetBytes(value, 0, value.Length, buf, 0); |         if (!value.IsNullOrEmpty()) Encoding.GetBytes(value, 0, value.Length, buf, 0); | ||||||
|  |  | ||||||
|         Write(buf, 0, max); |         Write(buf, 0, max); | ||||||
|         Pool.Shared.Return(buf); |         ArrayPool<Byte>.Shared.Return(buf); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// <summary>读取定长字符串。多余截取,少则补零</summary> |     /// <summary>读取定长字符串。多余截取,少则补零</summary> | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| using System.Globalization; | using System.Globalization; | ||||||
| using System.Xml; | using System.Xml; | ||||||
|  |  | ||||||
| using ThingsGateway.NewLife.Collections; |  | ||||||
| using ThingsGateway.NewLife.Reflection; | using ThingsGateway.NewLife.Reflection; | ||||||
|  |  | ||||||
| namespace ThingsGateway.NewLife.Serialization; | namespace ThingsGateway.NewLife.Serialization; | ||||||
| @@ -144,10 +143,10 @@ public class XmlGeneral : XmlHandlerBase | |||||||
|         else if (type == typeof(Byte[])) |         else if (type == typeof(Byte[])) | ||||||
|         { |         { | ||||||
|             // 用字符串长度作为预设缓冲区的长度 |             // 用字符串长度作为预设缓冲区的长度 | ||||||
|             var buf = Pool.Shared.Rent(reader.Value.Length); |             var buf = ArrayPool<Byte>.Shared.Rent(reader.Value.Length); | ||||||
|             var count = reader.ReadContentAsBase64(buf, 0, buf.Length); |             var count = reader.ReadContentAsBase64(buf, 0, buf.Length); | ||||||
|             value = buf.ReadBytes(0, count); |             value = buf.ReadBytes(0, count); | ||||||
|             Pool.Shared.Return(buf); |             ArrayPool<Byte>.Shared.Return(buf); | ||||||
|  |  | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -24,9 +24,8 @@ public static class ParallelExtensions | |||||||
|     /// <param name="body">要执行的操作</param> |     /// <param name="body">要执行的操作</param> | ||||||
|     public static void ParallelForEach<T>(this IEnumerable<T> source, Action<T> body) |     public static void ParallelForEach<T>(this IEnumerable<T> source, Action<T> body) | ||||||
|     { |     { | ||||||
|         // 创建并行操作的选项对象,设置最大并行度为当前处理器数量的一半 |  | ||||||
|         ParallelOptions options = new(); |         ParallelOptions options = new(); | ||||||
|         options.MaxDegreeOfParallelism = Environment.ProcessorCount / 2 == 0 ? 1 : Environment.ProcessorCount / 2; |         options.MaxDegreeOfParallelism = Environment.ProcessorCount; | ||||||
|         // 使用 Parallel.ForEach 执行指定的操作 |         // 使用 Parallel.ForEach 执行指定的操作 | ||||||
|         Parallel.ForEach(source, options, (variable) => |         Parallel.ForEach(source, options, (variable) => | ||||||
|         { |         { | ||||||
| @@ -42,9 +41,8 @@ public static class ParallelExtensions | |||||||
|     /// <param name="body">要执行的操作</param> |     /// <param name="body">要执行的操作</param> | ||||||
|     public static void ParallelForEach<T>(this IEnumerable<T> source, Action<T, ParallelLoopState, long> body) |     public static void ParallelForEach<T>(this IEnumerable<T> source, Action<T, ParallelLoopState, long> body) | ||||||
|     { |     { | ||||||
|         // 创建并行操作的选项对象,设置最大并行度为当前处理器数量的一半 |  | ||||||
|         ParallelOptions options = new(); |         ParallelOptions options = new(); | ||||||
|         options.MaxDegreeOfParallelism = Environment.ProcessorCount / 2 == 0 ? 1 : Environment.ProcessorCount / 2; |         options.MaxDegreeOfParallelism = Environment.ProcessorCount; | ||||||
|         // 使用 Parallel.ForEach 执行指定的操作 |         // 使用 Parallel.ForEach 执行指定的操作 | ||||||
|         Parallel.ForEach(source, options, (variable, state, index) => |         Parallel.ForEach(source, options, (variable, state, index) => | ||||||
|         { |         { | ||||||
|   | |||||||
| @@ -27,6 +27,13 @@ public class Startup : AppStartup | |||||||
|     { |     { | ||||||
|         services.AddBootstrapBlazor( |         services.AddBootstrapBlazor( | ||||||
|             option => option.JSModuleVersion = Random.Shared.Next(10000).ToString() |             option => option.JSModuleVersion = Random.Shared.Next(10000).ToString() | ||||||
|  |             , jsonLocalizationOptions => | ||||||
|  |             { | ||||||
|  |                 jsonLocalizationOptions.DisableGetLocalizerFromResourceManager = true; | ||||||
|  |                 jsonLocalizationOptions.DisableGetLocalizerFromService = true; | ||||||
|  |                 jsonLocalizationOptions.IgnoreLocalizerMissing = true; | ||||||
|  |                 jsonLocalizationOptions.UseKeyWhenValueIsNull = true; | ||||||
|  |             } | ||||||
|             ); |             ); | ||||||
|         services.AddConfigurableOptions<MenuOptions>(); |         services.AddConfigurableOptions<MenuOptions>(); | ||||||
|         services.ConfigureIconThemeOptions(options => options.ThemeKey = "fa"); |         services.ConfigureIconThemeOptions(options => options.ThemeKey = "fa"); | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
| 	</PropertyGroup> | 	</PropertyGroup> | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
| 		<PackageReference Include="BootstrapBlazor.FontAwesome" Version="9.0.2" /> | 		<PackageReference Include="BootstrapBlazor.FontAwesome" Version="9.0.2" /> | ||||||
| 		<PackageReference Include="BootstrapBlazor" Version="9.6.3" /> | 		<PackageReference Include="BootstrapBlazor" Version="9.7.0" /> | ||||||
| 		<PackageReference Include="Yitter.IdGenerator" Version="1.0.14" /> | 		<PackageReference Include="Yitter.IdGenerator" Version="1.0.14" /> | ||||||
| 	</ItemGroup> | 	</ItemGroup> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| //下载文件 | //下载文件 | ||||||
| export function blazor_downloadFile(url, fileName, dtoObject) { | export async function blazor_downloadFile(url, fileName, dtoObject) { | ||||||
|     const params = new URLSearchParams(); |     const params = new URLSearchParams(); | ||||||
|  |  | ||||||
|     // 将 dtoObject 的属性添加到 URLSearchParams 中 |     // 将 dtoObject 的属性添加到 URLSearchParams 中 | ||||||
| @@ -12,9 +12,10 @@ export function blazor_downloadFile(url, fileName, dtoObject) { | |||||||
|     // 构建完整的 URL |     // 构建完整的 URL | ||||||
|     const fullUrl = `${url}?${params.toString()}`; |     const fullUrl = `${url}?${params.toString()}`; | ||||||
|  |  | ||||||
|  |     try { | ||||||
|         // 发起 fetch 请求 |         // 发起 fetch 请求 | ||||||
|     fetch(fullUrl) |         const response = await fetch(fullUrl); | ||||||
|         .then(response => { |  | ||||||
|         // 获取响应头中的 content-disposition |         // 获取响应头中的 content-disposition | ||||||
|         const dispositionHeader = response.headers.get('content-disposition'); |         const dispositionHeader = response.headers.get('content-disposition'); | ||||||
|         let resolvedFileName = fileName; |         let resolvedFileName = fileName; | ||||||
| @@ -29,7 +30,8 @@ export function blazor_downloadFile(url, fileName, dtoObject) { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         // 将响应转换为 blob 对象 |         // 将响应转换为 blob 对象 | ||||||
|             return response.blob().then(blob => { |         const blob = await response.blob(); | ||||||
|  |  | ||||||
|         // 创建临时的文件 URL |         // 创建临时的文件 URL | ||||||
|         const fileUrl = window.URL.createObjectURL(blob); |         const fileUrl = window.URL.createObjectURL(blob); | ||||||
|  |  | ||||||
| @@ -46,35 +48,33 @@ export function blazor_downloadFile(url, fileName, dtoObject) { | |||||||
|  |  | ||||||
|         // 撤销临时的文件 URL |         // 撤销临时的文件 URL | ||||||
|         window.URL.revokeObjectURL(fileUrl); |         window.URL.revokeObjectURL(fileUrl); | ||||||
|             }); |  | ||||||
|         }) |         return true; | ||||||
|         .catch(error => { |     } catch (error) { | ||||||
|         console.error('DownFile error ', error); |         console.error('DownFile error ', error); | ||||||
|         }); |         throw error; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| //下载文件 | //下载文件 | ||||||
| export function postJson_downloadFile(url, fileName, jsonBody) { | export async function postJson_downloadFile(url, fileName, jsonBody) { | ||||||
|     const params = new URLSearchParams(); |  | ||||||
|  |  | ||||||
|  |     try { | ||||||
|     // 发起 fetch 请求 |         const response = await fetch(url, { | ||||||
|     fetch(url, { |  | ||||||
|             method: 'POST', |             method: 'POST', | ||||||
|             headers: { |             headers: { | ||||||
|                 'Content-Type': 'application/json' |                 'Content-Type': 'application/json' | ||||||
|             }, |             }, | ||||||
|             body: jsonBody |             body: jsonBody | ||||||
|     }) |         }); | ||||||
|         .then(response => { |  | ||||||
|             // 获取响应头中的 content-disposition |  | ||||||
|         const dispositionHeader = response.headers.get('content-disposition'); |         const dispositionHeader = response.headers.get('content-disposition'); | ||||||
|         let resolvedFileName = fileName; |         let resolvedFileName = fileName; | ||||||
|  |  | ||||||
|         if (dispositionHeader) { |         if (dispositionHeader) { | ||||||
|                 // 解析出文件名 |  | ||||||
|             const match = /filename[^;=\n]*=((['"]).*?\2|[^;\n]*)/.exec(dispositionHeader); |             const match = /filename[^;=\n]*=((['"]).*?\2|[^;\n]*)/.exec(dispositionHeader); | ||||||
|             const serverFileName = match && match[1] ? match[1].replace(/['"]/g, '') : null; |             const serverFileName = match && match[1] ? match[1].replace(/['"]/g, '') : null; | ||||||
|             if (serverFileName) { |             if (serverFileName) { | ||||||
| @@ -82,27 +82,22 @@ export function postJson_downloadFile(url, fileName, jsonBody) { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|             // 将响应转换为 blob 对象 |         const blob = await response.blob(); | ||||||
|             return response.blob().then(blob => { |  | ||||||
|                 // 创建临时的文件 URL |  | ||||||
|         const fileUrl = window.URL.createObjectURL(blob); |         const fileUrl = window.URL.createObjectURL(blob); | ||||||
|  |  | ||||||
|                 // 创建一个 <a> 元素并设置下载链接和文件名 |  | ||||||
|         const anchorElement = document.createElement('a'); |         const anchorElement = document.createElement('a'); | ||||||
|         anchorElement.href = fileUrl; |         anchorElement.href = fileUrl; | ||||||
|         anchorElement.download = resolvedFileName; |         anchorElement.download = resolvedFileName; | ||||||
|         anchorElement.style.display = 'none'; |         anchorElement.style.display = 'none'; | ||||||
|  |  | ||||||
|                 // 将 <a> 元素添加到 body 中并触发下载 |  | ||||||
|         document.body.appendChild(anchorElement); |         document.body.appendChild(anchorElement); | ||||||
|         anchorElement.click(); |         anchorElement.click(); | ||||||
|         document.body.removeChild(anchorElement); |         document.body.removeChild(anchorElement); | ||||||
|  |  | ||||||
|                 // 撤销临时的文件 URL |  | ||||||
|         window.URL.revokeObjectURL(fileUrl); |         window.URL.revokeObjectURL(fileUrl); | ||||||
|             }); |  | ||||||
|         }) |         return true; // 唯一新增的返回值 | ||||||
|         .catch(error => { |     } catch (error) { | ||||||
|         console.error('downfile error ', error); |         console.error('downfile error ', error); | ||||||
|         }); |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								src/Admin/ThingsGateway.SqlSugar/GlobalUsings.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								src/Admin/ThingsGateway.SqlSugar/GlobalUsings.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | global using ThingsGateway.NewLife.Extension; | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | using System.Security.Claims; | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
|  | public class ClaimsPrincipalService : IClaimsPrincipalService | ||||||
|  | { | ||||||
|  |  | ||||||
|  |     public ClaimsPrincipal? User => App.User; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,18 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | using System.Security.Claims; | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
|  | public interface IClaimsPrincipalService | ||||||
|  | { | ||||||
|  |     public ClaimsPrincipal? User { get; } | ||||||
|  | } | ||||||
| @@ -17,10 +17,10 @@ namespace ThingsGateway.Admin.Application; | |||||||
| 
 | 
 | ||||||
| public class SugarAopService : ISugarAopService | public class SugarAopService : ISugarAopService | ||||||
| { | { | ||||||
|     private IAppService _appService; |     private IClaimsPrincipalService _claimsPrincipalService; | ||||||
|     public SugarAopService(IAppService appService) |     public SugarAopService(IClaimsPrincipalService appService) | ||||||
|     { |     { | ||||||
|         _appService = appService; |         _claimsPrincipalService = appService; | ||||||
|     } |     } | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Aop设置 |     /// Aop设置 | ||||||
| @@ -85,7 +85,7 @@ public class SugarAopService : ISugarAopService | |||||||
|                 if (entityInfo.PropertyName == nameof(BaseEntity.CreateTime)) |                 if (entityInfo.PropertyName == nameof(BaseEntity.CreateTime)) | ||||||
|                     entityInfo.SetValue(DateTime.Now); |                     entityInfo.SetValue(DateTime.Now); | ||||||
| 
 | 
 | ||||||
|                 if (_appService.User != null) |                 if (_claimsPrincipalService.User != null) | ||||||
|                 { |                 { | ||||||
|                     //创建人 |                     //创建人 | ||||||
|                     if (entityInfo.PropertyName == nameof(BaseEntity.CreateUserId)) |                     if (entityInfo.PropertyName == nameof(BaseEntity.CreateUserId)) | ||||||
| @@ -103,7 +103,7 @@ public class SugarAopService : ISugarAopService | |||||||
|                 if (entityInfo.PropertyName == nameof(BaseEntity.UpdateTime)) |                 if (entityInfo.PropertyName == nameof(BaseEntity.UpdateTime)) | ||||||
|                     entityInfo.SetValue(DateTime.Now); |                     entityInfo.SetValue(DateTime.Now); | ||||||
|                 //更新人 |                 //更新人 | ||||||
|                 if (_appService.User != null) |                 if (_claimsPrincipalService.User != null) | ||||||
|                 { |                 { | ||||||
|                     if (entityInfo.PropertyName == nameof(BaseEntity.UpdateUserId)) |                     if (entityInfo.PropertyName == nameof(BaseEntity.UpdateUserId)) | ||||||
|                         entityInfo.SetValue(UserManager.UserId); |                         entityInfo.SetValue(UserManager.UserId); | ||||||
| @@ -117,6 +117,25 @@ public class SugarAopService : ISugarAopService | |||||||
|         db.Aop.DataExecuted = (value, entity) => |         db.Aop.DataExecuted = (value, entity) => | ||||||
|         { |         { | ||||||
|         }; |         }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         db.Aop.OnLogExecuted = (sql, pars) => | ||||||
|  |         { | ||||||
|  |             //执行时间超过1秒 | ||||||
|  |             if (db.Ado.SqlExecutionTime.TotalSeconds > 1) | ||||||
|  |             { | ||||||
|  |                 //代码CS文件名 | ||||||
|  |                 var fileName = db.Ado.SqlStackTrace.FirstFileName; | ||||||
|  |                 //代码行数 | ||||||
|  |                 var fileLine = db.Ado.SqlStackTrace.FirstLine; | ||||||
|  |                 //方法名 | ||||||
|  |                 var FirstMethodName = db.Ado.SqlStackTrace.FirstMethodName; | ||||||
|  | 
 | ||||||
|  |                 DbContext.WriteLog($"{fileName}-{FirstMethodName}-{fileLine} 执行时间超过1秒"); | ||||||
|  |                 DbContext.WriteLogWithSql(UtilMethods.GetNativeSql(sql, pars)); | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @@ -80,7 +80,9 @@ public static class DbContext | |||||||
|     { |     { | ||||||
|         db.CurrentConnectionConfig.MoreSettings = new ConnMoreSettings |         db.CurrentConnectionConfig.MoreSettings = new ConnMoreSettings | ||||||
|         { |         { | ||||||
|             SqlServerCodeFirstNvarchar = true//设置默认nvarchar |             SqlServerCodeFirstNvarchar = true, //设置默认nvarchar | ||||||
|  | 
 | ||||||
|  |             IsNoReadXmlDescription = true | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -37,7 +37,7 @@ public sealed class SqlSugarOption : ConnectionConfig | |||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 是否控制台显示Sql语句 |     /// 是否控制台显示Sql语句 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public bool IsShowSql { get; set; } |     public bool? IsShowSql { get; set; } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 更新数据 |     /// 更新数据 | ||||||
							
								
								
									
										44
									
								
								src/Admin/ThingsGateway.SqlSugar/Startup.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								src/Admin/ThingsGateway.SqlSugar/Startup.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | |||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||||
|  | //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||||
|  | //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||||
|  | //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||||
|  | //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||||
|  | //  使用文档:https://thingsgateway.cn/ | ||||||
|  | //  QQ群:605534569 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | using BootstrapBlazor.Components; | ||||||
|  |  | ||||||
|  | using Microsoft.AspNetCore.Builder; | ||||||
|  | using Microsoft.Extensions.DependencyInjection; | ||||||
|  |  | ||||||
|  | using SqlSugar; | ||||||
|  |  | ||||||
|  | namespace ThingsGateway.Admin.Application; | ||||||
|  |  | ||||||
|  | [AppStartup(1000000000)] | ||||||
|  | public class Startup : AppStartup | ||||||
|  | { | ||||||
|  |     public void Configure(IServiceCollection services) | ||||||
|  |     { | ||||||
|  |         services.AddConfigurableOptions<SqlSugarOptions>(); | ||||||
|  |  | ||||||
|  |         services.AddSingleton(typeof(IDataService<>), typeof(BaseService<>)); | ||||||
|  |         services.AddSingleton<ISugarAopService, SugarAopService>(); | ||||||
|  |         services.AddSingleton<ISugarConfigAopService, SugarConfigAopService>(); | ||||||
|  |  | ||||||
|  |         services.AddSingleton<IClaimsPrincipalService, ClaimsPrincipalService>(); | ||||||
|  |  | ||||||
|  |         StaticConfig.EnableAllWhereIF = true; | ||||||
|  |  | ||||||
|  |         services.AddSingleton<ISugarAopService, SugarAopService>(); | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void Use(IApplicationBuilder applicationBuilder) | ||||||
|  |     { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -17,33 +17,33 @@ namespace ThingsGateway.Admin.Application; | |||||||
| /// </summary> | /// </summary> | ||||||
| public static class UserManager | public static class UserManager | ||||||
| { | { | ||||||
|     private static readonly IAppService _appService; |     private static readonly IClaimsPrincipalService _claimsPrincipalService; | ||||||
|     static UserManager() |     static UserManager() | ||||||
|     { |     { | ||||||
|         _appService = App.RootServices.GetService<IAppService>(); |         _claimsPrincipalService = App.RootServices.GetService<IClaimsPrincipalService>(); | ||||||
|     } |     } | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 是否超级管理员 |     /// 是否超级管理员 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public static bool SuperAdmin => (_appService.User?.FindFirst(ClaimConst.SuperAdmin)?.Value).ToBoolean(false); |     public static bool SuperAdmin => (_claimsPrincipalService.User?.FindFirst(ClaimConst.SuperAdmin)?.Value).ToBoolean(false); | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 当前用户账号 |     /// 当前用户账号 | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public static string UserAccount => _appService.User?.FindFirst(ClaimConst.Account)?.Value; |     public static string UserAccount => _claimsPrincipalService.User?.FindFirst(ClaimConst.Account)?.Value; | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 当前用户Id |     /// 当前用户Id | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public static long UserId => (_appService.User?.FindFirst(ClaimConst.UserId)?.Value).ToLong(); |     public static long UserId => (_claimsPrincipalService.User?.FindFirst(ClaimConst.UserId)?.Value).ToLong(); | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 当前验证Id |     /// 当前验证Id | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public static long VerificatId => (_appService.User?.FindFirst(ClaimConst.VerificatId)?.Value).ToLong(); |     public static long VerificatId => (_claimsPrincipalService.User?.FindFirst(ClaimConst.VerificatId)?.Value).ToLong(); | ||||||
| 
 | 
 | ||||||
|     public static long OrgId => (_appService.User?.FindFirst(ClaimConst.OrgId)?.Value).ToLong(); |     public static long OrgId => (_claimsPrincipalService.User?.FindFirst(ClaimConst.OrgId)?.Value).ToLong(); | ||||||
| 
 | 
 | ||||||
|     public static long TenantId => (_appService.User?.FindFirst(ClaimConst.TenantId)?.Value)?.ToLong() ?? 0; |     public static long TenantId => (_claimsPrincipalService.User?.FindFirst(ClaimConst.TenantId)?.Value)?.ToLong() ?? 0; | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @@ -0,0 +1,29 @@ | |||||||
|  | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |  | ||||||
|  | 	<Import Project="$(SolutionDir)Version.props" /> | ||||||
|  | 	<Import Project="$(SolutionDir)PackNuget.props" /> | ||||||
|  |  | ||||||
|  | 	<PropertyGroup> | ||||||
|  | 		<GenerateDocumentationFile>True</GenerateDocumentationFile> | ||||||
|  | 	</PropertyGroup> | ||||||
|  | 	<PropertyGroup> | ||||||
|  | 		<TargetFrameworks>net8.0;net9.0;</TargetFrameworks> | ||||||
|  | 	</PropertyGroup> | ||||||
|  | 	 | ||||||
|  | 	<ItemGroup> | ||||||
|  | 		<PackageReference Include="SqlSugarCore" Version="5.1.4.193" /> | ||||||
|  | 		<PackageReference Include="BootstrapBlazor.TableExport" Version="9.2.5" /> | ||||||
|  | 	</ItemGroup> | ||||||
|  |  | ||||||
|  | 	<ItemGroup> | ||||||
|  | 		<None Include="..\README.md" Pack="true" PackagePath="\" /> | ||||||
|  | 		<None Include="..\README.zh-CN.md" Pack="true" PackagePath="\" /> | ||||||
|  | 		<None Remove="$(SolutionDir)..\README.md" Pack="false" PackagePath="\" /> | ||||||
|  | 		<None Remove="$(SolutionDir)..\README.zh-CN.md" Pack="false" PackagePath="\" /> | ||||||
|  | 	</ItemGroup> | ||||||
|  |  | ||||||
|  | 	<ItemGroup> | ||||||
|  | 		<ProjectReference Include="..\ThingsGateway.Razor\ThingsGateway.Razor.csproj" /> | ||||||
|  | 	</ItemGroup> | ||||||
|  |  | ||||||
|  | </Project> | ||||||
| @@ -1,9 +1,9 @@ | |||||||
| <Project> | <Project> | ||||||
|  |  | ||||||
| 	<PropertyGroup> | 	<PropertyGroup> | ||||||
| 		<PluginVersion>10.6.6</PluginVersion> | 		<PluginVersion>10.6.37</PluginVersion> | ||||||
| 		<ProPluginVersion>10.6.6</ProPluginVersion> | 		<ProPluginVersion>10.6.37</ProPluginVersion> | ||||||
| 		<AuthenticationVersion>2.1.7</AuthenticationVersion> | 		<AuthenticationVersion>2.1.8</AuthenticationVersion> | ||||||
| 	</PropertyGroup> | 	</PropertyGroup> | ||||||
|  |  | ||||||
| 	<PropertyGroup> | 	<PropertyGroup> | ||||||
|   | |||||||
| @@ -84,7 +84,7 @@ | |||||||
|                             </div> |                             </div> | ||||||
|                             <div class="col-12 col-md-4 p-1"> |                             <div class="col-12 col-md-4 p-1"> | ||||||
|  |  | ||||||
|                                 <BootstrapLabel Value=@(item.Value?.ToJsonNetString()) title=@(item.LastErrorMessage) class=@(item.IsOnline?"green--text":"red--text")>@(item.Value?.ToJsonNetString())</BootstrapLabel> |                                 <BootstrapLabel Value=@(item.Value?.ToSystemTextJsonString()) title=@(item.LastErrorMessage) class=@(item.IsOnline?"green--text":"red--text")>@(item.Value?.ToSystemTextJsonString())</BootstrapLabel> | ||||||
|  |  | ||||||
|                             </div> |                             </div> | ||||||
|                         </div> |                         </div> | ||||||
|   | |||||||
| @@ -90,7 +90,7 @@ public abstract class DeviceComponentBase : ComponentBase, IDisposable | |||||||
|                 var data = await Plc.ReadAsync(RegisterAddress, ArrayLength, DataType); |                 var data = await Plc.ReadAsync(RegisterAddress, ArrayLength, DataType); | ||||||
|                 if (data.IsSuccess) |                 if (data.IsSuccess) | ||||||
|                 { |                 { | ||||||
|                     Plc.Logger?.LogInformation(data.Content.ToJsonNetString()); |                     Plc.Logger?.LogInformation(data.Content.ToSystemTextJsonString()); | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|   | |||||||
| @@ -6,7 +6,9 @@ | |||||||
| @using BootstrapBlazor.Components | @using BootstrapBlazor.Components | ||||||
| @namespace ThingsGateway.Debug | @namespace ThingsGateway.Debug | ||||||
|  |  | ||||||
| <Card HeaderText=@HeaderText class=@("w-100") style=@($"{CardStyle}")> | <div class="w-100" style=@($"height:{HeightString}")> | ||||||
|  |  | ||||||
|  |     <Card HeaderText=@HeaderText class=@("w-100 h-100")> | ||||||
|     <HeaderTemplate> |     <HeaderTemplate> | ||||||
|         <div class="flex-fill"> |         <div class="flex-fill"> | ||||||
|         </div> |         </div> | ||||||
| @@ -36,7 +38,7 @@ | |||||||
|  |  | ||||||
|     </HeaderTemplate> |     </HeaderTemplate> | ||||||
|     <BodyTemplate> |     <BodyTemplate> | ||||||
|         <div style=@($"height:{HeightString};overflow-y:scroll")> |                 <div style=@($"height:calc(100% - 50px);overflow-y:scroll;flex-fill;")> | ||||||
|             <Virtualize Items="CurrentMessages??new  List<LogMessage>()" Context="itemMessage" ItemSize="60" OverscanCount=2> |             <Virtualize Items="CurrentMessages??new  List<LogMessage>()" Context="itemMessage" ItemSize="60" OverscanCount=2> | ||||||
|                 <ItemContent> |                 <ItemContent> | ||||||
|                     @*       <Tooltip Placement="Placement.Bottom" Title=@itemMessage.Message.Substring(0, Math.Min(itemMessage.Message.Length, 500))> *@ |                     @*       <Tooltip Placement="Placement.Bottom" Title=@itemMessage.Message.Substring(0, Math.Min(itemMessage.Message.Length, 500))> *@ | ||||||
| @@ -56,4 +58,4 @@ | |||||||
| </Card> | </Card> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | </div> | ||||||
|   | |||||||
| @@ -33,13 +33,11 @@ public partial class LogConsole : IDisposable | |||||||
|     [Parameter] |     [Parameter] | ||||||
|     public EventCallback<LogLevel> LogLevelChanged { get; set; } |     public EventCallback<LogLevel> LogLevelChanged { get; set; } | ||||||
|  |  | ||||||
|     [Parameter] |  | ||||||
|     public string CardStyle { get; set; } = "height: 100%;"; |  | ||||||
|     [Parameter] |     [Parameter] | ||||||
|     public string HeaderText { get; set; } = "Log"; |     public string HeaderText { get; set; } = "Log"; | ||||||
|  |  | ||||||
|     [Parameter] |     [Parameter] | ||||||
|     public string HeightString { get; set; } = "calc(100% - 50px)"; |     public string HeightString { get; set; } = "calc(100% - 300px)"; | ||||||
|  |  | ||||||
|     [Parameter, EditorRequired] |     [Parameter, EditorRequired] | ||||||
|     public string LogPath { get; set; } |     public string LogPath { get; set; } | ||||||
|   | |||||||
| @@ -71,8 +71,7 @@ public abstract class VariableObject | |||||||
|         var jToken = JToken.FromObject(value); |         var jToken = JToken.FromObject(value); | ||||||
|         if (!string.IsNullOrEmpty(variableRuntimeProperty.Attribute.WriteExpressions)) |         if (!string.IsNullOrEmpty(variableRuntimeProperty.Attribute.WriteExpressions)) | ||||||
|         { |         { | ||||||
|             object rawdata = jToken is JValue jValue ? jValue.Value : jToken is JArray jArray ? jArray : jToken.ToString(); |             object rawdata = jToken.GetObjectFromJToken(); | ||||||
|  |  | ||||||
|             object data = variableRuntimeProperty.Attribute.WriteExpressions.GetExpressionsResult(rawdata, Device?.Logger); |             object data = variableRuntimeProperty.Attribute.WriteExpressions.GetExpressionsResult(rawdata, Device?.Logger); | ||||||
|             jToken = JToken.FromObject(data); |             jToken = JToken.FromObject(data); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -28,11 +28,12 @@ public class DtuPlugin : PluginBase, ITcpReceivingPlugin | |||||||
|         set |         set | ||||||
|         { |         { | ||||||
|             _heartbeat = value; |             _heartbeat = value; | ||||||
|  |             if (!_heartbeat.IsNullOrEmpty()) | ||||||
|                 HeartbeatByte = new ArraySegment<byte>(Encoding.UTF8.GetBytes(value)); |                 HeartbeatByte = new ArraySegment<byte>(Encoding.UTF8.GetBytes(value)); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     private string _heartbeat; |     private string _heartbeat; | ||||||
|     private ArraySegment<byte> HeartbeatByte; |     private ArraySegment<byte> HeartbeatByte = new(); | ||||||
|  |  | ||||||
|     /// <inheritdoc/> |     /// <inheritdoc/> | ||||||
|     public async Task OnTcpReceiving(ITcpSession client, ByteBlockEventArgs e) |     public async Task OnTcpReceiving(ITcpSession client, ByteBlockEventArgs e) | ||||||
|   | |||||||
| @@ -61,14 +61,14 @@ internal sealed class HeartbeatAndReceivePlugin : PluginBase, ITcpConnectedPlugi | |||||||
|         { |         { | ||||||
|             return;//此处可判断,如果为服务器,则不用使用心跳。 |             return;//此处可判断,如果为服务器,则不用使用心跳。 | ||||||
|         } |         } | ||||||
|  |         if (HeartbeatTime > 0) | ||||||
|  |             SendHeartbeat = true; | ||||||
|         HeartbeatTime = Math.Max(HeartbeatTime, 1000); |         HeartbeatTime = Math.Max(HeartbeatTime, 1000); | ||||||
|  |  | ||||||
|         if (DtuId.IsNullOrWhiteSpace()) return; |         if (DtuId.IsNullOrWhiteSpace()) return; | ||||||
|  |  | ||||||
|         if (client is ITcpClient tcpClient) |         if (client is ITcpClient tcpClient) | ||||||
|         { |         { | ||||||
|             SendHeartbeat = true; |  | ||||||
|             await tcpClient.SendAsync(DtuIdByte).ConfigureAwait(false); |             await tcpClient.SendAsync(DtuIdByte).ConfigureAwait(false); | ||||||
|  |  | ||||||
|             if (Task == null) |             if (Task == null) | ||||||
|   | |||||||
| @@ -47,7 +47,7 @@ public static class PluginUtil | |||||||
|         Action<IPluginManager> action = a => { }; |         Action<IPluginManager> action = a => { }; | ||||||
|  |  | ||||||
|         action += GetTcpServicePlugin(channelOptions); |         action += GetTcpServicePlugin(channelOptions); | ||||||
|         if (!channelOptions.Heartbeat.IsNullOrWhiteSpace()) |         //if (!channelOptions.Heartbeat.IsNullOrWhiteSpace()) | ||||||
|         { |         { | ||||||
|             action += a => |             action += a => | ||||||
|             { |             { | ||||||
|   | |||||||
| @@ -8,17 +8,13 @@ | |||||||
| //  QQ群:605534569 | //  QQ群:605534569 | ||||||
| //------------------------------------------------------------------------------ | //------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | using System.Buffers; | ||||||
| using System.Text; | using System.Text; | ||||||
|  |  | ||||||
| using ThingsGateway.NewLife.Caching; | using ThingsGateway.NewLife.Caching; | ||||||
|  |  | ||||||
| namespace ThingsGateway.Foundation; | namespace ThingsGateway.Foundation; | ||||||
|  |  | ||||||
| public class LogDataCache |  | ||||||
| { |  | ||||||
|     public List<LogData> LogDatas { get; set; } |  | ||||||
|     public long Length { get; set; } |  | ||||||
| } |  | ||||||
| /// <summary> | /// <summary> | ||||||
| /// 日志数据 | /// 日志数据 | ||||||
| /// </summary> | /// </summary> | ||||||
| @@ -47,8 +43,19 @@ public class LogData | |||||||
|  |  | ||||||
|  |  | ||||||
| /// <summary>日志文本文件倒序读取</summary> | /// <summary>日志文本文件倒序读取</summary> | ||||||
|  |  | ||||||
|  | public class LogDataCache | ||||||
|  | { | ||||||
|  |     public List<LogData> LogDatas { get; set; } | ||||||
|  |     public long Length { get; set; } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// <summary>高性能日志文件读取器(支持倒序读取)</summary> | ||||||
| public class TextFileReader | public class TextFileReader | ||||||
| { | { | ||||||
|  |     private static readonly MemoryCache _cache = new() { Expire = 30 }; | ||||||
|  |     private static readonly MemoryCache _fileLocks = new(); | ||||||
|  |     private static readonly ArrayPool<byte> _bytePool = ArrayPool<byte>.Shared; | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// 获取指定目录下所有文件信息 |     /// 获取指定目录下所有文件信息 | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @@ -86,36 +93,56 @@ public class TextFileReader | |||||||
|         return result; |         return result; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     static MemoryCache _cache = new() { Expire = 30 }; |  | ||||||
|     public static OperResult<List<LogData>> LastLog(string file, int lineCount = 200) |     public static OperResult<List<LogData>> LastLog(string file, int lineCount = 200) | ||||||
|     { |     { | ||||||
|         lock (_cache) |         if (!File.Exists(file)) | ||||||
|         { |             return new OperResult<List<LogData>>("The file path is invalid"); | ||||||
|  |  | ||||||
|             OperResult<List<LogData>> result = new(); // 初始化结果对象 |         _fileLocks.SetExpire(file, TimeSpan.FromSeconds(30)); | ||||||
|  |         var fileLock = _fileLocks.GetOrAdd(file, _ => new object()); | ||||||
|  |         lock (fileLock) | ||||||
|  |         { | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 if (!File.Exists(file)) // 检查文件是否存在 |                 var fileInfo = new FileInfo(file); | ||||||
|  |                 var length = fileInfo.Length; | ||||||
|  |                 var cacheKey = $"{nameof(TextFileReader)}_{nameof(LastLog)}_{file})"; | ||||||
|  |                 if (_cache.TryGetValue<LogDataCache>(cacheKey, out var cachedData)) | ||||||
|                 { |                 { | ||||||
|                     result.OperCode = 999; |                     if (cachedData != null && cachedData.Length == length) | ||||||
|                     result.ErrorMessage = "The file path is invalid"; |                     { | ||||||
|                     return result; |                         return new OperResult<List<LogData>>() { Content = cachedData.LogDatas }; | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         _cache.Remove(cacheKey); | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 List<string> txt = new(); // 存储读取的文本内容 |                 using var fs = new FileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite, 4096, FileOptions.SequentialScan); | ||||||
|                 long ps = 0; // 保存起始位置 |                 var result = ReadLogsInverse(fs, lineCount, fileInfo.Length); | ||||||
|                 var key = $"{nameof(TextFileReader)}_{nameof(LastLog)}_{file})"; |  | ||||||
|                 long length = 0; |                 _cache.Set(cacheKey, new LogDataCache | ||||||
|                 using (var fs = new FileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) |                 { | ||||||
|  |                     LogDatas = result, | ||||||
|  |                     Length = fileInfo.Length, | ||||||
|  |                 }); | ||||||
|  |  | ||||||
|  |                 return new OperResult<List<LogData>>() { Content = result }; | ||||||
|  |             } | ||||||
|  |             catch (Exception ex) | ||||||
|  |             { | ||||||
|  |                 return new OperResult<List<LogData>>(ex); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static List<LogData> ReadLogsInverse(FileStream fs, int lineCount, long length) | ||||||
|     { |     { | ||||||
|         length = fs.Length; |         length = fs.Length; | ||||||
|                     var dataCache = _cache.Get<LogDataCache>(key); |         long ps = 0; // 保存起始位置 | ||||||
|                     if (dataCache != null && dataCache.Length == length) |         List<string> txt = new(); // 存储读取的文本内容 | ||||||
|                     { |  | ||||||
|                         result.Content = dataCache.LogDatas; |  | ||||||
|                         result.OperCode = 0; // 操作状态设为成功 |  | ||||||
|                         return result; // 返回解析结果 |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|         if (ps <= 0) // 如果起始位置小于等于0,将起始位置设置为文件长度 |         if (ps <= 0) // 如果起始位置小于等于0,将起始位置设置为文件长度 | ||||||
|             ps = length - 1; |             ps = length - 1; | ||||||
| @@ -128,10 +155,9 @@ public class TextFileReader | |||||||
|             if (ps <= 0) // 如果已经读取到文件开头则跳出循环 |             if (ps <= 0) // 如果已经读取到文件开头则跳出循环 | ||||||
|                 break; |                 break; | ||||||
|         } |         } | ||||||
|                 } |  | ||||||
|  |  | ||||||
|         // 使用单次 LINQ 操作进行过滤和解析 |         // 使用单次 LINQ 操作进行过滤和解析 | ||||||
|                 result.Content = txt |         var result = txt | ||||||
|               .Select(a => ParseCSV(a)) |               .Select(a => ParseCSV(a)) | ||||||
|                   .Where(data => data.Count >= 3) |                   .Where(data => data.Count >= 3) | ||||||
|                   .Select(data => |                   .Select(data => | ||||||
| @@ -147,98 +173,87 @@ public class TextFileReader | |||||||
|                   }) |                   }) | ||||||
|                   .ToList(); |                   .ToList(); | ||||||
|  |  | ||||||
|                 result.OperCode = 0; // 操作状态设为成功 |  | ||||||
|                 var data = _cache.Set<LogDataCache>(key, new LogDataCache() { Length = length, LogDatas = result.Content }); |  | ||||||
|  |  | ||||||
|         return result; // 返回解析结果 |         return result; // 返回解析结果 | ||||||
|     } |     } | ||||||
|             catch (Exception ex) // 捕获异常 |  | ||||||
|             { |  | ||||||
|                 result = new(ex); // 创建包含异常信息的结果对象 |  | ||||||
|                 return result; // 返回异常结果 |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static long InverseReadRow(FileStream fs, long position, out string value, int maxRead = 102400) |     private static long InverseReadRow(FileStream fs, long position, out string value, int maxRead = 102400) | ||||||
|     { |     { | ||||||
|         byte n = 0xD; // 换行符 |         byte n = 0xD; | ||||||
|         byte a = 0xA; // 回车符 |         byte a = 0xA; | ||||||
|         value = string.Empty; |         value = string.Empty; | ||||||
|         if (fs.Length == 0) return 0; // 若文件长度为0,则直接返回0作为新的位置 |  | ||||||
|  |         if (fs.Length == 0) return 0; | ||||||
|  |  | ||||||
|         var newPos = position; |         var newPos = position; | ||||||
|         List<byte> buffer = new List<byte>(maxRead); // 缓存读取的数据 |         byte[] buffer = _bytePool.Rent(maxRead); // 从池中租借字节数组 | ||||||
|  |         int index = 0; | ||||||
|  |  | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             var readLength = 0; |             while (true) | ||||||
|  |  | ||||||
|             while (true) // 循环读取一行数据,TextFileLogger.Separator行判定 |  | ||||||
|             { |             { | ||||||
|                 readLength++; |  | ||||||
|                 if (newPos <= 0) |                 if (newPos <= 0) | ||||||
|                     newPos = 0; |                     newPos = 0; | ||||||
|  |  | ||||||
|                 fs.Position = newPos; |                 fs.Position = newPos; | ||||||
|                 int byteRead = fs.ReadByte(); |                 int byteRead = fs.ReadByte(); | ||||||
|  |  | ||||||
|                 if (byteRead == -1) break; // 到达文件开头时跳出循环 |                 if (byteRead == -1) break; | ||||||
|  |  | ||||||
|                 buffer.Add((byte)byteRead); |                 if (index >= maxRead) | ||||||
|  |  | ||||||
|                 if (byteRead == n || byteRead == a)//判断当前字符是换行符 // TextFileLogger.Separator |  | ||||||
|                 { |  | ||||||
|                     if (MatchSeparator(buffer)) |  | ||||||
|                     { |  | ||||||
|                         // 去掉匹配的指定字符串 |  | ||||||
|                         buffer.RemoveRange(buffer.Count - TextFileLogger.SeparatorBytes.Length, TextFileLogger.SeparatorBytes.Length); |  | ||||||
|                         break; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if (buffer.Count > maxRead) // 超过最大字节数限制时丢弃数据 |  | ||||||
|                 { |                 { | ||||||
|                     newPos = -1; |                     newPos = -1; | ||||||
|                     return newPos; |                     return newPos; | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|  |                 buffer[index++] = (byte)byteRead; | ||||||
|  |  | ||||||
|  |                 if (byteRead == n || byteRead == a) | ||||||
|  |                 { | ||||||
|  |                     if (MatchSeparator(buffer, index)) | ||||||
|  |                     { | ||||||
|  |                         index -= TextFileLogger.SeparatorBytes.Length; | ||||||
|  |                         break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|                 newPos--; |                 newPos--; | ||||||
|                 if (newPos <= -1) |                 if (newPos <= -1) | ||||||
|                     break; |                     break; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (buffer.Count >= 10) |             if (index >= 10) | ||||||
|             { |             { | ||||||
|                 buffer.Reverse(); |                 Array.Reverse(buffer, 0, index); // 倒序 | ||||||
|                 value = Encoding.UTF8.GetString(buffer.ToArray()); // 转换为字符串 |                 value = Encoding.UTF8.GetString(buffer, 0, index); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             return newPos; // 返回新的读取位置 |             return newPos; | ||||||
|         } |         } | ||||||
|         finally |         finally | ||||||
|         { |         { | ||||||
|  |             _bytePool.Return(buffer); // 归还数组 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     private static bool MatchSeparator(List<byte> arr) |     private static bool MatchSeparator(byte[] arr, int length) | ||||||
|     { |  | ||||||
|         if (arr.Count < TextFileLogger.SeparatorBytes.Length) |  | ||||||
|     { |     { | ||||||
|  |         if (length < TextFileLogger.SeparatorBytes.Length) | ||||||
|             return false; |             return false; | ||||||
|         } |  | ||||||
|         var pos = arr.Count - 1; |         int pos = length - 1; | ||||||
|         for (int i = 0; i < TextFileLogger.SeparatorBytes.Length; i++) |         for (int i = 0; i < TextFileLogger.SeparatorBytes.Length; i++) | ||||||
|         { |         { | ||||||
|             if (arr[pos] != TextFileLogger.SeparatorBytes[i]) |             if (arr[pos] != TextFileLogger.SeparatorBytes[i]) | ||||||
|             { |  | ||||||
|                 return false; |                 return false; | ||||||
|             } |  | ||||||
|             pos--; |             pos--; | ||||||
|         } |         } | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     private static List<string> ParseCSV(string data) |     private static List<string> ParseCSV(string data) | ||||||
|     { |     { | ||||||
|         List<string> items = new List<string>(); |         List<string> items = new List<string>(); | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user