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