mirror of
				https://gitee.com/ThingsGateway/ThingsGateway.git
				synced 2025-10-31 15:43:59 +08:00 
			
		
		
		
	Compare commits
	
		
			24 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | b989aa5561 | ||
|   | f5b0b7ebd2 | ||
|   | 16881ae076 | ||
|   | af04112656 | ||
|   | a2863112dc | ||
|   | f531e4dfc5 | ||
|   | 8db9b32ba7 | ||
|   | dd5691cbef | ||
|   | de48b32af3 | ||
|   | 600b5042a1 | ||
|   | aac77029da | ||
|   | e50205f557 | ||
|   | e227411d1f | ||
|   | 2de0ed793f | ||
|   | cb0276f273 | ||
|   | 562b3f17c9 | ||
|   | 0f78f81c1c | ||
|   | 6594937d0a | ||
|   | 64bc6084be | ||
|   | 20434d5bd2 | ||
|   | 9ecc9380e6 | ||
|   | a57c55080b | ||
|   | b8ca06c6be | ||
|   | 5aff6461a1 | 
| @@ -64,24 +64,31 @@ public sealed class OperDescAttribute : MoAttribute | ||||
|  | ||||
|     public override void OnException(MethodContext context) | ||||
|     { | ||||
|         //插入异常日志 | ||||
|         SysOperateLog log = GetOperLog(LocalizerType, context); | ||||
|         if (App.HttpContext.Request.Path.StartsWithSegments("/_blazor")) | ||||
|         { | ||||
|             //插入异常日志 | ||||
|             SysOperateLog log = GetOperLog(LocalizerType, context); | ||||
|  | ||||
|         log.Category = LogCateGoryEnum.Exception;//操作类型为异常 | ||||
|         log.ExeStatus = false;//操作状态为失败 | ||||
|         if (context.Exception is AppFriendlyException exception) | ||||
|             log.ExeMessage = exception?.Message; | ||||
|         else | ||||
|             log.ExeMessage = context.Exception?.ToString(); | ||||
|             log.Category = LogCateGoryEnum.Exception;//操作类型为异常 | ||||
|             log.ExeStatus = false;//操作状态为失败 | ||||
|             if (context.Exception is AppFriendlyException exception) | ||||
|                 log.ExeMessage = exception?.Message; | ||||
|             else | ||||
|                 log.ExeMessage = context.Exception?.ToString(); | ||||
|  | ||||
|         OperDescAttribute.WriteToQueue(log); | ||||
|             OperDescAttribute.WriteToQueue(log); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public override void OnSuccess(MethodContext context) | ||||
|     { | ||||
|         //插入操作日志 | ||||
|         SysOperateLog log = GetOperLog(LocalizerType, context); | ||||
|         OperDescAttribute.WriteToQueue(log); | ||||
|         if (App.HttpContext.Request.Path.StartsWithSegments("/_blazor")) | ||||
|         { | ||||
|  | ||||
|             //插入操作日志 | ||||
|             SysOperateLog log = GetOperLog(LocalizerType, context); | ||||
|             OperDescAttribute.WriteToQueue(log); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
| @@ -115,7 +122,7 @@ public sealed class OperDescAttribute : MoAttribute | ||||
|     private SysOperateLog GetOperLog(Type? localizerType, MethodContext context) | ||||
|     { | ||||
|         var methodBase = context.Method; | ||||
|         var clientInfo = AppService.ClientInfo; | ||||
|         var userAgent = AppService.UserAgent; | ||||
|         string? paramJson = null; | ||||
|         if (IsRecordPar) | ||||
|         { | ||||
| @@ -127,10 +134,10 @@ public sealed class OperDescAttribute : MoAttribute | ||||
|             { | ||||
|                 parametersDict[parametersInfo[i].Name!] = args[i]; | ||||
|             } | ||||
|             paramJson = parametersDict.ToJsonNetString(); | ||||
|             paramJson = parametersDict.ToSystemTextJsonString(); | ||||
|         } | ||||
|         var result = context.ReturnValue; | ||||
|         var resultJson = IsRecordPar ? result?.ToJsonNetString() : null; | ||||
|         var resultJson = IsRecordPar ? result?.ToSystemTextJsonString() : null; | ||||
|         //操作日志表实体 | ||||
|         var log = new SysOperateLog | ||||
|         { | ||||
| @@ -138,8 +145,8 @@ public sealed class OperDescAttribute : MoAttribute | ||||
|             Category = LogCateGoryEnum.Operate, | ||||
|             ExeStatus = true, | ||||
|             OpIp = AppService?.RemoteIpAddress ?? string.Empty, | ||||
|             OpBrowser = clientInfo?.UA?.Family + clientInfo?.UA?.Major, | ||||
|             OpOs = clientInfo?.OS?.Family + clientInfo?.OS?.Major, | ||||
|             OpBrowser = userAgent?.Browser, | ||||
|             OpOs = userAgent?.Platform, | ||||
|             OpTime = DateTime.Now, | ||||
|             OpAccount = UserManager.UserAccount, | ||||
|             ReqUrl = null, | ||||
|   | ||||
| @@ -15,7 +15,7 @@ namespace ThingsGateway.Admin.Application; | ||||
|  | ||||
| [ApiDescriptionSettings(false)] | ||||
| [Route("api/auth")] | ||||
| [LoggingMonitor] | ||||
| [RequestAudit] | ||||
| public class AuthController : ControllerBase | ||||
| { | ||||
|     private readonly IAuthService _authService; | ||||
|   | ||||
| @@ -25,7 +25,7 @@ namespace ThingsGateway.Admin.Application; | ||||
| [Description("登录")] | ||||
| [Route("openapi/auth")] | ||||
| [Authorize(AuthenticationSchemes = "Bearer")] | ||||
| [LoggingMonitor] | ||||
| [RequestAudit] | ||||
| [ApiController] | ||||
| 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,290 @@ | ||||
| //------------------------------------------------------------------------------ | ||||
| //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||
| //  此代码版权(除特别声明外的代码)归作者本人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; | ||||
|  | ||||
| namespace ThingsGateway.Admin.Application; | ||||
|  | ||||
| public class RequestAuditFilter : IAsyncActionFilter | ||||
| { | ||||
|     public async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next) | ||||
|     { | ||||
|  | ||||
|         var controllerActionDescriptor = (context.ActionDescriptor as ControllerActionDescriptor); | ||||
|         // 获取动作方法描述器 | ||||
|         var actionMethod = controllerActionDescriptor?.MethodInfo; | ||||
|  | ||||
|         // 处理 Blazor Server | ||||
|         if (actionMethod == null) | ||||
|         { | ||||
|             _ = await next.Invoke().ConfigureAwait(false); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // 排除 WebSocket 请求处理 | ||||
|         if (context.HttpContext.IsWebSocketRequest()) | ||||
|         { | ||||
|             _ = await next().ConfigureAwait(false); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // 如果贴了 [SuppressMonitor] 特性则跳过 | ||||
|         if (actionMethod.IsDefined(typeof(SuppressRequestAuditAttribute), true) | ||||
|             || actionMethod.DeclaringType.IsDefined(typeof(SuppressRequestAuditAttribute), true)) | ||||
|         { | ||||
|             _ = await next().ConfigureAwait(false); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|  | ||||
|         // 只有方法贴有特性才进行审计 | ||||
|         if ( | ||||
|             !actionMethod.DeclaringType.IsDefined(typeof(RequestAuditAttribute), true) | ||||
|             && | ||||
|             !actionMethod.IsDefined(typeof(RequestAuditAttribute), true)) | ||||
|         { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // 计算接口执行时间 | ||||
|         var timeOperation = Stopwatch.StartNew(); | ||||
|         var resultContext = await next().ConfigureAwait(false); | ||||
|         timeOperation.Stop(); | ||||
|  | ||||
|  | ||||
|         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; | ||||
|  | ||||
|  | ||||
|         var result = resultContext.Result as IActionResult; | ||||
|         var data = result switch | ||||
|         { | ||||
|             // 处理内容结果 | ||||
|             ContentResult content => content.Content, | ||||
|             // 处理对象结果 | ||||
|             ObjectResult obj => obj.Value, | ||||
|             // 处理 JSON 对象 | ||||
|             JsonResult json => json.Value, | ||||
|             _ => null, | ||||
|         }; | ||||
|         logData.ReturnInformation = data; | ||||
|  | ||||
|  | ||||
|  | ||||
|         //获取客户端信息 | ||||
|         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> | ||||
|     [AutoGenerateColumn(Ignore = true)] | ||||
|     public string WorkingSet { get; set; } | ||||
|     public int WorkingSet { get; set; } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 更新时间 | ||||
|   | ||||
| @@ -17,6 +17,7 @@ using System.Runtime.InteropServices; | ||||
|  | ||||
| using ThingsGateway.Extension; | ||||
| using ThingsGateway.NewLife; | ||||
| using ThingsGateway.NewLife.Caching; | ||||
| using ThingsGateway.NewLife.Threading; | ||||
| using ThingsGateway.Schedule; | ||||
|  | ||||
| @@ -51,11 +52,20 @@ public class HardwareJob : IJob, IHardwareJob | ||||
|  | ||||
|     #endregion 属性 | ||||
|  | ||||
|     private MemoryCache MemoryCache = new() { }; | ||||
|     private const string CacheKey = "HistoryHardwareInfo"; | ||||
|     /// <inheritdoc/> | ||||
|     public async Task<List<HistoryHardwareInfo>> GetHistoryHardwareInfos() | ||||
|     { | ||||
|         using var db = DbContext.Db.GetConnectionScopeWithAttr<HistoryHardwareInfo>().CopyNew(); | ||||
|         return await db.Queryable<HistoryHardwareInfo>().ToListAsync().ConfigureAwait(false); | ||||
|         var historyHardwareInfos = MemoryCache.Get<List<HistoryHardwareInfo>>(CacheKey); | ||||
|         if (historyHardwareInfos == null) | ||||
|         { | ||||
|             using var db = DbContext.Db.GetConnectionScopeWithAttr<HistoryHardwareInfo>().CopyNew(); | ||||
|             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; | ||||
| @@ -94,7 +104,7 @@ public class HardwareJob : IJob, IHardwareJob | ||||
|             { | ||||
|                 HardwareInfo.MachineInfo.Refresh(); | ||||
|                 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; | ||||
|             } | ||||
|             catch (Exception ex) | ||||
| @@ -116,17 +126,22 @@ public class HardwareJob : IJob, IHardwareJob | ||||
|                             var his = new HistoryHardwareInfo() | ||||
|                             { | ||||
|                                 Date = TimerX.Now, | ||||
|                                 DriveUsage = (100 - (HardwareInfo.DriveInfo.TotalFreeSpace * 100.00 / HardwareInfo.DriveInfo.TotalSize)).ToString("F2"), | ||||
|                                 Battery = (HardwareInfo.MachineInfo.Battery * 100).ToString("F2"), | ||||
|                                 DriveUsage = (100 - (HardwareInfo.DriveInfo.TotalFreeSpace * 100.00 / HardwareInfo.DriveInfo.TotalSize)).ToInt(), | ||||
|                                 Battery = (HardwareInfo.MachineInfo.Battery * 100).ToInt(), | ||||
|                                 MemoryUsage = (HardwareInfo.WorkingSet), | ||||
|                                 CpuUsage = (HardwareInfo.MachineInfo.CpuRate * 100).ToString("F2"), | ||||
|                                 Temperature = (HardwareInfo.MachineInfo.Temperature).ToString("F2"), | ||||
|                                 CpuUsage = (HardwareInfo.MachineInfo.CpuRate * 100).ToInt(), | ||||
|                                 Temperature = (HardwareInfo.MachineInfo.Temperature).ToInt(), | ||||
|                             }; | ||||
|                             await db.Insertable(his).ExecuteCommandAsync(stoppingToken).ConfigureAwait(false); | ||||
|                             MemoryCache.Remove(CacheKey); | ||||
|                         } | ||||
|                         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; | ||||
|   | ||||
| @@ -19,23 +19,23 @@ public class HistoryHardwareInfo | ||||
| { | ||||
|     /// <inheritdoc/> | ||||
|     [SugarColumn(ColumnDescription = "磁盘使用率")] | ||||
|     public string DriveUsage { get; set; } | ||||
|     public int DriveUsage { get; set; } | ||||
|  | ||||
|     /// <inheritdoc/> | ||||
|     [SugarColumn(ColumnDescription = "内存")] | ||||
|     public string MemoryUsage { get; set; } | ||||
|     public int MemoryUsage { get; set; } | ||||
|  | ||||
|     /// <inheritdoc/> | ||||
|     [SugarColumn(ColumnDescription = "CPU使用率")] | ||||
|     public string CpuUsage { get; set; } | ||||
|     public int CpuUsage { get; set; } | ||||
|  | ||||
|     /// <inheritdoc/> | ||||
|     [SugarColumn(ColumnDescription = "温度")] | ||||
|     public string Temperature { get; set; } | ||||
|     public int Temperature { get; set; } | ||||
|  | ||||
|     /// <inheritdoc/> | ||||
|     [SugarColumn(ColumnDescription = "电池")] | ||||
|     public string Battery { get; set; } | ||||
|     public int Battery { get; set; } | ||||
|  | ||||
|     /// <inheritdoc/> | ||||
|     [SugarColumn(ColumnDescription = "时间")] | ||||
|   | ||||
| @@ -18,8 +18,6 @@ using ThingsGateway.Logging; | ||||
| using ThingsGateway.NewLife.Json.Extension; | ||||
| using ThingsGateway.Razor; | ||||
|  | ||||
| using UAParser; | ||||
|  | ||||
| namespace ThingsGateway.Admin.Application; | ||||
|  | ||||
| /// <summary> | ||||
| @@ -41,33 +39,31 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | ||||
|     /// <param name="flush"></param> | ||||
|     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; | ||||
|         // loggingMonitor.ReturnInformation.Value | ||||
|         requestAuditData.LogDateTime = logMsg.LogDateTime; | ||||
|         // requestAuditData.ReturnInformation.Value | ||||
|         //验证失败不记录日志 | ||||
|         bool save = false; | ||||
|         if (loggingMonitor.Validation == null) | ||||
|         if (requestAuditData.Validation == null) | ||||
|         { | ||||
|             var operation = logMsg.Context.Get(LoggingConst.Operation).ToString();//获取操作名称 | ||||
|             var client = (ClientInfo)logMsg.Context.Get(LoggingConst.Client);//获取客户端信息 | ||||
|             var path = logMsg.Context.Get(LoggingConst.Path).ToString();//获取操作名称 | ||||
|             var method = logMsg.Context.Get(LoggingConst.Method).ToString();//获取方法 | ||||
|             var operation = requestAuditData.Operation;//获取操作名称 | ||||
|             var client = requestAuditData.Client;//获取客户端信息 | ||||
|             var path = requestAuditData.Path;//获取操作名称 | ||||
|             var method = requestAuditData.Method;//获取方法 | ||||
|             //表示访问日志 | ||||
|             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 | ||||
|                 { | ||||
|                     //添加到异常日志 | ||||
|                     save = await CreateOperationLog(operation, path, loggingMonitor, client, flush).ConfigureAwait(false); | ||||
|                     save = await CreateOperationLog(operation, path, requestAuditData, client, flush).ConfigureAwait(false); | ||||
|                 } | ||||
|             } | ||||
|             else | ||||
| @@ -76,7 +72,7 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | ||||
|                 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 +87,21 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | ||||
|     /// </summary> | ||||
|     /// <param name="operation">操作名称</param> | ||||
|     /// <param name="path">请求地址</param> | ||||
|     /// <param name="loggingMonitor">loggingMonitor</param> | ||||
|     /// <param name="clientInfo">客户端信息</param> | ||||
|     /// <param name="requestAuditData">requestAuditData</param> | ||||
|     /// <param name="userAgent">客户端信息</param> | ||||
|     /// <param name="flush"></param> | ||||
|     /// <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字符串, | ||||
|         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字符串 | ||||
|         var resultJson = string.Empty; | ||||
|         if (loggingMonitor.ReturnInformation != null)//如果有返回值 | ||||
|         { | ||||
|             if (loggingMonitor.ReturnInformation.Value != null)//如果返回值不为空 | ||||
|             { | ||||
|                 resultJson = loggingMonitor.ReturnInformation.Value.ToJsonNetString(); | ||||
|             } | ||||
|         } | ||||
|         var resultJson = requestAuditData.ReturnInformation?.ToSystemTextJsonString(); | ||||
|  | ||||
|  | ||||
|         //操作日志表实体 | ||||
|         var sysLogOperate = new SysOperateLog | ||||
| @@ -119,29 +109,29 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | ||||
|             Name = operation, | ||||
|             Category = LogCateGoryEnum.Operate, | ||||
|             ExeStatus = true, | ||||
|             OpIp = loggingMonitor.RemoteIPv4, | ||||
|             OpBrowser = clientInfo?.UA?.Family + clientInfo?.UA?.Major, | ||||
|             OpOs = clientInfo?.OS?.Family + clientInfo?.OS?.Major, | ||||
|             OpTime = loggingMonitor.LogDateTime.LocalDateTime, | ||||
|             OpIp = requestAuditData.RemoteIPv4, | ||||
|             OpBrowser = userAgent?.Browser, | ||||
|             OpOs = userAgent?.Platform, | ||||
|             OpTime = requestAuditData.LogDateTime.LocalDateTime, | ||||
|             OpAccount = opAccount, | ||||
|             ReqMethod = loggingMonitor.HttpMethod, | ||||
|             ReqMethod = requestAuditData.Method, | ||||
|             ReqUrl = path, | ||||
|             ResultJson = resultJson, | ||||
|             ClassName = loggingMonitor.DisplayName, | ||||
|             MethodName = loggingMonitor.ActionName, | ||||
|             ClassName = requestAuditData.ControllerName, | ||||
|             MethodName = requestAuditData.ActionName, | ||||
|             ParamJson = paramJson, | ||||
|             VerificatId = UserManager.VerificatId, | ||||
|         }; | ||||
|         //如果异常不为空 | ||||
|         if (loggingMonitor.Exception != null) | ||||
|         if (requestAuditData.Exception != null) | ||||
|         { | ||||
|             sysLogOperate.Category = LogCateGoryEnum.Exception;//操作类型为异常 | ||||
|             sysLogOperate.ExeStatus = false;//操作状态为失败 | ||||
|  | ||||
|             if (loggingMonitor.Exception.Type == typeof(AppFriendlyException).ToString()) | ||||
|                 sysLogOperate.ExeMessage = loggingMonitor?.Exception.Message; | ||||
|             if (requestAuditData.Exception.Type == typeof(AppFriendlyException).ToString()) | ||||
|                 sysLogOperate.ExeMessage = requestAuditData?.Exception.Message; | ||||
|             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); | ||||
| @@ -160,17 +150,17 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | ||||
|     /// </summary> | ||||
|     /// <param name="operation">访问类型</param> | ||||
|     /// <param name="path"></param> | ||||
|     /// <param name="loggingMonitor">loggingMonitor</param> | ||||
|     /// <param name="clientInfo">客户端信息</param> | ||||
|     /// <param name="requestAuditData">requestAuditData</param> | ||||
|     /// <param name="userAgent">客户端信息</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 | ||||
|         var opAccount = "";//用户账号 | ||||
|         if (path == "/api/auth/login") | ||||
|         { | ||||
|             //如果是登录,用户信息就从返回值里拿 | ||||
|             var result = loggingMonitor.ReturnInformation?.Value?.ToJsonNetString();//返回值转json | ||||
|             var result = requestAuditData.ReturnInformation?.ToSystemTextJsonString();//返回值转json | ||||
|             var userInfo = result.FromJsonNetString<UnifyResult<LoginOutput>>();//格式化成user表 | ||||
|             opAccount = userInfo.Data.Account;//赋值账号 | ||||
|             verificatId = userInfo.Data.VerificatId; | ||||
| @@ -178,8 +168,8 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | ||||
|         else | ||||
|         { | ||||
|             //如果是登录出,用户信息就从AuthorizationClaims里拿 | ||||
|             opAccount = loggingMonitor.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(); | ||||
|             opAccount = requestAuditData.AuthorizationClaims.Where(it => it.Type == ClaimConst.Account).Select(it => it.Value).FirstOrDefault(); | ||||
|             verificatId = requestAuditData.AuthorizationClaims.Where(it => it.Type == ClaimConst.VerificatId).Select(it => it.Value).FirstOrDefault().ToLong(); | ||||
|         } | ||||
|         //日志表实体 | ||||
|         var sysLogVisit = new SysOperateLog | ||||
| @@ -187,19 +177,19 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter | ||||
|             Name = operation, | ||||
|             Category = path == "/api/auth/login" ? LogCateGoryEnum.Login : LogCateGoryEnum.Logout, | ||||
|             ExeStatus = true, | ||||
|             OpIp = loggingMonitor.RemoteIPv4, | ||||
|             OpBrowser = clientInfo?.UA?.Family + clientInfo?.UA?.Major, | ||||
|             OpOs = clientInfo?.OS?.Family + clientInfo?.OS?.Major, | ||||
|             OpTime = loggingMonitor.LogDateTime.LocalDateTime, | ||||
|             OpIp = requestAuditData.RemoteIPv4, | ||||
|             OpBrowser = userAgent?.Browser, | ||||
|             OpOs = userAgent?.Platform, | ||||
|             OpTime = requestAuditData.LogDateTime.LocalDateTime, | ||||
|             VerificatId = verificatId, | ||||
|             OpAccount = opAccount, | ||||
|  | ||||
|             ReqMethod = loggingMonitor.HttpMethod, | ||||
|             ReqMethod = requestAuditData.Method, | ||||
|             ReqUrl = path, | ||||
|             ResultJson = loggingMonitor.ReturnInformation?.Value?.ToJsonNetString(), | ||||
|             ClassName = loggingMonitor.DisplayName, | ||||
|             MethodName = loggingMonitor.ActionName, | ||||
|             ParamJson = loggingMonitor.Parameters?.ToJsonNetString(), | ||||
|             ResultJson = requestAuditData.ReturnInformation?.ToSystemTextJsonString(), | ||||
|             ClassName = requestAuditData.ControllerName, | ||||
|             MethodName = requestAuditData.ActionName, | ||||
|             ParamJson = requestAuditData.Parameters?.ToSystemTextJsonString(), | ||||
|         }; | ||||
|         _operateLogMessageQueue.Enqueue(sysLogVisit); | ||||
|  | ||||
|   | ||||
| @@ -15,12 +15,15 @@ using Microsoft.AspNetCore.WebUtilities; | ||||
|  | ||||
| using System.Security.Claims; | ||||
|  | ||||
| using UAParser; | ||||
|  | ||||
| namespace ThingsGateway.Admin.Application; | ||||
|  | ||||
| public class AppService : IAppService | ||||
| { | ||||
|     private readonly IUserAgentService UserAgentService; | ||||
|     public AppService(IUserAgentService userAgentService) | ||||
|     { | ||||
|         UserAgentService = userAgentService; | ||||
|     } | ||||
|     public string GetReturnUrl(string returnUrl) | ||||
|     { | ||||
|         var url = QueryHelpers.AddQueryString(CookieAuthenticationDefaults.LoginPath, new Dictionary<string, string?> | ||||
| @@ -41,18 +44,16 @@ public class AppService : IAppService | ||||
|         { | ||||
|         } | ||||
|     } | ||||
|     public Parser Parser = Parser.GetDefault(); | ||||
|     public ClientInfo? ClientInfo | ||||
|     public UserAgent? UserAgent | ||||
|     { | ||||
|         get | ||||
|         { | ||||
|             var str = App.HttpContext?.Request?.Headers?.UserAgent; | ||||
|             ClientInfo? clientInfo = null; | ||||
|             if (!string.IsNullOrEmpty(str)) | ||||
|             { | ||||
|                 clientInfo = Parser.Parse(str); | ||||
|                 return UserAgentService.Parse(str); | ||||
|             } | ||||
|             return clientInfo; | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -13,19 +13,17 @@ using Microsoft.Extensions.DependencyInjection; | ||||
|  | ||||
| using System.Security.Claims; | ||||
|  | ||||
| using UAParser; | ||||
|  | ||||
| namespace ThingsGateway.Admin.Application; | ||||
|  | ||||
| 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"; | ||||
|         ClientInfo = Parser.GetDefault().Parse(str); | ||||
|         UserAgent = userAgentService.Parse(str); | ||||
|         RemoteIpAddress = "127.0.0.1"; | ||||
|     } | ||||
|     public ClientInfo? ClientInfo { get; } | ||||
|     public UserAgent? UserAgent { get; } | ||||
|  | ||||
|     private static BlazorHybridAuthenticationStateProvider _authenticationStateProvider; | ||||
|     private static BlazorHybridAuthenticationStateProvider AuthenticationStateProvider | ||||
|   | ||||
| @@ -11,8 +11,6 @@ | ||||
|  | ||||
| using System.Security.Claims; | ||||
|  | ||||
| using UAParser; | ||||
|  | ||||
| namespace ThingsGateway.Admin.Application; | ||||
|  | ||||
| public interface IAppService | ||||
| @@ -20,7 +18,7 @@ public interface IAppService | ||||
|     /// <summary> | ||||
|     /// ClientInfo | ||||
|     /// </summary> | ||||
|     public ClientInfo? ClientInfo { get; } | ||||
|     public UserAgent? UserAgent { get; } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// ClaimsPrincipal | ||||
|   | ||||
| @@ -237,7 +237,7 @@ public class AuthService : IAuthService | ||||
|         var logingEvent = new LoginEvent | ||||
|         { | ||||
|             Ip = _appService.RemoteIpAddress, | ||||
|             Device = App.GetService<IAppService>().ClientInfo?.OS?.ToString(), | ||||
|             Device = App.GetService<IAppService>().UserAgent?.Platform, | ||||
|             Expire = expire, | ||||
|             SysUser = sysUser, | ||||
|             VerificatId = verificatId | ||||
|   | ||||
| @@ -77,7 +77,7 @@ internal sealed class SysDictService : BaseService<SysDict>, ISysDictService | ||||
|         //更新数据 | ||||
|         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); | ||||
|  | ||||
|   | ||||
| @@ -16,9 +16,9 @@ namespace ThingsGateway.Admin.Application; | ||||
| /// 内存推送事件服务 | ||||
| /// </summary> | ||||
| /// <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() | ||||
|     { | ||||
|   | ||||
| @@ -277,7 +277,7 @@ internal sealed class SysRoleService : BaseService<SysRole>, ISysRoleService | ||||
|         if (isSuperAdmin) | ||||
|             throw Oops.Bah(Localizer["CanotGrantAdmin"]); | ||||
|         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 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 | ||||
|                     { | ||||
|                         ApiUrl = it.ApiRoute, | ||||
|                     }.ToJsonNetString() | ||||
|                     }.ToSystemTextJsonString() | ||||
|                 }); | ||||
|                 relationRoles.AddRange(relationRolePer);//合并列表 | ||||
|             } | ||||
| @@ -410,7 +410,7 @@ internal sealed class SysRoleService : BaseService<SysRole>, ISysRoleService | ||||
|         if (sysRole != null) | ||||
|         { | ||||
|             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);//添加到数据库 | ||||
|             await ClearTokenUtil.DeleteUserCacheByRoleIds(new List<long> { input.Id }).ConfigureAwait(false);//清除角色下用户缓存 | ||||
|         } | ||||
|   | ||||
| @@ -435,7 +435,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService | ||||
|         if (sysUser != null) | ||||
|         { | ||||
|             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);//添加到数据库 | ||||
|             DeleteUserFromCache(input.Id); | ||||
|         } | ||||
| @@ -557,7 +557,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService | ||||
|     public async Task GrantResourceAsync(GrantResourceData input) | ||||
|     { | ||||
|         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 sysUser = await GetUserByIdAsync(input.Id).ConfigureAwait(false);//获取用户 | ||||
|         await CheckApiDataScopeAsync(sysUser.OrgId, sysUser.CreateUserId).ConfigureAwait(false); | ||||
| @@ -613,7 +613,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService | ||||
|                     TargetId = it.ApiRoute, | ||||
|                     Category = RelationCategoryEnum.UserHasPermission, | ||||
|                     ExtJson = new RelationPermission { ApiUrl = it.ApiRoute } | ||||
|                             .ToJsonNetString() | ||||
|                             .ToSystemTextJsonString() | ||||
|                 }); | ||||
|                 relationUsers.AddRange(relationUserPer);//合并列表 | ||||
|             } | ||||
|   | ||||
| @@ -203,7 +203,7 @@ internal sealed class UserCenterService : BaseService<SysUser>, IUserCenterServi | ||||
|     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); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -10,9 +10,6 @@ | ||||
|  | ||||
| using SqlSugar; | ||||
|  | ||||
| using ThingsGateway.List; | ||||
| using ThingsGateway.NewLife.Json.Extension; | ||||
|  | ||||
| namespace ThingsGateway.Admin.Application; | ||||
|  | ||||
| /// <summary> | ||||
| @@ -169,7 +166,7 @@ internal sealed class VerificatInfoService : BaseService<VerificatInfo>, IVerifi | ||||
|     public void RemoveAllClientId() | ||||
|     { | ||||
|         using var db = GetDB(); | ||||
|         db.Updateable<VerificatInfo>().SetColumns(a=>a.ClientIds==null).Where(a => a.Id > 0).ExecuteCommand(); | ||||
|         db.Updateable<VerificatInfo>().SetColumns(a => a.ClientIds == null).Where(a => a.Id > 0).ExecuteCommand(); | ||||
|         VerificatInfoService.RemoveCache(); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -80,7 +80,9 @@ public static class DbContext | ||||
|     { | ||||
|         db.CurrentConnectionConfig.MoreSettings = new ConnMoreSettings | ||||
|         { | ||||
|             SqlServerCodeFirstNvarchar = true//设置默认nvarchar | ||||
|             SqlServerCodeFirstNvarchar = true, //设置默认nvarchar | ||||
|  | ||||
|             IsNoReadXmlDescription = true | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -36,6 +36,7 @@ public class Startup : AppStartup | ||||
|         services.AddSingleton<ISugarAopService, SugarAopService>(); | ||||
|         services.AddSingleton<ISugarConfigAopService, SugarConfigAopService>(); | ||||
|  | ||||
|         services.AddSingleton<IUserAgentService, UserAgentService>(); | ||||
|         services.AddSingleton<IAppService, AppService>(); | ||||
|  | ||||
|         StaticConfig.EnableAllWhereIF = true; | ||||
|   | ||||
| @@ -18,8 +18,7 @@ | ||||
| 	</ItemGroup> | ||||
| 	 | ||||
| 	<ItemGroup> | ||||
| 		<PackageReference Include="BootstrapBlazor.TableExport" Version="9.2.4" /> | ||||
| 		<PackageReference Include="UAParser" Version="3.1.47" /> | ||||
| 		<PackageReference Include="BootstrapBlazor.TableExport" Version="9.2.5" /> | ||||
| 		<PackageReference Include="Rougamo.Fody" Version="5.0.0" /> | ||||
| 		<PackageReference Include="SqlSugarCore" Version="5.1.4.193" /> | ||||
| 	</ItemGroup> | ||||
|   | ||||
| @@ -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; | ||||
|             var ids = CurrentUser.ModuleList.Select(a => a.Id).ToHashSet(); | ||||
|             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) | ||||
|             { | ||||
|   | ||||
| @@ -39,19 +39,4 @@ | ||||
|     </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.Extensions.Localization; | ||||
|  | ||||
| using System.Diagnostics.CodeAnalysis; | ||||
|  | ||||
| using ThingsGateway.Admin.Application; | ||||
| using ThingsGateway.Admin.Razor; | ||||
| using ThingsGateway.Extension; | ||||
| @@ -31,118 +29,8 @@ namespace ThingsGateway.AdminServer; | ||||
| [IgnoreRolePermission] | ||||
| [Route("/")] | ||||
| [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] | ||||
|     private BlazorAppContext AppContext { get; set; } | ||||
|   | ||||
| @@ -14,7 +14,6 @@ using Microsoft.AspNetCore.DataProtection; | ||||
| using Microsoft.AspNetCore.DataProtection.AuthenticatedEncryption; | ||||
| using Microsoft.AspNetCore.DataProtection.AuthenticatedEncryption.ConfigurationModel; | ||||
| using Microsoft.AspNetCore.HttpOverrides; | ||||
| using Microsoft.AspNetCore.Mvc.Controllers; | ||||
| using Microsoft.AspNetCore.StaticFiles; | ||||
| using Microsoft.Extensions.Localization; | ||||
| using Microsoft.Extensions.Options; | ||||
| @@ -29,8 +28,8 @@ using System.Text.Unicode; | ||||
| using ThingsGateway.Admin.Application; | ||||
| using ThingsGateway.Admin.Razor; | ||||
| using ThingsGateway.Extension; | ||||
| using ThingsGateway.Logging; | ||||
| using ThingsGateway.NewLife.Caching; | ||||
| using ThingsGateway.NewLife.Extension; | ||||
|  | ||||
| namespace ThingsGateway.AdminServer; | ||||
|  | ||||
| @@ -161,7 +160,8 @@ public class Startup : AppStartup | ||||
|         { | ||||
|             options.WriteFilter = (logMsg) => | ||||
|             { | ||||
|                 return true; | ||||
|                 if (logMsg.Message.IsNullOrEmpty()) return false; | ||||
|                 else return true; | ||||
|             }; | ||||
|  | ||||
|             options.MessageFormat = (logMsg) => | ||||
| @@ -211,39 +211,40 @@ public class Startup : AppStartup | ||||
|         #region api日志 | ||||
|  | ||||
|         //Monitor日志配置 | ||||
|         services.AddMonitorLogging(options => | ||||
|         { | ||||
|             options.JsonIndented = true;// 是否美化 JSON | ||||
|             options.GlobalEnabled = false;//全局启用 | ||||
|             options.ConfigureLogger((logger, logContext, context) => | ||||
|             { | ||||
|                 var httpContext = context.HttpContext;//获取httpContext | ||||
|         //services.AddMonitorLogging(options => | ||||
|         //{ | ||||
|         //    options.JsonIndented = true;// 是否美化 JSON | ||||
|         //    options.GlobalEnabled = false;//全局启用 | ||||
|         //    options.ConfigureLogger((logger, logContext, context) => | ||||
|         //    { | ||||
|         //        var httpContext = context.HttpContext;//获取httpContext | ||||
|  | ||||
|                 //获取客户端信息 | ||||
|                 var client = App.GetService<IAppService>().ClientInfo; | ||||
|                 // 获取控制器/操作描述器 | ||||
|                 var controllerActionDescriptor = context.ActionDescriptor as ControllerActionDescriptor; | ||||
|                 //操作名称默认是控制器名加方法名,自定义操作名称要在action上加Description特性 | ||||
|                 var option = $"{controllerActionDescriptor.ControllerName}/{controllerActionDescriptor.ActionName}"; | ||||
|         //        //获取客户端信息 | ||||
|         //        var client = App.GetService<IAppService>().UserAgent; | ||||
|         //        // 获取控制器/操作描述器 | ||||
|         //        var controllerActionDescriptor = context.ActionDescriptor as ControllerActionDescriptor; | ||||
|         //        //操作名称默认是控制器名加方法名,自定义操作名称要在action上加Description特性 | ||||
|         //        var option = $"{controllerActionDescriptor.ControllerName}/{controllerActionDescriptor.ActionName}"; | ||||
|  | ||||
|                 var desc = App.CreateLocalizerByType(controllerActionDescriptor.ControllerTypeInfo.AsType())[controllerActionDescriptor.MethodInfo.Name]; | ||||
|                 //获取特性 | ||||
|                 option = desc.Value;//则将操作名称赋值为控制器上写的title | ||||
|         //        var desc = App.CreateLocalizerByType(controllerActionDescriptor.ControllerTypeInfo.AsType())[controllerActionDescriptor.MethodInfo.Name]; | ||||
|         //        //获取特性 | ||||
|         //        option = desc.Value;//则将操作名称赋值为控制器上写的title | ||||
|  | ||||
|                 logContext.Set(LoggingConst.CateGory, option);//传操作名称 | ||||
|                 logContext.Set(LoggingConst.Operation, option);//传操作名称 | ||||
|                 logContext.Set(LoggingConst.Client, client);//客户端信息 | ||||
|                 logContext.Set(LoggingConst.Path, httpContext.Request.Path.Value);//请求地址 | ||||
|                 logContext.Set(LoggingConst.Method, httpContext.Request.Method);//请求方法 | ||||
|             }); | ||||
|         }); | ||||
|         //        logContext.Set(LoggingConst.CateGory, option);//传操作名称 | ||||
|         //        logContext.Set(LoggingConst.Operation, option);//传操作名称 | ||||
|         //        logContext.Set(LoggingConst.Client, client);//客户端信息 | ||||
|         //        logContext.Set(LoggingConst.Path, httpContext.Request.Path.Value);//请求地址 | ||||
|         //        logContext.Set(LoggingConst.Method, httpContext.Request.Method);//请求方法 | ||||
|         //    }); | ||||
|         //}); | ||||
|         services.AddMvcFilter<RequestAuditFilter>(); | ||||
|  | ||||
|         //日志写入数据库配置 | ||||
|         services.AddDatabaseLogging<DatabaseLoggingWriter>(options => | ||||
|         { | ||||
|             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); | ||||
| #endif | ||||
|         services.AddDataProtection() | ||||
|             .PersistKeysToFileSystem(new DirectoryInfo("../keys")) | ||||
|             .PersistKeysToFileSystem(new DirectoryInfo("keys")) | ||||
|             .ProtectKeysWithCertificate(certificate) | ||||
|             .UseCryptographicAlgorithms(new AuthenticatedEncryptorConfiguration | ||||
|             { | ||||
|   | ||||
| @@ -20,7 +20,7 @@ namespace ThingsGateway.Logging; | ||||
| /// </summary> | ||||
| /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | ||||
| [SuppressSniffer] | ||||
| public sealed class DatabaseLogger : ILogger | ||||
| public sealed class DatabaseLogger : ILogger, IDisposable | ||||
| { | ||||
|     /// <summary> | ||||
|     /// 记录器类别名称 | ||||
| @@ -60,6 +60,11 @@ public sealed class DatabaseLogger : ILogger | ||||
|         return _databaseLoggerProvider.ScopeProvider?.Push(state); | ||||
|     } | ||||
|  | ||||
|     public void Dispose() | ||||
|     { | ||||
|         _databaseLoggerProvider.RemoveCache(_logName); | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 检查是否已启用给定日志级别 | ||||
|     /// </summary> | ||||
|   | ||||
| @@ -14,6 +14,8 @@ using Microsoft.Extensions.Logging; | ||||
|  | ||||
| using System.Collections.Concurrent; | ||||
|  | ||||
| using ThingsGateway.Extension.Generic; | ||||
|  | ||||
| namespace ThingsGateway.Logging; | ||||
|  | ||||
| /// <summary> | ||||
| @@ -54,6 +56,8 @@ public sealed class DatabaseLoggerProvider : ILoggerProvider, ISupportExternalSc | ||||
|     /// <remarks>实现不间断写入</remarks> | ||||
|     private Task _processQueueTask; | ||||
|  | ||||
|  | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 构造函数 | ||||
|     /// </summary> | ||||
| @@ -82,7 +86,10 @@ public sealed class DatabaseLoggerProvider : ILoggerProvider, ISupportExternalSc | ||||
|     { | ||||
|         return _databaseLoggers.GetOrAdd(categoryName, name => new DatabaseLogger(name, this)); | ||||
|     } | ||||
|  | ||||
|     public void RemoveCache(string categoryName) | ||||
|     { | ||||
|         _databaseLoggers.Remove(categoryName); | ||||
|     } | ||||
|     /// <summary> | ||||
|     /// 设置作用域提供器 | ||||
|     /// </summary> | ||||
|   | ||||
| @@ -18,8 +18,17 @@ namespace ThingsGateway.Logging; | ||||
| /// </summary> | ||||
| /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | ||||
| [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> | ||||
| @@ -31,6 +40,11 @@ public sealed class EmptyLogger : ILogger | ||||
|         return default; | ||||
|     } | ||||
|  | ||||
|     public void Dispose() | ||||
|     { | ||||
|         _emptyLoggerProvider.RemoveCache(_logName); | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 检查是否已启用给定日志级别 | ||||
|     /// </summary> | ||||
|   | ||||
| @@ -13,6 +13,8 @@ using Microsoft.Extensions.Logging; | ||||
|  | ||||
| using System.Collections.Concurrent; | ||||
|  | ||||
| using ThingsGateway.Extension.Generic; | ||||
|  | ||||
| namespace ThingsGateway.Logging; | ||||
|  | ||||
| /// <summary> | ||||
| @@ -34,9 +36,12 @@ public sealed class EmptyLoggerProvider : ILoggerProvider | ||||
|     /// <returns><see cref="ILogger"/></returns> | ||||
|     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> | ||||
|   | ||||
| @@ -18,7 +18,7 @@ namespace ThingsGateway.Logging; | ||||
| /// </summary> | ||||
| /// <remarks>https://docs.microsoft.com/zh-cn/dotnet/core/extensions/custom-logging-provider</remarks> | ||||
| [SuppressSniffer] | ||||
| public sealed class FileLogger : ILogger | ||||
| public sealed class FileLogger : ILogger, IDisposable | ||||
| { | ||||
|     /// <summary> | ||||
|     /// 记录器类别名称 | ||||
| @@ -58,6 +58,11 @@ public sealed class FileLogger : ILogger | ||||
|         return _fileLoggerProvider.ScopeProvider?.Push(state); | ||||
|     } | ||||
|  | ||||
|     public void Dispose() | ||||
|     { | ||||
|         _fileLoggerProvider.RemoveCache(_logName); | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 检查是否已启用给定日志级别 | ||||
|     /// </summary> | ||||
|   | ||||
| @@ -13,6 +13,8 @@ using Microsoft.Extensions.Logging; | ||||
|  | ||||
| using System.Collections.Concurrent; | ||||
|  | ||||
| using ThingsGateway.Extension.Generic; | ||||
|  | ||||
| namespace ThingsGateway.Logging; | ||||
|  | ||||
| /// <summary> | ||||
| @@ -116,6 +118,10 @@ public sealed class FileLoggerProvider : ILoggerProvider, ISupportExternalScope | ||||
|     { | ||||
|         return _fileLoggers.GetOrAdd(categoryName, name => new FileLogger(name, this)); | ||||
|     } | ||||
|     public void RemoveCache(string categoryName) | ||||
|     { | ||||
|         _fileLoggers.Remove(categoryName); | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 设置作用域提供器 | ||||
|   | ||||
| @@ -17,7 +17,6 @@ namespace ThingsGateway.Logging; | ||||
| [SuppressSniffer] | ||||
| public sealed class LogContext : IDisposable | ||||
| { | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 日志上下文数据 | ||||
|     /// </summary> | ||||
|   | ||||
| @@ -39,7 +39,7 @@ | ||||
| 		<PackageReference Include="System.Text.RegularExpressions" Version="4.3.1" /> | ||||
| 		<PackageReference Include="Mapster" Version="7.4.0" /> | ||||
| 		<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 Condition=" '$(TargetFramework)' == 'net8.0' "> | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| using System.Buffers; | ||||
| using System.Runtime.CompilerServices; | ||||
| using System.Runtime.CompilerServices; | ||||
|  | ||||
| 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; | ||||
| #endif | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| using System.Buffers; | ||||
| using System.Runtime.InteropServices; | ||||
| using System.Runtime.InteropServices; | ||||
| using System.Text; | ||||
|  | ||||
| using ThingsGateway.NewLife.Collections; | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| using System.Buffers; | ||||
| using System.Text; | ||||
| using System.Text; | ||||
|  | ||||
| namespace ThingsGateway.NewLife.Collections; | ||||
|  | ||||
| @@ -95,7 +94,7 @@ public static class Pool | ||||
|     { | ||||
|         //if (ms == null) return null; | ||||
|  | ||||
|         var buf = returnResult ? ms.ToArray() : Empty; | ||||
|         var buf = returnResult ? ms.ToArray() : Array.Empty<byte>(); | ||||
|  | ||||
|         Pool.MemoryStream.Return(ms); | ||||
|  | ||||
| @@ -133,11 +132,5 @@ public static class Pool | ||||
|     } | ||||
|     #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.Reflection; | ||||
|  | ||||
| using ThingsGateway.NewLife.Collections; | ||||
|  | ||||
| namespace ThingsGateway.NewLife.Extension; | ||||
|  | ||||
| /// <summary>工具类</summary> | ||||
| @@ -452,12 +450,12 @@ public class DefaultConvert | ||||
|                     // 凑够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); | ||||
|  | ||||
|                         var dec = BitConverter.ToDouble(bts, 0).ToDecimal(); | ||||
|  | ||||
|                         Pool.Shared.Return(bts); | ||||
|                         ArrayPool<Byte>.Shared.Return(bts); | ||||
|  | ||||
|                         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> | ||||
| /// json扩展 | ||||
| /// </summary> | ||||
| public static class JsonExtensions | ||||
| public static class JsonExtension | ||||
| { | ||||
|     /// <summary> | ||||
|     /// 默认Json规则 | ||||
|     /// </summary> | ||||
|     public static JsonSerializerSettings IndentedOptions; | ||||
|     public static JsonSerializerSettings NoneIndentedOptions; | ||||
|     static JsonExtensions() | ||||
|     static JsonExtension() | ||||
|     { | ||||
|         IndentedOptions = new JsonSerializerSettings | ||||
|         { | ||||
| @@ -81,4 +81,52 @@ public static class JsonExtensions | ||||
|     { | ||||
|         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,692 @@ | ||||
| //------------------------------------------------------------------------------ | ||||
| //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||
| //  此代码版权(除特别声明外的代码)归作者本人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 | ||||
|         }; | ||||
|         // 如有自定义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 | ||||
|         }; | ||||
|         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 | ||||
| //------------------------------------------------------------------------------ | ||||
|  | ||||
| global using System.Buffers; | ||||
|  | ||||
| global using ThingsGateway.NewLife.Extension; | ||||
|   | ||||
| @@ -5,7 +5,6 @@ using System.Net.Sockets; | ||||
| using System.Runtime.InteropServices; | ||||
|  | ||||
| using ThingsGateway.NewLife.Caching; | ||||
| using ThingsGateway.NewLife.Collections; | ||||
| using ThingsGateway.NewLife.Net; | ||||
|  | ||||
| namespace ThingsGateway.NewLife; | ||||
| @@ -52,7 +51,7 @@ public static class NetHelper | ||||
| #endif | ||||
|         { | ||||
|             UInt32 dummy = 0; | ||||
|             var inOptionValues = Pool.Shared.Rent(Marshal.SizeOf(dummy) * 3); | ||||
|             var inOptionValues = ArrayPool<Byte>.Shared.Rent(Marshal.SizeOf(dummy) * 3); | ||||
|  | ||||
|             // 是否启用Keep-Alive | ||||
|             BitConverter.GetBytes((UInt32)(isKeepAlive ? 1 : 0)).CopyTo(inOptionValues, 0); | ||||
| @@ -63,7 +62,7 @@ public static class NetHelper | ||||
|  | ||||
|             socket.IOControl(IOControlCode.KeepAliveValues, inOptionValues, null); | ||||
|  | ||||
|             Pool.Shared.Return(inOptionValues); | ||||
|             ArrayPool<Byte>.Shared.Return(inOptionValues); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
| @@ -538,11 +537,11 @@ public static class NetHelper | ||||
|     private static void Wake(String mac) | ||||
|     { | ||||
|         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++) | ||||
|             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++) | ||||
|             bts[i] = 0xFF; | ||||
|         for (Int32 i = 6, k = 0; i < bts.Length; i++, k++) | ||||
| @@ -560,8 +559,8 @@ public static class NetHelper | ||||
|         client.Close(); | ||||
|         //client.SendAsync(bts, bts.Length, new IPEndPoint(IPAddress.Broadcast, 7)); | ||||
|  | ||||
|         Pool.Shared.Return(bts); | ||||
|         Pool.Shared.Return(buffer); | ||||
|         ArrayPool<Byte>.Shared.Return(bts); | ||||
|         ArrayPool<Byte>.Shared.Return(buffer); | ||||
|     } | ||||
|     #endregion | ||||
|  | ||||
|   | ||||
| @@ -627,6 +627,8 @@ public class DefaultReflect : IReflect | ||||
|     /// <returns></returns> | ||||
|     public virtual Type? GetElementType(Type type) | ||||
|     { | ||||
|         if (type == null) return null; | ||||
|  | ||||
|         if (type.HasElementType) return type.GetElementType(); | ||||
|  | ||||
|         if (type.As<IEnumerable>()) | ||||
|   | ||||
| @@ -1,8 +1,6 @@ | ||||
| using System.Buffers; | ||||
| using System.Reflection; | ||||
| using System.Reflection; | ||||
| using System.Runtime.InteropServices; | ||||
|  | ||||
| using ThingsGateway.NewLife.Collections; | ||||
| using ThingsGateway.NewLife.Reflection; | ||||
|  | ||||
| namespace ThingsGateway.NewLife.Serialization; | ||||
| @@ -495,7 +493,7 @@ public class Binary : FormatterBase, IBinary | ||||
|     /// <param name="max"></param> | ||||
|     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) | ||||
|         { | ||||
|             var a = (Byte)(value[j] - '0'); | ||||
| @@ -504,7 +502,7 @@ public class Binary : FormatterBase, IBinary | ||||
|         } | ||||
|  | ||||
|         Write(buf, 0, max); | ||||
|         Pool.Shared.Return(buf); | ||||
|         ArrayPool<Byte>.Shared.Return(buf); | ||||
|     } | ||||
|  | ||||
|     /// <summary>写入定长字符串。多余截取,少则补零</summary> | ||||
| @@ -512,11 +510,11 @@ public class Binary : FormatterBase, IBinary | ||||
|     /// <param name="max"></param> | ||||
|     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); | ||||
|  | ||||
|         Write(buf, 0, max); | ||||
|         Pool.Shared.Return(buf); | ||||
|         ArrayPool<Byte>.Shared.Return(buf); | ||||
|     } | ||||
|  | ||||
|     /// <summary>读取定长字符串。多余截取,少则补零</summary> | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| using System.Globalization; | ||||
| using System.Xml; | ||||
|  | ||||
| using ThingsGateway.NewLife.Collections; | ||||
| using ThingsGateway.NewLife.Reflection; | ||||
|  | ||||
| namespace ThingsGateway.NewLife.Serialization; | ||||
| @@ -144,10 +143,10 @@ public class XmlGeneral : XmlHandlerBase | ||||
|         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); | ||||
|             value = buf.ReadBytes(0, count); | ||||
|             Pool.Shared.Return(buf); | ||||
|             ArrayPool<Byte>.Shared.Return(buf); | ||||
|  | ||||
|             return true; | ||||
|         } | ||||
|   | ||||
| @@ -73,7 +73,7 @@ public partial class CultureChooser | ||||
|     { | ||||
|         if (_firstRender) | ||||
|         { | ||||
|             if (OperatingSystem.IsBrowser() || !Runtime.IsWeb) | ||||
|             if (OperatingSystem.IsBrowser() || !Runtime.IsWeb || App.EffectiveTypes.FirstOrDefault(a => a.Name.Contains("WebView")) != null) | ||||
|             { | ||||
|                 var cultureName = item.Value; | ||||
|                 if (cultureName != CultureInfo.CurrentCulture.Name) | ||||
|   | ||||
| @@ -24,9 +24,8 @@ public static class ParallelExtensions | ||||
|     /// <param name="body">要执行的操作</param> | ||||
|     public static void ParallelForEach<T>(this IEnumerable<T> source, Action<T> body) | ||||
|     { | ||||
|         // 创建并行操作的选项对象,设置最大并行度为当前处理器数量的一半 | ||||
|         ParallelOptions options = new(); | ||||
|         options.MaxDegreeOfParallelism = Environment.ProcessorCount / 2 == 0 ? 1 : Environment.ProcessorCount / 2; | ||||
|         options.MaxDegreeOfParallelism = Environment.ProcessorCount; | ||||
|         // 使用 Parallel.ForEach 执行指定的操作 | ||||
|         Parallel.ForEach(source, options, (variable) => | ||||
|         { | ||||
| @@ -42,9 +41,8 @@ public static class ParallelExtensions | ||||
|     /// <param name="body">要执行的操作</param> | ||||
|     public static void ParallelForEach<T>(this IEnumerable<T> source, Action<T, ParallelLoopState, long> body) | ||||
|     { | ||||
|         // 创建并行操作的选项对象,设置最大并行度为当前处理器数量的一半 | ||||
|         ParallelOptions options = new(); | ||||
|         options.MaxDegreeOfParallelism = Environment.ProcessorCount / 2 == 0 ? 1 : Environment.ProcessorCount / 2; | ||||
|         options.MaxDegreeOfParallelism = Environment.ProcessorCount; | ||||
|         // 使用 Parallel.ForEach 执行指定的操作 | ||||
|         Parallel.ForEach(source, options, (variable, state, index) => | ||||
|         { | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| 	</PropertyGroup> | ||||
| 	<ItemGroup> | ||||
| 		<PackageReference Include="BootstrapBlazor.FontAwesome" Version="9.0.2" /> | ||||
| 		<PackageReference Include="BootstrapBlazor" Version="9.6.3" /> | ||||
| 		<PackageReference Include="BootstrapBlazor" Version="9.6.4" /> | ||||
| 		<PackageReference Include="Yitter.IdGenerator" Version="1.0.14" /> | ||||
| 	</ItemGroup> | ||||
|  | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| <Project> | ||||
|  | ||||
| 	<PropertyGroup> | ||||
| 		<PluginVersion>10.6.3</PluginVersion> | ||||
| 		<ProPluginVersion>10.6.3</ProPluginVersion> | ||||
| 		<PluginVersion>10.6.21</PluginVersion> | ||||
| 		<ProPluginVersion>10.6.21</ProPluginVersion> | ||||
| 		<AuthenticationVersion>2.1.7</AuthenticationVersion> | ||||
| 	</PropertyGroup> | ||||
|  | ||||
|   | ||||
| @@ -84,7 +84,7 @@ | ||||
|                             </div> | ||||
|                             <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> | ||||
|   | ||||
| @@ -90,7 +90,7 @@ public abstract class DeviceComponentBase : ComponentBase, IDisposable | ||||
|                 var data = await Plc.ReadAsync(RegisterAddress, ArrayLength, DataType); | ||||
|                 if (data.IsSuccess) | ||||
|                 { | ||||
|                     Plc.Logger?.LogInformation(data.Content.ToJsonNetString()); | ||||
|                     Plc.Logger?.LogInformation(data.Content.ToSystemTextJsonString()); | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|   | ||||
| @@ -71,8 +71,7 @@ public abstract class VariableObject | ||||
|         var jToken = JToken.FromObject(value); | ||||
|         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); | ||||
|             jToken = JToken.FromObject(data); | ||||
|         } | ||||
|   | ||||
| @@ -10,8 +10,8 @@ | ||||
|  | ||||
| 	<ItemGroup> | ||||
| 		<PackageReference Include="Microsoft.Extensions.Localization.Abstractions" Version="9.0.5" /> | ||||
| 		<PackageReference Include="TouchSocket" Version="3.1.2" /> | ||||
| 		<PackageReference Include="TouchSocket.SerialPorts" Version="3.1.2" /> | ||||
| 		<PackageReference Include="TouchSocket" Version="3.1.5" /> | ||||
| 		<PackageReference Include="TouchSocket.SerialPorts" Version="3.1.5" /> | ||||
| 	</ItemGroup> | ||||
|  | ||||
| 	<ItemGroup> | ||||
|   | ||||
| @@ -48,25 +48,88 @@ public static class JTokenUtil | ||||
|     /// 根据JToken获取Object类型值<br></br> | ||||
|     /// 对应返回 对象字典 或 类型数组 或 类型值 | ||||
|     /// </summary> | ||||
|     public static object? GetObjectFromJToken(this JToken jtoken) | ||||
|     public static object? GetObjectFromJToken(this JToken token) | ||||
|     { | ||||
|         if (jtoken == null) | ||||
|         if (token == null) | ||||
|             return null; | ||||
|         switch (jtoken.Type) | ||||
|  | ||||
|         switch (token.Type) | ||||
|         { | ||||
|             case JTokenType.Object: | ||||
|                 // 如果是对象类型,递归调用本方法获取嵌套的键值对 | ||||
|                 return jtoken.Children<JProperty>() | ||||
|                     .ToDictionary(prop => prop.Name, prop => GetObjectFromJToken(prop.Value)); | ||||
|                 var obj = new Dictionary<string, object>(); | ||||
|                 foreach (var prop in (JObject)token) | ||||
|                     obj[prop.Key] = GetObjectFromJToken(prop.Value); | ||||
|                 return obj; | ||||
|  | ||||
|             case JTokenType.Array: | ||||
|                 // 如果是数组类型,递归调用本方法获取嵌套的元素 | ||||
|                 return jtoken.Select(GetObjectFromJToken).ToArray(); | ||||
|                 var array = (JArray)token; | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.Integer)) | ||||
|                     return array.Select(x => x.Value<long>()).ToList(); | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.Float)) | ||||
|                     return array.Select(x => x.Value<double>()).ToList(); | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.String)) | ||||
|                     return array.Select(x => x.Value<string>()).ToList(); | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.Boolean)) | ||||
|                     return array.Select(x => x.Value<bool>()).ToList(); | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.Date)) | ||||
|                     return array.Select(x => x.Value<DateTime>()).ToList(); | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.TimeSpan)) | ||||
|                     return array.Select(x => x.Value<TimeSpan>()).ToList(); | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.Guid)) | ||||
|                     return array.Select(x => x.Value<Guid>()).ToList(); | ||||
|  | ||||
|                 if (array.All(x => x.Type == JTokenType.Uri)) | ||||
|                     return array.Select(x => x.Value<Uri>()).ToList(); | ||||
|  | ||||
|                 // 否则递归 | ||||
|                 return array.Select(x => GetObjectFromJToken(x)).ToList(); | ||||
|  | ||||
|             case JTokenType.Integer: | ||||
|                 return token.ToObject<long>(); | ||||
|  | ||||
|             case JTokenType.Float: | ||||
|                 return token.ToObject<double>(); | ||||
|  | ||||
|             case JTokenType.String: | ||||
|                 return token.ToObject<string>(); | ||||
|  | ||||
|             case JTokenType.Boolean: | ||||
|                 return token.ToObject<bool>(); | ||||
|  | ||||
|             case JTokenType.Null: | ||||
|             case JTokenType.Undefined: | ||||
|                 return null; | ||||
|  | ||||
|             case JTokenType.Date: | ||||
|                 return token.ToObject<DateTime>(); | ||||
|  | ||||
|             case JTokenType.TimeSpan: | ||||
|                 return token.ToObject<TimeSpan>(); | ||||
|  | ||||
|             case JTokenType.Guid: | ||||
|                 return token.ToObject<Guid>(); | ||||
|  | ||||
|             case JTokenType.Uri: | ||||
|                 return token.ToObject<Uri>(); | ||||
|  | ||||
|             case JTokenType.Bytes: | ||||
|                 return token.ToObject<byte[]>(); | ||||
|  | ||||
|             case JTokenType.Comment: | ||||
|             case JTokenType.Raw: | ||||
|             case JTokenType.Property: | ||||
|             case JTokenType.Constructor: | ||||
|             default: | ||||
|                 // 其他类型直接转换为对应的 Object 类型值 | ||||
|                 return (jtoken as JValue)?.Value; | ||||
|                 return token.ToString(); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     #region json | ||||
|   | ||||
| @@ -35,12 +35,14 @@ namespace ThingsGateway.Gateway.Application; | ||||
| [ApiDescriptionSettings("ThingsGateway.OpenApi", Order = 200)] | ||||
| [Route("openApi/control")] | ||||
| [RolePermission] | ||||
| [LoggingMonitor] | ||||
| [RequestAudit] | ||||
| [ApiController] | ||||
| [Authorize(AuthenticationSchemes = "Bearer")] | ||||
| public class ControlController : ControllerBase | ||||
| { | ||||
|  | ||||
|  | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 清空全部缓存 | ||||
|     /// </summary> | ||||
| @@ -141,7 +143,7 @@ public class ControlController : ControllerBase | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return await GlobalData.RpcService.InvokeDeviceMethodAsync($"WebApi-{UserManager.UserAccount}-{App.HttpContext?.GetRemoteIpAddressToIPv4()}", deviceDatas).ConfigureAwait(false); | ||||
|         return (await GlobalData.RpcService.InvokeDeviceMethodAsync($"WebApi-{UserManager.UserAccount}-{App.HttpContext?.GetRemoteIpAddressToIPv4()}", deviceDatas).ConfigureAwait(false)).ToDictionary(a => a.Key, a => a.Value.ToDictionary(b => b.Key, b => (OperResult)b.Value)); | ||||
|  | ||||
|     } | ||||
|  | ||||
| @@ -175,8 +177,50 @@ public class ControlController : ControllerBase | ||||
|         return GlobalData.VariableRuntimeService.BatchSaveVariableAsync(variables.Adapt<List<Variable>>(), type, restart, default); | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 删除通道 | ||||
|     /// </summary> | ||||
|     [HttpPost("deleteChannel")] | ||||
|     [DisplayName("删除通道")] | ||||
|     public Task<bool> DeleteChannelAsync([FromBody] List<long> ids, bool restart) | ||||
|     { | ||||
|         if (ids == null || ids.Count == 0) ids = GlobalData.Channels.Keys.ToList(); | ||||
|         return GlobalData.ChannelRuntimeService.DeleteChannelAsync(ids, restart, default); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 删除设备 | ||||
|     /// </summary> | ||||
|     [HttpPost("deleteDevice")] | ||||
|     [DisplayName("删除设备")] | ||||
|     public Task<bool> DeleteDeviceAsync([FromBody] List<long> ids, bool restart) | ||||
|     { | ||||
|         if (ids == null || ids.Count == 0) ids = GlobalData.IdDevices.Keys.ToList(); | ||||
|         return GlobalData.DeviceRuntimeService.DeleteDeviceAsync(ids, restart, default); | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 删除变量 | ||||
|     /// </summary> | ||||
|     [HttpPost("deleteVariable")] | ||||
|     [DisplayName("删除变量")] | ||||
|     public Task<bool> DeleteVariableAsync([FromBody] List<long> ids, bool restart) | ||||
|     { | ||||
|         if (ids == null || ids.Count == 0) ids = GlobalData.IdVariables.Keys.ToList(); | ||||
|         return GlobalData.VariableRuntimeService.DeleteVariableAsync(ids, restart, default); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 增加测试数据 | ||||
|     /// </summary> | ||||
|     [HttpPost("insertTestData")] | ||||
|     [DisplayName("增加测试数据")] | ||||
|     public Task InsertTestDataAsync(int testVariableCount, int testDeviceCount, string slaveUrl, bool businessEnable, bool restart) | ||||
|     { | ||||
|         return GlobalData.VariableRuntimeService.InsertTestDataAsync(testVariableCount, testDeviceCount, slaveUrl, businessEnable, restart, default); | ||||
|     } | ||||
| } | ||||
| public class ChannelInput | ||||
| { | ||||
|   | ||||
| @@ -10,8 +10,6 @@ | ||||
|  | ||||
| using System.Globalization; | ||||
| using System.Text; | ||||
| using System.Text.Encodings.Web; | ||||
| using System.Text.Json; | ||||
| using System.Text.RegularExpressions; | ||||
|  | ||||
| using ThingsGateway.NewLife.Json.Extension; | ||||
| @@ -92,7 +90,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|  | ||||
|                         // 如果是报警列表,则将整个分组转换为 JSON 字符串 | ||||
|                         var gList = group.Select(a => a).ToList(); | ||||
|                         string json = gList.ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                         string json = gList.ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                         // 将主题和 JSON 内容添加到列表中 | ||||
|                         topicJsonList.Add(new(topic, json, gList.Count)); | ||||
|                     } | ||||
| @@ -101,7 +99,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         // 如果不是报警列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                         foreach (var gro in group) | ||||
|                         { | ||||
|                             string json = JsonExtensions.ToJsonNetString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             string json = SystemTextJsonExtension.ToSystemTextJsonString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicJsonList.Add(new(topic, json, 1)); | ||||
|                         } | ||||
| @@ -114,14 +112,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsAlarmList) | ||||
|             { | ||||
|                 var gList = data.Select(a => a).ToList(); | ||||
|                 string json = gList.ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 string json = gList.ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.AlarmTopic, json, gList.Count)); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     string json = JsonExtensions.ToJsonNetString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     string json = SystemTextJsonExtension.ToSystemTextJsonString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.AlarmTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -160,7 +158,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         { | ||||
|                             // 如果是设备列表,则将整个分组转换为 JSON 字符串 | ||||
|                             var gList = group.Select(a => a).ToList(); | ||||
|                             string json = gList.ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             string json = gList.ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicJsonList.Add(new(topic, json, gList.Count)); | ||||
|                         } | ||||
| @@ -169,7 +167,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                             // 如果不是设备列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                             foreach (var gro in group) | ||||
|                             { | ||||
|                                 string json = JsonExtensions.ToJsonNetString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 string json = SystemTextJsonExtension.ToSystemTextJsonString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 // 将主题和 JSON 内容添加到列表中 | ||||
|                                 topicJsonList.Add(new(topic, json, 1)); | ||||
|                             } | ||||
| @@ -183,14 +181,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsDeviceList) | ||||
|             { | ||||
|                 var gList = data.Select(a => a).ToList(); | ||||
|                 string json = gList.ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 string json = gList.ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.DeviceTopic, json, gList.Count)); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     string json = JsonExtensions.ToJsonNetString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     string json = SystemTextJsonExtension.ToSystemTextJsonString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.DeviceTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -227,7 +225,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         { | ||||
|                             // 如果是变量列表,则将整个分组转换为 JSON 字符串 | ||||
|                             var gList = group.Select(a => a).ToList(); | ||||
|                             string json = gList.ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             string json = gList.ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicJsonList.Add(new(topic, json, gList.Count)); | ||||
|                         } | ||||
| @@ -236,7 +234,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                             // 如果不是变量列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                             foreach (var gro in group) | ||||
|                             { | ||||
|                                 string json = JsonExtensions.ToJsonNetString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 string json = SystemTextJsonExtension.ToSystemTextJsonString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 // 将主题和 JSON 内容添加到列表中 | ||||
|                                 topicJsonList.Add(new(topic, json, 1)); | ||||
|                             } | ||||
| @@ -250,14 +248,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsVariableList) | ||||
|             { | ||||
|                 var gList = data.Select(a => a).ToList(); | ||||
|                 string json = gList.ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 string json = gList.ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, gList.Count)); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     string json = JsonExtensions.ToJsonNetString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     string json = SystemTextJsonExtension.ToSystemTextJsonString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -303,7 +301,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         { | ||||
|  | ||||
|                             // 如果是变量列表,则将整个分组转换为 JSON 字符串 | ||||
|                             string json = group.Select(a => a).GroupBy(a => a.DeviceName, b => b).ToDictionary(a => a.Key, b => b.ToList()).ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             string json = group.Select(a => a).GroupBy(a => a.DeviceName, b => b).ToDictionary(a => a.Key, b => b.ToList()).ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicJsonList.Add(new(topic, json, group.Count())); | ||||
|                         } | ||||
| @@ -312,7 +310,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                             // 如果不是变量列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                             foreach (var gro in group) | ||||
|                             { | ||||
|                                 string json = JsonExtensions.ToJsonNetString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 string json = SystemTextJsonExtension.ToSystemTextJsonString(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 // 将主题和 JSON 内容添加到列表中 | ||||
|                                 topicJsonList.Add(new(topic, json, 1)); | ||||
|                             } | ||||
| @@ -325,14 +323,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|         { | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsVariableList) | ||||
|             { | ||||
|                 string json = data.Select(a => a).GroupBy(a => a.DeviceName, b => b).ToDictionary(a => a.Key, b => b.ToList()).ToJsonNetString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 string json = data.Select(a => a).GroupBy(a => a.DeviceName, b => b).ToDictionary(a => a.Key, b => b.ToList()).ToSystemTextJsonString(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, data.Count())); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     string json = JsonExtensions.ToJsonNetString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     string json = SystemTextJsonExtension.ToSystemTextJsonString(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicJsonList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -355,23 +353,6 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|     //        block.Dispose(); | ||||
|     //    } | ||||
|     //} | ||||
|     protected static JsonSerializerOptions NoWriteIndentedJsonSerializerOptions = new JsonSerializerOptions | ||||
|     { | ||||
|         Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, | ||||
|         WriteIndented = false | ||||
|     }; | ||||
|     protected static JsonSerializerOptions WriteIndentedJsonSerializerOptions = new JsonSerializerOptions | ||||
|     { | ||||
|         Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping, | ||||
|         WriteIndented = true | ||||
|     }; | ||||
|  | ||||
|     protected static byte[] Serialize(object data, bool writeIndented) | ||||
|     { | ||||
|         if (data == null) return Array.Empty<byte>(); | ||||
|         byte[] payload = JsonSerializer.SerializeToUtf8Bytes(data, data.GetType(), writeIndented ? WriteIndentedJsonSerializerOptions : NoWriteIndentedJsonSerializerOptions); | ||||
|         return payload; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     protected List<TopicArray> GetAlarmTopicArrays(IEnumerable<AlarmModel> item) | ||||
| @@ -401,7 +382,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                     if (_businessPropertyWithCacheIntervalScript.IsAlarmList) | ||||
|                     { | ||||
|                         var gList = group.Select(a => a).ToList(); | ||||
|                         var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                         var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                         // 将主题和 JSON 内容添加到列表中 | ||||
|                         topicArrayList.Add(new(topic, json, gList.Count)); | ||||
|                     } | ||||
| @@ -410,7 +391,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         // 如果不是报警列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                         foreach (var gro in group) | ||||
|                         { | ||||
|                             var json = Serialize(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             var json = gro.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicArrayList.Add(new(topic, json, 1)); | ||||
|                         } | ||||
| @@ -423,14 +404,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsAlarmList) | ||||
|             { | ||||
|                 var gList = data.Select(a => a).ToList(); | ||||
|                 var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.AlarmTopic, json, gList.Count)); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     var json = Serialize(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     var json = group.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.AlarmTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -468,7 +449,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         { | ||||
|                             // 如果是设备列表,则将整个分组转换为 JSON 字符串 | ||||
|                             var gList = group.Select(a => a).ToList(); | ||||
|                             var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicArrayList.Add(new(topic, json, gList.Count)); | ||||
|                         } | ||||
| @@ -477,7 +458,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                             // 如果不是设备列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                             foreach (var gro in group) | ||||
|                             { | ||||
|                                 var json = Serialize(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 var json = gro.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 // 将主题和 JSON 内容添加到列表中 | ||||
|                                 topicArrayList.Add(new(topic, json, 1)); | ||||
|                             } | ||||
| @@ -491,14 +472,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsDeviceList) | ||||
|             { | ||||
|                 var gList = data.Select(a => a).ToList(); | ||||
|                 var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.DeviceTopic, json, gList.Count)); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     var json = Serialize(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     var json = group.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.DeviceTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -535,7 +516,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         { | ||||
|                             // 如果是变量列表,则将整个分组转换为 JSON 字符串 | ||||
|                             var gList = group.Select(a => a).ToList(); | ||||
|                             var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicArrayList.Add(new(topic, json, gList.Count)); | ||||
|                         } | ||||
| @@ -544,7 +525,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                             // 如果不是变量列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                             foreach (var gro in group) | ||||
|                             { | ||||
|                                 var json = Serialize(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 var json = gro.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 // 将主题和 JSON 内容添加到列表中 | ||||
|                                 topicArrayList.Add(new(topic, json, 1)); | ||||
|                             } | ||||
| @@ -558,14 +539,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsVariableList) | ||||
|             { | ||||
|                 var gList = data.Select(a => a).ToList(); | ||||
|                 var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, gList.Count)); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     var json = Serialize(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     var json = group.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -611,7 +592,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                         { | ||||
|                             // 如果是变量列表,则将整个分组转换为 JSON 字符串 | ||||
|                             var gList = group.Select(a => a).ToList(); | ||||
|                             var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                             // 将主题和 JSON 内容添加到列表中 | ||||
|                             topicArrayList.Add(new(topic, json, gList.Count)); | ||||
|                         } | ||||
| @@ -620,7 +601,7 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|                             // 如果不是变量列表,则将每个分组元素分别转换为 JSON 字符串 | ||||
|                             foreach (var gro in group) | ||||
|                             { | ||||
|                                 var json = Serialize(gro, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 var json = gro.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                                 // 将主题和 JSON 内容添加到列表中 | ||||
|                                 topicArrayList.Add(new(topic, json, 1)); | ||||
|                             } | ||||
| @@ -634,14 +615,14 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|             if (_businessPropertyWithCacheIntervalScript.IsVariableList) | ||||
|             { | ||||
|                 var gList = data.Select(a => a).ToList(); | ||||
|                 var json = Serialize(gList, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 var json = gList.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                 topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, gList.Count)); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 foreach (var group in data) | ||||
|                 { | ||||
|                     var json = Serialize(group, _businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     var json = group.ToSystemTextJsonUtf8Bytes(_businessPropertyWithCacheIntervalScript.JsonFormattingIndented); | ||||
|                     topicArrayList.Add(new(_businessPropertyWithCacheIntervalScript.VariableTopic, json, 1)); | ||||
|                 } | ||||
|             } | ||||
| @@ -653,9 +634,9 @@ public abstract partial class BusinessBaseWithCacheIntervalScript<VarModel, DevM | ||||
|     protected string GetDetailLogString(TopicArray topicArray, int queueCount) | ||||
|     { | ||||
|         if (queueCount > 0) | ||||
|             return $"Up Topic:{topicArray.Topic}{Environment.NewLine}PayLoad:{Encoding.UTF8.GetString(topicArray.Json)} {Environment.NewLine} VarModelQueue:{queueCount}"; | ||||
|             return $"Up Topic:{topicArray.Topic}{Environment.NewLine}PayLoad:{Encoding.UTF8.GetString(topicArray.Payload)} {Environment.NewLine} VarModelQueue:{queueCount}"; | ||||
|         else | ||||
|             return $"Up Topic:{topicArray.Topic}{Environment.NewLine}PayLoad:{Encoding.UTF8.GetString(topicArray.Json)}"; | ||||
|             return $"Up Topic:{topicArray.Topic}{Environment.NewLine}PayLoad:{Encoding.UTF8.GetString(topicArray.Payload)}"; | ||||
|     } | ||||
|  | ||||
|     protected string GetCountLogString(TopicArray topicArray, int queueCount) | ||||
|   | ||||
| @@ -21,7 +21,7 @@ public class BusinessPropertyWithCacheIntervalScript : BusinessPropertyWithCache | ||||
|     /// 是否显示详细日志 | ||||
|     /// </summary> | ||||
|     [DynamicProperty] | ||||
|     public bool DetailLog { get; set; } = true; | ||||
|     public bool DetailLog { get; set; } = false; | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 缩进格式化 | ||||
|   | ||||
| @@ -14,11 +14,11 @@ public struct TopicArray | ||||
| { | ||||
|     public TopicArray(string topic, byte[] json, int count) | ||||
|     { | ||||
|         Topic = topic; Json = json; Count = count; | ||||
|         Topic = topic; Payload = json; Count = count; | ||||
|     } | ||||
|  | ||||
|     public int Count { get; set; } = 1; | ||||
|     public byte[] Json { get; set; } | ||||
|     public byte[] Payload { get; set; } | ||||
|     public string Topic { get; set; } | ||||
|  | ||||
| } | ||||
| @@ -21,5 +21,5 @@ public abstract class VariablePropertyBase | ||||
|     /// 启用 | ||||
|     /// </summary> | ||||
|     [DynamicProperty()] | ||||
|     public bool Enable { get; set; } = true; | ||||
|     public virtual bool Enable { get; set; } = true; | ||||
| } | ||||
|   | ||||
| @@ -31,7 +31,7 @@ namespace ThingsGateway.Gateway.Application; | ||||
| /// 采集插件,继承实现不同PLC通讯 | ||||
| /// <para></para> | ||||
| /// </summary> | ||||
| public abstract class CollectBase : DriverBase | ||||
| public abstract class CollectBase : DriverBase, IRpcDriver | ||||
| { | ||||
|     /// <summary> | ||||
|     /// 插件配置项 | ||||
| @@ -356,7 +356,7 @@ public abstract class CollectBase : DriverBase | ||||
|             { | ||||
|                 // 方法调用成功时记录日志并增加成功计数器 | ||||
|                 if (LogMessage.LogLevel <= TouchSocket.Core.LogLevel.Trace) | ||||
|                     LogMessage?.Trace(string.Format("{0} - Execute method[{1}] - Succeeded {2}", DeviceName, readVariableMethods.MethodInfo.Name, readResult.Content?.ToJsonNetString())); | ||||
|                     LogMessage?.Trace(string.Format("{0} - Execute method[{1}] - Succeeded {2}", DeviceName, readVariableMethods.MethodInfo.Name, readResult.Content?.ToSystemTextJsonString())); | ||||
|                 readResultCount.deviceMethodsVariableSuccessNum++; | ||||
|                 CurrentDevice.SetDeviceStatus(TimerX.Now, false); | ||||
|             } | ||||
| @@ -550,7 +550,7 @@ public abstract class CollectBase : DriverBase | ||||
|     /// <param name="writeInfoLists">要写入的变量及其对应的数据</param> | ||||
|     /// <param name="cancellationToken">取消操作的通知</param> | ||||
|     /// <returns>写入操作的结果字典</returns> | ||||
|     internal async ValueTask<Dictionary<string, OperResult<object>>> InvokeMethodAsync(Dictionary<VariableRuntime, JToken> writeInfoLists, CancellationToken cancellationToken) | ||||
|     public async ValueTask<Dictionary<string, Dictionary<string, IOperResult>>> InvokeMethodAsync(Dictionary<VariableRuntime, JToken> writeInfoLists, CancellationToken cancellationToken) | ||||
|     { | ||||
|         // 初始化结果字典 | ||||
|         Dictionary<string, OperResult<object>> results = new Dictionary<string, OperResult<object>>(); | ||||
| @@ -562,7 +562,7 @@ public abstract class CollectBase : DriverBase | ||||
|             if (!string.IsNullOrEmpty(deviceVariable.WriteExpressions)) | ||||
|             { | ||||
|                 // 提取原始数据 | ||||
|                 object rawdata = jToken is JValue jValue ? jValue.Value : jToken is JArray jArray ? jArray : jToken.ToString(); | ||||
|                 object rawdata = jToken.GetObjectFromJToken(); | ||||
|                 try | ||||
|                 { | ||||
|                     // 根据写入表达式转换数据 | ||||
| @@ -610,7 +610,13 @@ public abstract class CollectBase : DriverBase | ||||
|         } | ||||
|  | ||||
|         // 将转换失败的变量和写入成功的变量的操作结果合并到结果字典中 | ||||
|         return results.Concat(operResults).ToDictionary(a => a.Key, a => a.Value); | ||||
|         return new Dictionary<string, Dictionary<string, IOperResult>>() | ||||
|         { | ||||
|             { | ||||
|              DeviceName , | ||||
|              results.Concat(operResults).ToDictionary(a => a.Key, a => (IOperResult)a.Value) | ||||
|             } | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
| @@ -619,7 +625,7 @@ public abstract class CollectBase : DriverBase | ||||
|     /// <param name="writeInfoLists">要写入的变量及其对应的数据</param> | ||||
|     /// <param name="cancellationToken">取消操作的通知</param> | ||||
|     /// <returns>写入操作的结果字典</returns> | ||||
|     internal async ValueTask<Dictionary<string, OperResult>> InVokeWriteAsync(Dictionary<VariableRuntime, JToken> writeInfoLists, CancellationToken cancellationToken) | ||||
|     public async ValueTask<Dictionary<string, Dictionary<string, IOperResult>>> InVokeWriteAsync(Dictionary<VariableRuntime, JToken> writeInfoLists, CancellationToken cancellationToken) | ||||
|     { | ||||
|         // 初始化结果字典 | ||||
|         Dictionary<string, OperResult> results = new Dictionary<string, OperResult>(); | ||||
| @@ -632,7 +638,7 @@ public abstract class CollectBase : DriverBase | ||||
|             if (!string.IsNullOrEmpty(deviceVariable.WriteExpressions)) | ||||
|             { | ||||
|                 // 提取原始数据 | ||||
|                 object rawdata = jToken is JValue jValue ? jValue.Value : jToken is JArray jArray ? jArray : jToken.ToString(); | ||||
|                 object rawdata = jToken.GetObjectFromJToken(); | ||||
|                 try | ||||
|                 { | ||||
|                     // 根据写入表达式转换数据 | ||||
| @@ -664,7 +670,14 @@ public abstract class CollectBase : DriverBase | ||||
|             cancellationToken).ConfigureAwait(false); | ||||
|  | ||||
|         // 将转换失败的变量和写入成功的变量的操作结果合并到结果字典中 | ||||
|         return results.Concat(results1).ToDictionary(a => a.Key, a => a.Value); | ||||
|  | ||||
|         return new Dictionary<string, Dictionary<string, IOperResult>>() | ||||
|         { | ||||
|             { | ||||
|                 DeviceName , | ||||
|                 results.Concat(results1).ToDictionary(a => a.Key, a => (IOperResult)a.Value) | ||||
|             } | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /// <summary> | ||||
|   | ||||
| @@ -326,11 +326,14 @@ public abstract class DriverBase : DisposableObject, IDriver | ||||
|  | ||||
|     protected override void Dispose(bool disposing) | ||||
|     { | ||||
|         TextLogger?.Dispose(); | ||||
|         _logger?.TryDispose(); | ||||
|         IdVariableRuntimes?.Clear(); | ||||
|         IdVariableRuntimes = null; | ||||
|         var device = CurrentDevice; | ||||
|         if (device != null) | ||||
|             device.Driver = null; | ||||
|  | ||||
|         LogMessage?.Logs?.ForEach(a => a.TryDispose()); | ||||
|         LogMessage = null; | ||||
|         pluginPropertyEditorItems?.Clear(); | ||||
|   | ||||
| @@ -0,0 +1,23 @@ | ||||
| //------------------------------------------------------------------------------ | ||||
| //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||
| //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||
| //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||
| //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||
| //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||
| //  使用文档:https://thingsgateway.cn/ | ||||
| //  QQ群:605534569 | ||||
| //------------------------------------------------------------------------------ | ||||
|  | ||||
|  | ||||
| using Newtonsoft.Json.Linq; | ||||
|  | ||||
| namespace ThingsGateway.Gateway.Application; | ||||
|  | ||||
| public interface IRpcDriver | ||||
| { | ||||
|  | ||||
|     ValueTask<Dictionary<string, Dictionary<string, IOperResult>>> InvokeMethodAsync(Dictionary<VariableRuntime, JToken> writeInfoLists, CancellationToken cancellationToken); | ||||
|  | ||||
|     ValueTask<Dictionary<string, Dictionary<string, IOperResult>>> InVokeWriteAsync(Dictionary<VariableRuntime, JToken> writeInfoLists, CancellationToken cancellationToken); | ||||
|  | ||||
| } | ||||
| @@ -140,7 +140,7 @@ public class Channel : ChannelOptionsBase, IPrimaryIdEntity, IBaseDataEntity, IB | ||||
|     /// <summary> | ||||
|     /// 缓存超时 | ||||
|     /// </summary> | ||||
|     [SugarColumn(ColumnDescription = "缓存超时", IsNullable = true, DefaultValue = "500")] | ||||
|     [SugarColumn(ColumnDescription = "缓存超时", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     [MinValue(100)] | ||||
|     public override int CacheTimeout { get; set; } = 500; | ||||
| @@ -148,7 +148,7 @@ public class Channel : ChannelOptionsBase, IPrimaryIdEntity, IBaseDataEntity, IB | ||||
|     /// <summary> | ||||
|     /// 连接超时 | ||||
|     /// </summary> | ||||
|     [SugarColumn(ColumnDescription = "连接超时", IsNullable = true, DefaultValue = "3000")] | ||||
|     [SugarColumn(ColumnDescription = "连接超时", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     [MinValue(100)] | ||||
|     public override int ConnectTimeout { get; set; } = 3000; | ||||
| @@ -156,26 +156,26 @@ public class Channel : ChannelOptionsBase, IPrimaryIdEntity, IBaseDataEntity, IB | ||||
|     /// <summary> | ||||
|     /// 最大并发数 | ||||
|     /// </summary> | ||||
|     [SugarColumn(ColumnDescription = "最大并发数", IsNullable = true, DefaultValue = "1")] | ||||
|     [SugarColumn(ColumnDescription = "最大并发数", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     [MinValue(1)] | ||||
|     public override int MaxConcurrentCount { get; set; } = 1; | ||||
|  | ||||
|     [SugarColumn(ColumnDescription = "最大连接数", IsNullable = true, DefaultValue = "10000")] | ||||
|     [SugarColumn(ColumnDescription = "最大连接数", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     public override int MaxClientCount { get; set; } = 10000; | ||||
|  | ||||
|     [SugarColumn(ColumnDescription = "客户端滑动过期时间", IsNullable = true, DefaultValue = "120000")] | ||||
|     [SugarColumn(ColumnDescription = "客户端滑动过期时间", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     public override int CheckClearTime { get; set; } = 120000; | ||||
|  | ||||
|     [SugarColumn(ColumnDescription = "心跳内容", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     public override string Heartbeat { get; set; } = "Heartbeat"; | ||||
|     public override string Heartbeat { get; set; } | ||||
|  | ||||
|     #region dtu终端 | ||||
|  | ||||
|     [SugarColumn(ColumnDescription = "心跳间隔", IsNullable = true, DefaultValue = "60000")] | ||||
|     [SugarColumn(ColumnDescription = "心跳间隔", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     public override int HeartbeatTime { get; set; } = 60000; | ||||
|  | ||||
| @@ -185,7 +185,7 @@ public class Channel : ChannelOptionsBase, IPrimaryIdEntity, IBaseDataEntity, IB | ||||
|  | ||||
|     #endregion | ||||
|  | ||||
|     [SugarColumn(ColumnDescription = "Dtu类型", IsNullable = true, DefaultValue = "0")] | ||||
|     [SugarColumn(ColumnDescription = "Dtu类型", IsNullable = true)] | ||||
|     [AutoGenerateColumn(Visible = true, Filterable = true, Sortable = true)] | ||||
|     public override DtuSeviceType DtuSeviceType { get; set; } | ||||
|  | ||||
|   | ||||
| @@ -0,0 +1,464 @@ | ||||
| //------------------------------------------------------------------------------ | ||||
| //  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充 | ||||
| //  此代码版权(除特别声明外的代码)归作者本人Diego所有 | ||||
| //  源代码使用协议遵循本仓库的开源协议及附加协议 | ||||
| //  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway | ||||
| //  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway | ||||
| //  使用文档:https://thingsgateway.cn/ | ||||
| //  QQ群:605534569 | ||||
| //------------------------------------------------------------------------------ | ||||
|  | ||||
| using System.Collections; | ||||
|  | ||||
| namespace ThingsGateway.Gateway.Application; | ||||
|  | ||||
| public class ThreadSafeStringDictionary<T> : IDictionary<string, T>, IReadOnlyDictionary<string, T> | ||||
| { | ||||
|     private const int DEFAULT_PARTITIONS = 128; | ||||
|     private readonly Dictionary<string, T>[] _partitions; | ||||
|     private readonly object[] _partitionLocks; | ||||
|     private readonly IEqualityComparer<string> _comparer; | ||||
|  | ||||
|     public ThreadSafeStringDictionary() : this(DEFAULT_PARTITIONS, null) { } | ||||
|  | ||||
|     public ThreadSafeStringDictionary(int partitionCount, IEqualityComparer<string> comparer) | ||||
|     { | ||||
|         if (partitionCount < 1) | ||||
|             throw new ArgumentOutOfRangeException(nameof(partitionCount)); | ||||
|  | ||||
|         _partitions = new Dictionary<string, T>[partitionCount]; | ||||
|         _partitionLocks = new object[partitionCount]; | ||||
|         _comparer = comparer ?? StringComparer.Ordinal; | ||||
|  | ||||
|         for (int i = 0; i < partitionCount; i++) | ||||
|         { | ||||
|             _partitions[i] = new Dictionary<string, T>(_comparer); | ||||
|             _partitionLocks[i] = new object(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private int GetPartitionIndex(string key) | ||||
|     { | ||||
|         if (key == null) throw new ArgumentNullException(nameof(key)); | ||||
|         return Math.Abs(_comparer.GetHashCode(key)) % _partitions.Length; | ||||
|     } | ||||
|  | ||||
|     // 基本操作 | ||||
|     public T this[string key] | ||||
|     { | ||||
|         get | ||||
|         { | ||||
|             int index = GetPartitionIndex(key); | ||||
|             lock (_partitionLocks[index]) | ||||
|             { | ||||
|                 return _partitions[index][key]; | ||||
|             } | ||||
|         } | ||||
|         set | ||||
|         { | ||||
|             int index = GetPartitionIndex(key); | ||||
|             lock (_partitionLocks[index]) | ||||
|             { | ||||
|                 _partitions[index][key] = value; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public ICollection<string> Keys => GetAllItems().Select(kv => kv.Key).ToList(); | ||||
|     public ICollection<T> Values => GetAllItems().Select(kv => kv.Value).ToList(); | ||||
|  | ||||
|     public int Count | ||||
|     { | ||||
|         get | ||||
|         { | ||||
|             int count = 0; | ||||
|             for (int i = 0; i < _partitions.Length; i++) | ||||
|             { | ||||
|                 lock (_partitionLocks[i]) | ||||
|                 { | ||||
|                     count += _partitions[i].Count; | ||||
|                 } | ||||
|             } | ||||
|             return count; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool IsReadOnly => false; | ||||
|  | ||||
|     IEnumerable<string> IReadOnlyDictionary<string, T>.Keys => Keys; | ||||
|  | ||||
|     IEnumerable<T> IReadOnlyDictionary<string, T>.Values => Values; | ||||
|  | ||||
|     public void Add(string key, T value) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             _partitions[index].Add(key, value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public void Add(KeyValuePair<string, T> item) => Add(item.Key, item.Value); | ||||
|  | ||||
|     public void Clear() | ||||
|     { | ||||
|         for (int i = 0; i < _partitions.Length; i++) | ||||
|         { | ||||
|             lock (_partitionLocks[i]) | ||||
|             { | ||||
|                 _partitions[i].Clear(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool Contains(KeyValuePair<string, T> item) | ||||
|     { | ||||
|         int index = GetPartitionIndex(item.Key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].TryGetValue(item.Key, out var value) && | ||||
|                    EqualityComparer<T>.Default.Equals(value, item.Value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool ContainsKey(string key) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].ContainsKey(key); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool Remove(string key) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].Remove(key); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool Remove(KeyValuePair<string, T> item) | ||||
|     { | ||||
|         int index = GetPartitionIndex(item.Key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             if (_partitions[index].TryGetValue(item.Key, out var value) && | ||||
|                 EqualityComparer<T>.Default.Equals(value, item.Value)) | ||||
|             { | ||||
|                 return _partitions[index].Remove(item.Key); | ||||
|             } | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool TryGetValue(string key, out T value) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].TryGetValue(key, out value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public void CopyTo(KeyValuePair<string, T>[] array, int arrayIndex) | ||||
|     { | ||||
|         if (array == null) throw new ArgumentNullException(nameof(array)); | ||||
|         if (arrayIndex < 0) throw new ArgumentOutOfRangeException(nameof(arrayIndex)); | ||||
|         if (array.Length - arrayIndex < Count) throw new ArgumentException("Target array too small"); | ||||
|  | ||||
|         foreach (var item in GetAllItems()) | ||||
|         { | ||||
|             array[arrayIndex++] = item; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // 枚举器实现 | ||||
|     public IEnumerator<KeyValuePair<string, T>> GetEnumerator() | ||||
|     { | ||||
|         return GetAllItems().GetEnumerator(); | ||||
|     } | ||||
|  | ||||
|     IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); | ||||
|  | ||||
|     public void AddRange(IEnumerable<KeyValuePair<string, T>> items) | ||||
|     { | ||||
|         var grouped = items.GroupBy(item => GetPartitionIndex(item.Key)); | ||||
|  | ||||
|         foreach (var group in grouped) | ||||
|         { | ||||
|             lock (_partitionLocks[group.Key]) | ||||
|             { | ||||
|                 foreach (var item in group) | ||||
|                 { | ||||
|                     _partitions[group.Key][item.Key] = item.Value; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public Dictionary<string, T> GetSnapshot() | ||||
|     { | ||||
|         var snapshot = new Dictionary<string, T>(_comparer); | ||||
|  | ||||
|         for (int i = 0; i < _partitions.Length; i++) | ||||
|         { | ||||
|             lock (_partitionLocks[i]) | ||||
|             { | ||||
|                 foreach (var kvp in _partitions[i]) | ||||
|                 { | ||||
|                     snapshot[kvp.Key] = kvp.Value; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return snapshot; | ||||
|     } | ||||
|  | ||||
|     private IEnumerable<KeyValuePair<string, T>> GetAllItems() | ||||
|     { | ||||
|         for (int i = 0; i < _partitions.Length; i++) | ||||
|         { | ||||
|             lock (_partitionLocks[i]) | ||||
|             { | ||||
|                 foreach (var item in _partitions[i]) // 直接枚举原字典 | ||||
|                 { | ||||
|                     yield return item; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| public class ThreadSafeLongDictionary<T> : IDictionary<long, T>, IReadOnlyDictionary<long, T> | ||||
| { | ||||
|     private const int DEFAULT_PARTITIONS = 128; | ||||
|     private readonly Dictionary<long, T>[] _partitions; | ||||
|     private readonly object[] _partitionLocks; | ||||
|  | ||||
|     public ThreadSafeLongDictionary() : this(DEFAULT_PARTITIONS) { } | ||||
|  | ||||
|     public ThreadSafeLongDictionary(int partitionCount) | ||||
|     { | ||||
|         if (partitionCount < 1) | ||||
|             throw new ArgumentOutOfRangeException(nameof(partitionCount)); | ||||
|  | ||||
|         _partitions = new Dictionary<long, T>[partitionCount]; | ||||
|         _partitionLocks = new object[partitionCount]; | ||||
|  | ||||
|         for (int i = 0; i < partitionCount; i++) | ||||
|         { | ||||
|             _partitions[i] = new Dictionary<long, T>(); | ||||
|             _partitionLocks[i] = new object(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private int GetPartitionIndex(long key) | ||||
|     { | ||||
|         // 使用混合哈希算法减少碰撞 | ||||
|         uint hash = (uint)key; | ||||
|         hash = ((hash >> 16) ^ hash) * 0x45d9f3b; | ||||
|         hash = ((hash >> 16) ^ hash) * 0x45d9f3b; | ||||
|         hash = (hash >> 16) ^ hash; | ||||
|         return (int)(hash % _partitions.Length); | ||||
|     } | ||||
|  | ||||
|     public T this[long key] | ||||
|     { | ||||
|         get | ||||
|         { | ||||
|             int index = GetPartitionIndex(key); | ||||
|             lock (_partitionLocks[index]) | ||||
|             { | ||||
|                 return _partitions[index][key]; | ||||
|             } | ||||
|         } | ||||
|         set | ||||
|         { | ||||
|             int index = GetPartitionIndex(key); | ||||
|             lock (_partitionLocks[index]) | ||||
|             { | ||||
|                 _partitions[index][key] = value; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public ICollection<long> Keys => GetAllItems().Select(kv => kv.Key).ToList(); | ||||
|     public ICollection<T> Values => GetAllItems().Select(kv => kv.Value).ToList(); | ||||
|  | ||||
|     public int Count | ||||
|     { | ||||
|         get | ||||
|         { | ||||
|             int count = 0; | ||||
|             for (int i = 0; i < _partitions.Length; i++) | ||||
|             { | ||||
|                 lock (_partitionLocks[i]) | ||||
|                 { | ||||
|                     count += _partitions[i].Count; | ||||
|                 } | ||||
|             } | ||||
|             return count; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool IsReadOnly => false; | ||||
|  | ||||
|     IEnumerable<long> IReadOnlyDictionary<long, T>.Keys => Keys; | ||||
|  | ||||
|     IEnumerable<T> IReadOnlyDictionary<long, T>.Values => Values; | ||||
|  | ||||
|     public void Add(long key, T value) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             _partitions[index].Add(key, value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public void Add(KeyValuePair<long, T> item) => Add(item.Key, item.Value); | ||||
|  | ||||
|     public void Clear() | ||||
|     { | ||||
|         for (int i = 0; i < _partitions.Length; i++) | ||||
|         { | ||||
|             lock (_partitionLocks[i]) | ||||
|             { | ||||
|                 _partitions[i].Clear(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool Contains(KeyValuePair<long, T> item) | ||||
|     { | ||||
|         int index = GetPartitionIndex(item.Key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].TryGetValue(item.Key, out var value) && | ||||
|                    EqualityComparer<T>.Default.Equals(value, item.Value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool ContainsKey(long key) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].ContainsKey(key); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool Remove(long key) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].Remove(key); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool Remove(KeyValuePair<long, T> item) | ||||
|     { | ||||
|         int index = GetPartitionIndex(item.Key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             if (_partitions[index].TryGetValue(item.Key, out var value) && | ||||
|                 EqualityComparer<T>.Default.Equals(value, item.Value)) | ||||
|             { | ||||
|                 return _partitions[index].Remove(item.Key); | ||||
|             } | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public bool TryGetValue(long key, out T value) | ||||
|     { | ||||
|         int index = GetPartitionIndex(key); | ||||
|         lock (_partitionLocks[index]) | ||||
|         { | ||||
|             return _partitions[index].TryGetValue(key, out value); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public void CopyTo(KeyValuePair<long, T>[] array, int arrayIndex) | ||||
|     { | ||||
|         if (array == null) throw new ArgumentNullException(nameof(array)); | ||||
|         if (arrayIndex < 0) throw new ArgumentOutOfRangeException(nameof(arrayIndex)); | ||||
|         if (array.Length - arrayIndex < Count) throw new ArgumentException("Target array too small"); | ||||
|  | ||||
|         foreach (var item in GetAllItems()) | ||||
|         { | ||||
|             array[arrayIndex++] = item; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public IEnumerator<KeyValuePair<long, T>> GetEnumerator() | ||||
|     { | ||||
|         return GetAllItems().GetEnumerator(); | ||||
|     } | ||||
|  | ||||
|     IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); | ||||
|  | ||||
|     public void AddRange(IEnumerable<KeyValuePair<long, T>> items) | ||||
|     { | ||||
|         var grouped = items.GroupBy(item => GetPartitionIndex(item.Key)); | ||||
|  | ||||
|         foreach (var group in grouped) | ||||
|         { | ||||
|             lock (_partitionLocks[group.Key]) | ||||
|             { | ||||
|                 foreach (var item in group) | ||||
|                 { | ||||
|                     _partitions[group.Key][item.Key] = item.Value; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public Dictionary<long, T> GetSnapshot() | ||||
|     { | ||||
|         var snapshot = new Dictionary<long, T>(); | ||||
|  | ||||
|         for (int i = 0; i < _partitions.Length; i++) | ||||
|         { | ||||
|             lock (_partitionLocks[i]) | ||||
|             { | ||||
|                 foreach (var kvp in _partitions[i]) | ||||
|                 { | ||||
|                     snapshot[kvp.Key] = kvp.Value; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return snapshot; | ||||
|     } | ||||
|  | ||||
|     private IEnumerable<KeyValuePair<long, T>> GetAllItems() | ||||
|     { | ||||
|         for (int i = 0; i < _partitions.Length; i++) | ||||
|         { | ||||
|             lock (_partitionLocks[i]) | ||||
|             { | ||||
|                 foreach (var item in _partitions[i]) | ||||
|                 { | ||||
|                     yield return item; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public string GetPartitionStats() | ||||
|     { | ||||
|         var stats = new System.Text.StringBuilder(); | ||||
|         for (int i = 0; i < _partitions.Length; i++) | ||||
|         { | ||||
|             lock (_partitionLocks[i]) | ||||
|             { | ||||
|                 stats.AppendLine($"Partition {i}: {_partitions[i].Count} items"); | ||||
|             } | ||||
|         } | ||||
|         return stats.ToString(); | ||||
|     } | ||||
| } | ||||
| @@ -80,7 +80,14 @@ | ||||
|     "WriteVariablesAsync": "Write variables", | ||||
|     "RemoveAllCache": "Remove all cache", | ||||
|     "RemoveCache": "Remove device/channel Cache", | ||||
|     "RestartAllThread": "Restart all thread" | ||||
|     "RestartAllThread": "Restart all thread", | ||||
|     "BatchSaveChannelAsync": "BatchSaveChannel", | ||||
|     "BatchSaveDeviceAsync": "BatchSaveDevice", | ||||
|     "BatchSaveVariableAsync": "BatchSaveVariable", | ||||
|     "DeleteChannelAsync": "DeleteChannel", | ||||
|     "DeleteDeviceAsync": "DeleteDevice", | ||||
|     "DeleteVariableAsync": "DeleteVariable", | ||||
|     "InsertTestDataAsync": "InsertTestData" | ||||
|   }, | ||||
|   "ThingsGateway.Gateway.Application.RuntimeInfoController": { | ||||
|     "RuntimeInfoController": "Get runtime information", | ||||
|   | ||||
| @@ -78,7 +78,14 @@ | ||||
|     "PauseDeviceThreadAsync": "控制设备线程启停", | ||||
|     "RestartAllThread": "重启全部线程", | ||||
|     "RestartDeviceThreadAsync": "重启设备线程", | ||||
|     "WriteVariablesAsync": "写入变量" | ||||
|     "WriteVariablesAsync": "写入变量", | ||||
|     "BatchSaveChannelAsync": "保存通道", | ||||
|     "BatchSaveDeviceAsync": "保存设备", | ||||
|     "BatchSaveVariableAsync": "保存变量", | ||||
|     "DeleteChannelAsync": "删除通道", | ||||
|     "DeleteDeviceAsync": "删除设备", | ||||
|     "DeleteVariableAsync": "删除变量", | ||||
|     "InsertTestDataAsync": "增加测试数据" | ||||
|   }, | ||||
|   "ThingsGateway.Gateway.Application.RuntimeInfoController": { | ||||
|     "RuntimeInfoController": "获取运行态信息", | ||||
|   | ||||
| @@ -120,10 +120,7 @@ public class ChannelRuntime : Channel, IChannelOptions, IDisposable | ||||
|         // 通过插件名称获取插件信息 | ||||
|         PluginInfo = GlobalData.PluginService.GetList().FirstOrDefault(A => A.FullName == PluginName); | ||||
|  | ||||
|         if (PluginInfo == null) | ||||
|         { | ||||
|             //throw new Exception($"Plugin {PluginName} not found"); | ||||
|         } | ||||
|         GlobalData.Channels.TryRemove(Id, out _); | ||||
|  | ||||
|         GlobalData.Channels.TryAdd(Id, this); | ||||
|  | ||||
|   | ||||
| @@ -21,7 +21,7 @@ public class DeviceBasicData | ||||
| { | ||||
|     [Newtonsoft.Json.JsonIgnore] | ||||
|     [System.Text.Json.Serialization.JsonIgnore] | ||||
|     public DeviceRuntime DeviceRuntime{ get; set; } | ||||
|     public DeviceRuntime DeviceRuntime { get; set; } | ||||
|  | ||||
|     /// <inheritdoc cref="PrimaryIdEntity.Id"/> | ||||
|     public long Id { get; set; } | ||||
|   | ||||
| @@ -224,7 +224,16 @@ public class DeviceRuntime : Device, IDisposable | ||||
|     [AutoGenerateColumn(Ignore = true)] | ||||
|     public IDriver? Driver { get; internal set; } | ||||
|  | ||||
|     [System.Text.Json.Serialization.JsonIgnore] | ||||
|     [Newtonsoft.Json.JsonIgnore] | ||||
|     [AdaptIgnore] | ||||
|     [AutoGenerateColumn(Ignore = true)] | ||||
|     public IRpcDriver? RpcDriver { get; set; } | ||||
|  | ||||
|     [System.Text.Json.Serialization.JsonIgnore] | ||||
|     [Newtonsoft.Json.JsonIgnore] | ||||
|     [AutoGenerateColumn(Ignore = true)] | ||||
|     public string? Tag { get; set; } | ||||
|  | ||||
|     public void Init(ChannelRuntime channelRuntime) | ||||
|     { | ||||
|   | ||||
| @@ -87,18 +87,13 @@ public class VariableMethod | ||||
|             } | ||||
|  | ||||
|             dynamic result; | ||||
|             switch (MethodInfo.TaskType) | ||||
|             switch (MethodInfo.ReturnKind) | ||||
|             { | ||||
|                 case TaskReturnType.Task: | ||||
|                     await MethodInfo.InvokeAsync(driverBase, os).ConfigureAwait(false); | ||||
|                     result = OperResult.Success; | ||||
|                 case MethodReturnKind.Awaitable: | ||||
|                 case MethodReturnKind.AwaitableObject: | ||||
|                     result = await MethodInfo.InvokeAsync(driverBase, os).ConfigureAwait(false); | ||||
|                     break; | ||||
|  | ||||
|                 case TaskReturnType.TaskObject: | ||||
|                     result = await MethodInfo.InvokeObjectAsync(driverBase, os).ConfigureAwait(false); | ||||
|                     break; | ||||
|  | ||||
|                 case TaskReturnType.None: | ||||
|                 default: | ||||
|                     result = MethodInfo.Invoke(driverBase, os); | ||||
|                     break; | ||||
|   | ||||
| @@ -12,9 +12,12 @@ using BootstrapBlazor.Components; | ||||
|  | ||||
| using Mapster; | ||||
|  | ||||
| using Newtonsoft.Json.Linq; | ||||
|  | ||||
| using SqlSugar; | ||||
|  | ||||
| using ThingsGateway.Gateway.Application.Extensions; | ||||
| using ThingsGateway.NewLife.Extension; | ||||
| using ThingsGateway.NewLife.Json.Extension; | ||||
|  | ||||
| namespace ThingsGateway.Gateway.Application; | ||||
| @@ -32,10 +35,6 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|     private bool? _isOnlineChanged; | ||||
|     protected object? _value; | ||||
|  | ||||
|     public VariableRuntime() | ||||
|     { | ||||
|          | ||||
|     } | ||||
|     /// <summary> | ||||
|     /// 变化时间 | ||||
|     /// </summary> | ||||
| @@ -142,7 +141,7 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|     /// 实时值 | ||||
|     /// </summary> | ||||
|     [AutoGenerateColumn(Visible = true, Order = 6)] | ||||
|     public string? RuntimeType => Value?.GetType()?.Name; | ||||
|     public string? RuntimeType => Value?.GetType()?.ToString(); | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 设置变量值与时间/质量戳 | ||||
| @@ -210,6 +209,11 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             if (data is JToken jToken) | ||||
|             { | ||||
|                 data = jToken.GetObjectFromJToken(); | ||||
|             } | ||||
|  | ||||
|             //判断变化,插件传入的Value可能是基础类型,也有可能是class,比较器无法识别是否变化,这里json处理序列化比较 | ||||
|             //检查IComparable | ||||
|             if (!data.Equals(_value)) | ||||
| @@ -221,7 +225,7 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|                 else | ||||
|                 { | ||||
|                     if (_value != null) | ||||
|                         changed = data.ToJsonNetString(false) != _value.ToJsonNetString(false); | ||||
|                         changed = data.ToSystemTextJsonString(false) != _value.ToSystemTextJsonString(false); | ||||
|                     else | ||||
|                         changed = true; | ||||
|                 } | ||||
| @@ -336,7 +340,7 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|     public void Init(DeviceRuntime deviceRuntime) | ||||
|     { | ||||
|  | ||||
|         GlobalData.AlarmEnableIdVariables.TryRemove(Id, out _); | ||||
|         GlobalData.AlarmEnableIdVariables.Remove(Id); | ||||
|         if (GlobalData.RealAlarmIdVariables.TryRemove(Id, out var oldAlarm)) | ||||
|         { | ||||
|             oldAlarm.EventType = EventTypeEnum.Finish; | ||||
| @@ -345,12 +349,12 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|         } | ||||
|  | ||||
|  | ||||
|         DeviceRuntime?.VariableRuntimes?.TryRemove(Name, out _); | ||||
|         DeviceRuntime?.VariableRuntimes?.Remove(Name); | ||||
|  | ||||
|         DeviceRuntime = deviceRuntime; | ||||
|  | ||||
|         DeviceRuntime?.VariableRuntimes?.TryAdd(Name, this); | ||||
|         GlobalData.IdVariables.TryRemove(Id, out _); | ||||
|         GlobalData.IdVariables.Remove(Id); | ||||
|         GlobalData.IdVariables.TryAdd(Id, this); | ||||
|         if (AlarmEnable) | ||||
|         { | ||||
| @@ -362,11 +366,11 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|     public void Dispose() | ||||
|     { | ||||
|  | ||||
|         DeviceRuntime?.VariableRuntimes?.TryRemove(Name, out _); | ||||
|         DeviceRuntime?.VariableRuntimes?.Remove(Name); | ||||
|  | ||||
|         GlobalData.IdVariables.TryRemove(Id, out _); | ||||
|         GlobalData.IdVariables.Remove(Id); | ||||
|  | ||||
|         GlobalData.AlarmEnableIdVariables.TryRemove(Id, out _); | ||||
|         GlobalData.AlarmEnableIdVariables.Remove(Id); | ||||
|         if (GlobalData.RealAlarmIdVariables.TryRemove(Id, out var oldAlarm)) | ||||
|         { | ||||
|             oldAlarm.EventType = EventTypeEnum.Finish; | ||||
| @@ -378,7 +382,7 @@ public class VariableRuntime : Variable, IVariable, IDisposable | ||||
|     } | ||||
|  | ||||
|     /// <inheritdoc/> | ||||
|     public async ValueTask<OperResult> RpcAsync(string value, string? executive = "brower", CancellationToken cancellationToken = default) | ||||
|     public async ValueTask<IOperResult> RpcAsync(string value, string? executive = "brower", CancellationToken cancellationToken = default) | ||||
|     { | ||||
|         var data = await GlobalData.RpcService.InvokeDeviceMethodAsync(executive, new Dictionary<string, Dictionary<string, string>>() | ||||
|         { | ||||
|   | ||||
| @@ -386,7 +386,7 @@ internal sealed class ChannelService : BaseService<Channel>, IChannelService | ||||
|                     if (channel == null) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, Localizer["ImportNullError"])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["ImportNullError"])); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -409,7 +409,7 @@ internal sealed class ChannelService : BaseService<Channel>, IChannelService | ||||
|                     if (stringBuilder.Length > 0) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, stringBuilder.ToString())); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, stringBuilder.ToString())); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -430,19 +430,19 @@ internal sealed class ChannelService : BaseService<Channel>, IChannelService | ||||
|  | ||||
|                     if (channel.IsUp && ((dataScope != null && dataScope?.Count > 0 && !dataScope.Contains(channel.CreateOrgId)) || dataScope?.Count == 0 && channel.CreateUserId != UserManager.UserId)) | ||||
|                     { | ||||
|                         importPreviewOutput.Results.Add((row++, false, "Operation not permitted")); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, "Operation not permitted")); | ||||
|                     } | ||||
|                     else | ||||
|                     { | ||||
|                         channels.Add(channel); | ||||
|                         importPreviewOutput.Results.Add((row++, true, null)); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), true, null)); | ||||
|                     } | ||||
|                     return; | ||||
|                 } | ||||
|                 catch (Exception ex) | ||||
|                 { | ||||
|                     importPreviewOutput.HasError = true; | ||||
|                     importPreviewOutput.Results.Add((row++, false, ex.Message)); | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, ex.Message)); | ||||
|                     return; | ||||
|                 } | ||||
|             }); | ||||
|   | ||||
| @@ -432,7 +432,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     if (device == null) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, Localizer["ImportNullError"])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["ImportNullError"])); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -449,7 +449,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                         { | ||||
|                             // 如果找不到对应的冗余设备,则添加错误信息到导入预览结果并返回 | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((row++, false, Localizer["RedundantDeviceError"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["RedundantDeviceError"])); | ||||
|                             return; | ||||
|                         } | ||||
|                     } | ||||
| @@ -459,7 +459,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                         if (device.RedundantEnable) | ||||
|                         { | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((row++, false, Localizer["RedundantDeviceError"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["RedundantDeviceError"])); | ||||
|                             return; | ||||
|                         } | ||||
|                     } | ||||
| @@ -473,7 +473,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                         { | ||||
|                             // 如果找不到对应的通道信息,则添加错误信息到导入预览结果并返回 | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((row++, false, Localizer["ChannelError"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["ChannelError"])); | ||||
|                             return; | ||||
|                         } | ||||
|                     } | ||||
| @@ -481,7 +481,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     { | ||||
|                         // 如果未提供通道信息,则添加错误信息到导入预览结果并返回 | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, Localizer["ChannelError"])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["ChannelError"])); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -504,7 +504,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     if (stringBuilder.Length > 0) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, stringBuilder.ToString())); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, stringBuilder.ToString())); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -527,12 +527,12 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     // 将设备添加到设备列表中,并添加成功信息到导入预览结果 | ||||
|                     if (device.IsUp && ((dataScope != null && dataScope?.Count > 0 && !dataScope.Contains(device.CreateOrgId)) || dataScope?.Count == 0 && device.CreateUserId != UserManager.UserId)) | ||||
|                     { | ||||
|                         importPreviewOutput.Results.Add((row++, false, "Operation not permitted")); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, "Operation not permitted")); | ||||
|                     } | ||||
|                     else | ||||
|                     { | ||||
|                         devices.Add(device); | ||||
|                         importPreviewOutput.Results.Add((row++, true, null)); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), true, null)); | ||||
|                     } | ||||
|                     return; | ||||
|                 } | ||||
| @@ -540,7 +540,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                 { | ||||
|                     // 捕获异常并添加错误信息到导入预览结果 | ||||
|                     importPreviewOutput.HasError = true; | ||||
|                     importPreviewOutput.Results.Add((row++, false, ex.Message)); | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, ex.Message)); | ||||
|                     return; | ||||
|                 } | ||||
|             }); | ||||
| @@ -573,7 +573,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|             if (driverPluginType == null) | ||||
|             { | ||||
|                 importPreviewOutput.HasError = true; | ||||
|                 importPreviewOutput.Results.Add((row++, false, Localizer["NotNull", sheetName])); | ||||
|                 importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["NotNull", sheetName])); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
| @@ -617,7 +617,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     if (propertys.Item1 == null) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, Localizer["PluginNotNull"])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["PluginNotNull"])); | ||||
|                         continue; | ||||
|                     } | ||||
|  | ||||
| @@ -625,7 +625,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     if (!item.TryGetValue(ExportString.DeviceName, out var deviceName)) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, Localizer["DeviceNotNull"])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["DeviceNotNull"])); | ||||
|                         continue; | ||||
|                     } | ||||
|  | ||||
| @@ -637,7 +637,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     if (!hasDevice) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, Localizer["NotNull", value])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["NotNull", value])); | ||||
|                         continue; | ||||
|                     } | ||||
|  | ||||
| @@ -648,7 +648,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     if (pluginProp == null) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, Localizer["ImportNullError"])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["ImportNullError"])); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -671,7 +671,7 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|                     if (stringBuilder.Length > 0) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((row++, false, stringBuilder.ToString())); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, stringBuilder.ToString())); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -687,14 +687,14 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService | ||||
|  | ||||
|                     // 更新设备导入预览数据中对应设备的属性信息,并添加成功信息到导入预览结果 | ||||
|                     deviceImportPreview.Data[value].DevicePropertys = devices; | ||||
|                     importPreviewOutput.Results.Add((row++, true, null)); | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Increment(ref row), true, null)); | ||||
|                     continue; | ||||
|                 } | ||||
|                 catch (Exception ex) | ||||
|                 { | ||||
|                     // 捕获异常并添加错误信息到导入预览结果并返回 | ||||
|                     importPreviewOutput.HasError = true; | ||||
|                     importPreviewOutput.Results.Add((row++, false, ex.Message)); | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, ex.Message)); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -21,8 +21,8 @@ namespace ThingsGateway.Gateway.Application; | ||||
| internal sealed class ChannelThreadManage : IChannelThreadManage | ||||
| { | ||||
|     private ILogger _logger; | ||||
|     private static IDispatchService<ChannelRuntime> channelRuntimeDispatchService; | ||||
|     private static IDispatchService<ChannelRuntime> ChannelRuntimeDispatchService | ||||
|     private IDispatchService<ChannelRuntime> channelRuntimeDispatchService; | ||||
|     private IDispatchService<ChannelRuntime> ChannelRuntimeDispatchService | ||||
|     { | ||||
|         get | ||||
|         { | ||||
|   | ||||
| @@ -38,8 +38,8 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage | ||||
|     /// </summary> | ||||
|     public static volatile int CycleInterval = ManageHelper.ChannelThreadOptions.MaxCycleInterval; | ||||
|  | ||||
|     private static IDispatchService<DeviceRuntime> devicelRuntimeDispatchService; | ||||
|     private static IDispatchService<DeviceRuntime> DeviceRuntimeDispatchService | ||||
|     private IDispatchService<DeviceRuntime> devicelRuntimeDispatchService; | ||||
|     private IDispatchService<DeviceRuntime> DeviceRuntimeDispatchService | ||||
|     { | ||||
|         get | ||||
|         { | ||||
| @@ -96,6 +96,7 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage | ||||
|     } | ||||
|  | ||||
|     #endregion 动态配置 | ||||
|     Microsoft.Extensions.Logging.ILogger? _logger; | ||||
|  | ||||
|     /// <summary> | ||||
|     /// 通道线程构造函数,用于初始化通道线程实例。 | ||||
| @@ -113,9 +114,9 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage | ||||
|         // 设置通道信息 | ||||
|         CurrentChannel = channelRuntime; | ||||
|  | ||||
|         var logger = App.RootServices.GetService<Microsoft.Extensions.Logging.ILoggerFactory>().CreateLogger($"DeviceThreadManage[{channelRuntime.Name}]"); | ||||
|         _logger = App.RootServices.GetService<Microsoft.Extensions.Logging.ILoggerFactory>().CreateLogger($"DeviceThreadManage[{channelRuntime.Name}]"); | ||||
|         // 添加默认日志记录器 | ||||
|         LogMessage.AddLogger(new EasyLogger(logger.Log_Out) { LogLevel = TouchSocket.Core.LogLevel.Trace }); | ||||
|         LogMessage.AddLogger(new EasyLogger(_logger.Log_Out) { LogLevel = TouchSocket.Core.LogLevel.Trace }); | ||||
|  | ||||
|         // 根据配置获取通道实例 | ||||
|         Channel = channelRuntime.GetChannel(config); | ||||
| @@ -898,7 +899,7 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage | ||||
|         try | ||||
|         { | ||||
|             await NewDeviceLock.WaitAsync().ConfigureAwait(false); | ||||
|  | ||||
|             _logger?.TryDispose(); | ||||
|             await PrivateRemoveDevicesAsync(Drivers.Keys).ConfigureAwait(false); | ||||
|             if (Channel?.Collects.Count == 0) | ||||
|                 Channel?.SafeDispose(); | ||||
|   | ||||
| @@ -19,5 +19,5 @@ public interface IRpcService | ||||
|     /// <param name="deviceDatas">指定键为变量名称,值为附带方法参数或写入值,方法参数会按逗号分割解析</param> | ||||
|     /// <param name="cancellationToken"><see cref="CancellationToken"/> 取消令箭</param> | ||||
|     /// <returns></returns> | ||||
|     Task<Dictionary<string, Dictionary<string, OperResult>>> InvokeDeviceMethodAsync(string sourceDes, Dictionary<string, Dictionary<string, string>> deviceDatas, CancellationToken cancellationToken = default); | ||||
|     Task<Dictionary<string, Dictionary<string, IOperResult>>> InvokeDeviceMethodAsync(string sourceDes, Dictionary<string, Dictionary<string, string>> deviceDatas, CancellationToken cancellationToken = default); | ||||
| } | ||||
|   | ||||
| @@ -42,16 +42,15 @@ internal sealed class RpcService : IRpcService | ||||
|     private IStringLocalizer Localizer { get; } | ||||
|  | ||||
|     /// <inheritdoc /> | ||||
|     public async Task<Dictionary<string, Dictionary<string, OperResult>>> InvokeDeviceMethodAsync(string sourceDes, Dictionary<string, Dictionary<string, string>> deviceDatas, CancellationToken cancellationToken = default) | ||||
|     public async Task<Dictionary<string, Dictionary<string, IOperResult>>> InvokeDeviceMethodAsync(string sourceDes, Dictionary<string, Dictionary<string, string>> deviceDatas, CancellationToken cancellationToken = default) | ||||
|     { | ||||
|         // 初始化用于存储将要写入的变量和方法的字典 | ||||
|         Dictionary<CollectBase, Dictionary<VariableRuntime, JToken>> writeVariables = new(); | ||||
|         Dictionary<CollectBase, Dictionary<VariableRuntime, JToken>> writeMethods = new(); | ||||
|         Dictionary<IRpcDriver, Dictionary<VariableRuntime, JToken>> writeVariables = new(); | ||||
|         Dictionary<IRpcDriver, Dictionary<VariableRuntime, JToken>> writeMethods = new(); | ||||
|         // 用于存储结果的并发字典 | ||||
|         ConcurrentDictionary<string, Dictionary<string, OperResult>> results = new(); | ||||
|         ConcurrentDictionary<string, Dictionary<string, IOperResult>> results = new(); | ||||
|         deviceDatas.ForEach(a => results.TryAdd(a.Key, new())); | ||||
|  | ||||
|  | ||||
|         var deviceDict = GlobalData.Devices; | ||||
|  | ||||
|         // 对每个要操作的变量进行检查和处理 | ||||
| @@ -68,7 +67,8 @@ internal sealed class RpcService : IRpcService | ||||
|             } | ||||
|  | ||||
|             // 查找变量对应的设备 | ||||
|             var collect = device.Driver as CollectBase; | ||||
|             var collect = device.Driver as IRpcDriver; | ||||
|             collect ??= device.RpcDriver; | ||||
|             if (collect == null) | ||||
|             { | ||||
|                 // 如果设备不存在,则添加错误信息到结果中并继续下一个设备的处理 | ||||
| @@ -78,7 +78,7 @@ internal sealed class RpcService : IRpcService | ||||
|                 continue; | ||||
|             } | ||||
|             // 检查设备状态,如果设备处于暂停状态,则添加相应的错误信息到结果中并继续下一个变量的处理 | ||||
|             if (collect.CurrentDevice.DeviceStatus == DeviceStatusEnum.Pause) | ||||
|             if (device.DeviceStatus == DeviceStatusEnum.Pause) | ||||
|             { | ||||
|                 deviceData.Value.ForEach(a => | ||||
|                 results[deviceData.Key].TryAdd(a.Key, new OperResult(Localizer["DevicePause", deviceData.Key])) | ||||
| @@ -136,46 +136,51 @@ internal sealed class RpcService : IRpcService | ||||
|                 // 写入日志 | ||||
|                 foreach (var resultItem in result) | ||||
|                 { | ||||
|                     var empty = string.IsNullOrEmpty(resultItem.Key); | ||||
|                     string operObj = empty ? deviceDatas[driverData.Key.DeviceName].Select(x => x.Key).ToJsonNetString() : resultItem.Key; | ||||
|  | ||||
|                     string parJson = empty ? deviceDatas.Select(x => x.Value).ToJsonNetString() : deviceDatas[driverData.Key.DeviceName][resultItem.Key]; | ||||
|  | ||||
|                     if (!resultItem.Value.IsSuccess || _rpcLogOptions.SuccessLog) | ||||
|                         _logQueues.Enqueue( | ||||
|                             new RpcLog() | ||||
|                             { | ||||
|                                 LogTime = DateTime.Now, | ||||
|                                 OperateMessage = resultItem.Value.IsSuccess ? null : resultItem.Value.ToString(), | ||||
|                                 IsSuccess = resultItem.Value.IsSuccess, | ||||
|                                 OperateMethod = Localizer["WriteVariable"], | ||||
|                                 OperateDevice = driverData.Key.DeviceName, | ||||
|                                 OperateObject = operObj, | ||||
|                                 OperateSource = sourceDes, | ||||
|                                 ParamJson = parJson, | ||||
|                                 ResultJson = null | ||||
|                             } | ||||
|                         ); | ||||
|  | ||||
|                     // 不返回详细错误 | ||||
|                     if (!resultItem.Value.IsSuccess) | ||||
|                     foreach (var variableResult in resultItem.Value) | ||||
|                     { | ||||
|                         OperResult result1 = resultItem.Value; | ||||
|                         result1.Exception = null; | ||||
|                         result[resultItem.Key] = result1; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 // 将结果添加到结果字典中 | ||||
|                 results[driverData.Key.DeviceName].AddRange(result); | ||||
|                         string operObj = variableResult.Key; | ||||
|  | ||||
|                         string parJson = deviceDatas[resultItem.Key][variableResult.Key]; | ||||
|  | ||||
|                         if (!variableResult.Value.IsSuccess || _rpcLogOptions.SuccessLog) | ||||
|                             _logQueues.Enqueue( | ||||
|                                 new RpcLog() | ||||
|                                 { | ||||
|                                     LogTime = DateTime.Now, | ||||
|                                     OperateMessage = variableResult.Value.IsSuccess ? null : variableResult.Value.ToString(), | ||||
|                                     IsSuccess = variableResult.Value.IsSuccess, | ||||
|                                     OperateMethod = Localizer["WriteVariable"], | ||||
|                                     OperateDevice = resultItem.Key, | ||||
|                                     OperateObject = operObj, | ||||
|                                     OperateSource = sourceDes, | ||||
|                                     ParamJson = parJson, | ||||
|                                     ResultJson = null | ||||
|                                 } | ||||
|                             ); | ||||
|  | ||||
|                         // 不返回详细错误 | ||||
|                         if (!variableResult.Value.IsSuccess) | ||||
|                         { | ||||
|                             var result1 = variableResult.Value; | ||||
|                             result1.Exception = null; | ||||
|                             resultItem.Value[variableResult.Key] = result1; | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     // 将结果添加到结果字典中 | ||||
|                     results[resultItem.Key].AddRange(resultItem.Value); | ||||
|                 } | ||||
|             } | ||||
|             catch (Exception ex) | ||||
|             { | ||||
|                 // 将异常信息添加到结果字典中 | ||||
|                 results[driverData.Key.DeviceName].AddRange(driverData.Value.Select((KeyValuePair<VariableRuntime, JToken> a) => | ||||
|                 foreach (var item in driverData.Value) | ||||
|                 { | ||||
|                     return new KeyValuePair<string, OperResult>(a.Key.Name, new OperResult(ex)); | ||||
|                 })); | ||||
|                     results[item.Key.DeviceName].Add(item.Key.Name, new OperResult(ex)); | ||||
|                 } | ||||
|             } | ||||
|         }, cancellationToken).ConfigureAwait(false); | ||||
|  | ||||
| @@ -192,42 +197,51 @@ internal sealed class RpcService : IRpcService | ||||
|                 // 写入日志 | ||||
|                 foreach (var resultItem in result) | ||||
|                 { | ||||
|                     // 写入日志 | ||||
|                     if (!resultItem.Value.IsSuccess || _rpcLogOptions.SuccessLog) | ||||
|                         _logQueues.Enqueue( | ||||
|                             new RpcLog() | ||||
|                             { | ||||
|                                 LogTime = DateTime.Now, | ||||
|                                 OperateMessage = resultItem.Value.IsSuccess ? null : resultItem.Value.ToString(), | ||||
|                                 IsSuccess = resultItem.Value.IsSuccess, | ||||
|                                 OperateMethod = operateMethods[resultItem.Key], | ||||
|                                 OperateDevice = driverData.Key.DeviceName, | ||||
|                                 OperateObject = resultItem.Key, | ||||
|                                 OperateSource = sourceDes, | ||||
|                                 ParamJson = deviceDatas[driverData.Key.DeviceName][resultItem.Key]?.ToString(), | ||||
|                                 ResultJson = resultItem.Value.Content?.ToJsonNetString() | ||||
|                             } | ||||
|                         ); | ||||
|  | ||||
|                     // 不返回详细错误 | ||||
|                     if (!resultItem.Value.IsSuccess) | ||||
|                     foreach (var variableResult in resultItem.Value) | ||||
|                     { | ||||
|                         OperResult<object> result1 = resultItem.Value; | ||||
|                         result1.Exception = null; | ||||
|                         result[resultItem.Key] = result1; | ||||
|                     } | ||||
|                 } | ||||
|                         string operObj = variableResult.Key; | ||||
|  | ||||
|                 results[driverData.Key.DeviceName].AddRange(result.ToDictionary(a => a.Key, a => (OperResult)a.Value)); | ||||
|                         string parJson = deviceDatas[resultItem.Key][variableResult.Key]; | ||||
|  | ||||
|                         // 写入日志 | ||||
|                         if (!variableResult.Value.IsSuccess || _rpcLogOptions.SuccessLog) | ||||
|                             _logQueues.Enqueue( | ||||
|                                 new RpcLog() | ||||
|                                 { | ||||
|                                     LogTime = DateTime.Now, | ||||
|                                     OperateMessage = variableResult.Value.IsSuccess ? null : variableResult.Value.ToString(), | ||||
|                                     IsSuccess = variableResult.Value.IsSuccess, | ||||
|                                     OperateMethod = operateMethods[variableResult.Key], | ||||
|                                     OperateDevice = resultItem.Key, | ||||
|                                     OperateObject = operObj, | ||||
|                                     OperateSource = sourceDes, | ||||
|                                     ParamJson = parJson?.ToString(), | ||||
|                                     ResultJson = variableResult.Value is IOperResult<object> operResult ? operResult.Content?.ToSystemTextJsonString() : string.Empty | ||||
|                                 } | ||||
|                             ); | ||||
|  | ||||
|                         // 不返回详细错误 | ||||
|                         if (!variableResult.Value.IsSuccess) | ||||
|                         { | ||||
|                             var result1 = variableResult.Value; | ||||
|                             result1.Exception = null; | ||||
|                             resultItem.Value[variableResult.Key] = result1; | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|  | ||||
|                     results[resultItem.Key].AddRange(resultItem.Value.ToDictionary(a => a.Key, a => a.Value)); | ||||
|                 } | ||||
|  | ||||
|             } | ||||
|             catch (Exception ex) | ||||
|             { | ||||
|                 // 将异常信息添加到结果字典中 | ||||
|                 results[driverData.Key.DeviceName].AddRange(driverData.Value.Select((KeyValuePair<VariableRuntime, JToken> a) => | ||||
|                 foreach (var item in driverData.Value) | ||||
|                 { | ||||
|                     return new KeyValuePair<string, OperResult>(a.Key.Name, new OperResult(ex)); | ||||
|                 })); | ||||
|                     results[item.Key.DeviceName].Add(item.Key.Name, new OperResult(ex)); | ||||
|                 } | ||||
|             } | ||||
|         }, cancellationToken).ConfigureAwait(false); | ||||
|  | ||||
|   | ||||
| @@ -619,7 +619,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                     if (deviceId == null) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, Localizer["NotNull", deviceName])); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["NotNull", deviceName])); | ||||
|                         return; | ||||
|                     } | ||||
|                     // 手动补录变量ID和设备ID | ||||
| @@ -642,7 +642,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                     if (stringBuilder.Length > 0) | ||||
|                     { | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, stringBuilder.ToString())); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, stringBuilder.ToString())); | ||||
|                         return; | ||||
|                     } | ||||
|  | ||||
| @@ -661,19 +661,19 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                     } | ||||
|                     if (device.IsUp && ((dataScope != null && dataScope?.Count > 0 && !dataScope.Contains(variable.CreateOrgId)) || dataScope?.Count == 0 && variable.CreateUserId != UserManager.UserId)) | ||||
|                     { | ||||
|                         importPreviewOutput.Results.Add((row++, false, "Operation not permitted")); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, "Operation not permitted")); | ||||
|                     } | ||||
|                     else | ||||
|                     { | ||||
|                         variables.Add(variable); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), true, null)); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), true, null)); | ||||
|                     } | ||||
|                 } | ||||
|                 catch (Exception ex) | ||||
|                 { | ||||
|                     // 捕获异常并添加错误信息到导入预览结果 | ||||
|                     importPreviewOutput.HasError = true; | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, ex.Message)); | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, ex.Message)); | ||||
|                 } | ||||
|             }); | ||||
|  | ||||
| @@ -702,7 +702,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                 if (driverPluginType == null) | ||||
|                 { | ||||
|                     importPreviewOutput.HasError = true; | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, Localizer["NotNull", sheetName])); | ||||
|                     importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["NotNull", sheetName])); | ||||
|                     return deviceImportPreview; | ||||
|                 } | ||||
|  | ||||
| @@ -742,7 +742,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                         if (propertys.Item3?.Count == null || propertys.Item1 == null) | ||||
|                         { | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, Localizer["ImportNullError"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["ImportNullError"])); | ||||
|                             return; | ||||
|                         } | ||||
|  | ||||
| @@ -753,7 +753,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                         if (pluginProp == null) | ||||
|                         { | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, Localizer["ImportNullError"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["ImportNullError"])); | ||||
|                             return; | ||||
|                         } | ||||
|  | ||||
| @@ -768,13 +768,13 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                         if (businessDevName == null || businessDevice == null || collectDevName == null || collectDevice == null) | ||||
|                         { | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, Localizer["DeviceNotNull"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["DeviceNotNull"])); | ||||
|                             return; | ||||
|                         } | ||||
|                         if (variableNameObj == null) | ||||
|                         { | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, Localizer["VariableNotNull"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["VariableNotNull"])); | ||||
|                             return; | ||||
|                         } | ||||
|  | ||||
| @@ -797,7 +797,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                         if (stringBuilder.Length > 0) | ||||
|                         { | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, stringBuilder.ToString())); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, stringBuilder.ToString())); | ||||
|                             return; | ||||
|                         } | ||||
|  | ||||
| @@ -818,12 +818,12 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                         { | ||||
|                             deviceVariable.VariablePropertys ??= new(); | ||||
|                             deviceVariable.VariablePropertys?.AddOrUpdate(businessDevice.Id, dependencyProperties); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), true, null)); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), true, null)); | ||||
|                         } | ||||
|                         else | ||||
|                         { | ||||
|                             importPreviewOutput.HasError = true; | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, Localizer["VariableNotNull"])); | ||||
|                             importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, Localizer["VariableNotNull"])); | ||||
|                             return; | ||||
|                         } | ||||
|                     } | ||||
| @@ -831,7 +831,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|                     { | ||||
|                         // 捕获异常并添加错误信息到导入预览结果 | ||||
|                         importPreviewOutput.HasError = true; | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, ex.Message)); | ||||
|                         importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, ex.Message)); | ||||
|                     } | ||||
|                 }); | ||||
|             } | ||||
| @@ -839,7 +839,7 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService | ||||
|             { | ||||
|                 // 捕获异常并添加错误信息到导入预览结果 | ||||
|                 importPreviewOutput.HasError = true; | ||||
|                 importPreviewOutput.Results.Add((Interlocked.Add(ref row, 1), false, ex.Message)); | ||||
|                 importPreviewOutput.Results.Add((Interlocked.Increment(ref row), false, ex.Message)); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -187,7 +187,7 @@ public static class VariableServiceHelpers | ||||
|  | ||||
|  | ||||
|  | ||||
|         variableExports = new(variableExports.OrderBy(a => a[ExportString.DeviceName]).ThenBy(a => a[varName])); | ||||
|         //variableExports = new(variableExports.OrderBy(a => a[ExportString.DeviceName]).ThenBy(a => a[varName])); | ||||
|  | ||||
|         //添加设备页 | ||||
|         sheets.Add(ExportString.VariableName, variableExports); | ||||
|   | ||||
| @@ -8,8 +8,8 @@ | ||||
| 	<ItemGroup> | ||||
| 		<PackageReference Include="Portable.BouncyCastle" Version="1.9.0" /> | ||||
| 		<PackageReference Include="Rougamo.Fody" Version="5.0.0" /> | ||||
| 		<PackageReference Include="TouchSocket.Dmtp" Version="3.1.2" /> | ||||
| 		<PackageReference Include="TouchSocket.WebApi.Swagger" Version="3.1.2" /> | ||||
| 		<PackageReference Include="TouchSocket.Dmtp" Version="3.1.5" /> | ||||
| 		<PackageReference Include="TouchSocket.WebApi.Swagger" Version="3.1.5" /> | ||||
| 		<PackageReference Include="ThingsGateway.Authentication" Version="$(AuthenticationVersion)" /> | ||||
|  | ||||
| 	</ItemGroup> | ||||
|   | ||||
| @@ -74,7 +74,7 @@ public partial class TcpServiceComponent : IDriverUIBase | ||||
|                     } | ||||
|                     return data; | ||||
|                 }).ToList(); | ||||
|                 data[i].PluginInfos = pluginInfos.ToJsonNetString(); | ||||
|                 data[i].PluginInfos = pluginInfos.ToSystemTextJsonString(); | ||||
|             } | ||||
|  | ||||
|             var query = data.GetQueryData(options); | ||||
|   | ||||
| @@ -34,7 +34,7 @@ public partial class USheet | ||||
|             await _sheetExcel.PushDataAsync(new UniverSheetData() | ||||
|             { | ||||
|                 CommandName = "SetWorkbook", | ||||
|                 WorkbookData = Model.ToJsonNetString(), | ||||
|                 WorkbookData = Model.ToSystemTextJsonString(), | ||||
|             }); | ||||
|         }); | ||||
|     } | ||||
|   | ||||
| @@ -403,7 +403,7 @@ finally | ||||
|                 Title = GatewayLocalizer["DeleteConfirmTitle"], | ||||
|                 BodyTemplate = (__builder) => | ||||
|                 { | ||||
|                     var data = modelIds.Select(a => a.Name).ToJsonNetString(); | ||||
|                     var data = modelIds.Select(a => a.Name).ToSystemTextJsonString(); | ||||
|                     __builder.OpenElement(0, "div"); | ||||
|                     __builder.AddAttribute(1, "class", "w-100 "); | ||||
|                     __builder.OpenElement(2, "span"); | ||||
| @@ -932,7 +932,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e => | ||||
|                 Title = GatewayLocalizer["DeleteConfirmTitle"], | ||||
|                 BodyTemplate = (__builder) => | ||||
|                 { | ||||
|                     var data = modelIds.Select(a => a.Name).ToJsonNetString(); | ||||
|                     var data = modelIds.Select(a => a.Name).ToSystemTextJsonString(); | ||||
|                     __builder.OpenElement(0, "div"); | ||||
|                     __builder.AddAttribute(1, "class", "w-100 "); | ||||
|                     __builder.OpenElement(2, "span"); | ||||
|   | ||||
| @@ -34,7 +34,7 @@ public partial class ScriptCheck | ||||
|     private Type type; | ||||
|     protected override void OnInitialized() | ||||
|     { | ||||
|         Input = Data.ToJsonNetString(); | ||||
|         Input = Data.ToSystemTextJsonString(); | ||||
|         type = Data.GetType(); | ||||
|         base.OnInitialized(); | ||||
|     } | ||||
| @@ -45,7 +45,7 @@ public partial class ScriptCheck | ||||
|         { | ||||
|             Data = (IEnumerable<object>)Newtonsoft.Json.JsonConvert.DeserializeObject(Input, type); | ||||
|             var value = Data.GetDynamicModel(Script); | ||||
|             Output = value.ToJsonNetString(); | ||||
|             Output = value.ToSystemTextJsonString(); | ||||
|         } | ||||
|  | ||||
|         catch (Exception ex) | ||||
|   | ||||
| @@ -34,7 +34,7 @@ public partial class VariableEditComponent | ||||
|         var ret = ""; | ||||
|         if (d != null) | ||||
|         { | ||||
|             ret = d.ToJsonNetString(); | ||||
|             ret = d.ToSystemTextJsonString(); | ||||
|         } | ||||
|         return ret; | ||||
|     } | ||||
|   | ||||
| @@ -58,7 +58,7 @@ public partial class VariableRuntimeInfo : IDisposable | ||||
|         var ret = ""; | ||||
|         if (d is TableColumnContext<VariableRuntime, object?> data && data?.Value != null) | ||||
|         { | ||||
|             ret = data.Value.ToJsonNetString(); | ||||
|             ret = data.Value.ToSystemTextJsonString(); | ||||
|         } | ||||
|         return Task.FromResult(ret); | ||||
|     } | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
| 	</PropertyGroup> | ||||
| 	<ItemGroup> | ||||
| 		<ProjectReference Include="..\ThingsGateway.Gateway.Application\ThingsGateway.Gateway.Application.csproj" /> | ||||
| 		<PackageReference Include="BootstrapBlazor.UniverSheet" Version="9.0.4" /> | ||||
| 		<PackageReference Include="BootstrapBlazor.UniverSheet" Version="9.0.5" /> | ||||
| 		<PackageReference Include="BootstrapBlazor.WinBox" Version="9.0.7" /> | ||||
| 		<PackageReference Include="BootstrapBlazor.CodeEditor" Version="9.0.1" /> | ||||
| 		<ProjectReference Include="..\..\Admin\ThingsGateway.Admin.Razor\ThingsGateway.Admin.Razor.csproj" /> | ||||
|   | ||||
| @@ -18,7 +18,7 @@ namespace ThingsGateway.Management; | ||||
| [ApiDescriptionSettings("ThingsGateway.OpenApi", Order = 200)] | ||||
| [Route("openApi/autoUpdate")] | ||||
| [RolePermission] | ||||
| [LoggingMonitor] | ||||
| [RequestAudit] | ||||
| [ApiController] | ||||
| [Authorize(AuthenticationSchemes = "Bearer")] | ||||
| public class AutoUpdateController : ControllerBase | ||||
|   | ||||
| @@ -16,7 +16,6 @@ namespace ThingsGateway.Management; | ||||
|  | ||||
| public class DeviceDataWithValue | ||||
| { | ||||
|     public long Id { get; set; } | ||||
|     /// <inheritdoc cref="Device.Name"/> | ||||
|     public string Name { get; set; } | ||||
|  | ||||
| @@ -34,3 +33,17 @@ public class DeviceDataWithValue | ||||
|     /// <inheritdoc cref="DeviceRuntime.ReadOnlyVariableRuntimes"/> | ||||
|     public Dictionary<string, VariableDataWithValue> ReadOnlyVariableRuntimes { get; set; } | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| public class DataWithDatabase | ||||
| { | ||||
|     public Channel Channel { get; set; } | ||||
|     public List<DeviceDataWithDatabase> DeviceVariables { get; set; } | ||||
| } | ||||
|  | ||||
| public class DeviceDataWithDatabase | ||||
| { | ||||
|     public Device Device { get; set; } | ||||
|     public List<Variable> Variables { get; set; } | ||||
| } | ||||
| @@ -44,6 +44,7 @@ | ||||
|     "RedundancyDisable": "Redundant gateway site not enabled" | ||||
|   }, | ||||
|   "ThingsGateway.Management.RedundancyOptions": { | ||||
|     "ForcedSync": "Forced Synchronous", | ||||
|     "Enable": "Enable Dual-Machine Redundancy", | ||||
|     "MasterUri": "Master Node URL", | ||||
|     "IsMaster": "IsMaster", | ||||
| @@ -59,8 +60,9 @@ | ||||
|     "Switch": "Switch", | ||||
|  | ||||
|     "Restart": "The redundant service will be restarted soon", | ||||
|     "Confirm": "Confirm switching to redundant state" | ||||
|     "Confirm": "Confirm switching to redundant state", | ||||
|  | ||||
|     "ForcedSyncWarning": "Forcing synchronization will generate database configuration information.Are you sure you want to continue?" | ||||
|  | ||||
|   }, | ||||
|  | ||||
|   | ||||
| @@ -46,7 +46,7 @@ | ||||
|   }, | ||||
|  | ||||
|   "ThingsGateway.Management.RedundancyOptions": { | ||||
|  | ||||
|     "ForcedSync": "强制同步", | ||||
|     "Enable": "启用双机冗余", | ||||
|     "MasterUri": "主站Url", | ||||
|     "IsMaster": "是否为主站", | ||||
| @@ -62,7 +62,8 @@ | ||||
|     "Switch": "切换", | ||||
|  | ||||
|     "Restart": "即将重新启动冗余服务", | ||||
|     "Confirm": "确认切换冗余状态" | ||||
|     "Confirm": "确认切换冗余状态", | ||||
|     "ForcedSyncWarning": "强制同步会生成数据库配置信息,是否继续?" | ||||
|   }, | ||||
|  | ||||
|   "ThingsGateway.Management._Imports": { | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user