Compare commits
7 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
dbfc9a5bb4 | ||
|
|
1b758aa41a | ||
|
|
43bdc70899 | ||
|
|
fadda000a6 | ||
|
|
45a8c91a5a | ||
|
|
8e938f18be | ||
|
|
ab1b364c54 |
@@ -20,9 +20,11 @@ namespace ThingsGateway.Admin.Application;
|
|||||||
public class AppService : IAppService
|
public class AppService : IAppService
|
||||||
{
|
{
|
||||||
private readonly IUserAgentService UserAgentService;
|
private readonly IUserAgentService UserAgentService;
|
||||||
public AppService(IUserAgentService userAgentService)
|
private readonly IClaimsPrincipalService ClaimsPrincipalService;
|
||||||
|
public AppService(IUserAgentService userAgentService, IClaimsPrincipalService claimsPrincipalService)
|
||||||
{
|
{
|
||||||
UserAgentService = userAgentService;
|
UserAgentService = userAgentService;
|
||||||
|
ClaimsPrincipalService = claimsPrincipalService;
|
||||||
}
|
}
|
||||||
public string GetReturnUrl(string returnUrl)
|
public string GetReturnUrl(string returnUrl)
|
||||||
{
|
{
|
||||||
@@ -70,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();
|
||||||
|
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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,19 +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<ISugarAopService, SugarAopService>();
|
|
||||||
services.AddSingleton<ISugarConfigAopService, SugarConfigAopService>();
|
|
||||||
|
|
||||||
services.AddSingleton<IUserAgentService, UserAgentService>();
|
services.AddSingleton<IUserAgentService, UserAgentService>();
|
||||||
services.AddSingleton<IAppService, AppService>();
|
services.AddSingleton<IAppService, AppService>();
|
||||||
|
|
||||||
StaticConfig.EnableAllWhereIF = true;
|
|
||||||
|
|
||||||
services.AddConfigurableOptions<EmailOptions>();
|
services.AddConfigurableOptions<EmailOptions>();
|
||||||
services.AddConfigurableOptions<HardwareInfoOptions>();
|
services.AddConfigurableOptions<HardwareInfoOptions>();
|
||||||
|
|
||||||
@@ -57,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>();
|
||||||
|
|||||||
@@ -18,9 +18,7 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="BootstrapBlazor.TableExport" Version="9.2.5" />
|
|
||||||
<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" />
|
||||||
@@ -49,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>
|
||||||
|
|||||||
@@ -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"
|
||||||
|
|||||||
@@ -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",
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,7 +29,6 @@ using ThingsGateway.Admin.Application;
|
|||||||
using ThingsGateway.Admin.Razor;
|
using ThingsGateway.Admin.Razor;
|
||||||
using ThingsGateway.Extension;
|
using ThingsGateway.Extension;
|
||||||
using ThingsGateway.NewLife.Caching;
|
using ThingsGateway.NewLife.Caching;
|
||||||
using ThingsGateway.NewLife.Extension;
|
|
||||||
|
|
||||||
namespace ThingsGateway.AdminServer;
|
namespace ThingsGateway.AdminServer;
|
||||||
|
|
||||||
@@ -161,7 +160,8 @@ public class Startup : AppStartup
|
|||||||
{
|
{
|
||||||
options.WriteFilter = (logMsg) =>
|
options.WriteFilter = (logMsg) =>
|
||||||
{
|
{
|
||||||
if (logMsg.Message.IsNullOrEmpty()) return false;
|
if (App.HostApplicationLifetime.ApplicationStopping.IsCancellationRequested && logMsg.LogLevel >= LogLevel.Warning) return false;
|
||||||
|
if (string.IsNullOrEmpty(logMsg.Message)) return false;
|
||||||
else return true;
|
else return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -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
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -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.4" />
|
<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>
|
||||||
|
|
||||||
|
|||||||
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);
|
||||||
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.29</PluginVersion>
|
<PluginVersion>10.7.0</PluginVersion>
|
||||||
<ProPluginVersion>10.6.29</ProPluginVersion>
|
<ProPluginVersion>10.7.0</ProPluginVersion>
|
||||||
<AuthenticationVersion>2.1.8</AuthenticationVersion>
|
<AuthenticationVersion>2.2.0</AuthenticationVersion>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
|
|||||||
@@ -6,7 +6,9 @@
|
|||||||
@using BootstrapBlazor.Components
|
@using BootstrapBlazor.Components
|
||||||
@namespace ThingsGateway.Debug
|
@namespace ThingsGateway.Debug
|
||||||
|
|
||||||
<Card HeaderText=@HeaderText class=@("w-100") style=@($"{CardStyle}")>
|
<div class="w-100" style=@($"height:{HeightString}")>
|
||||||
|
|
||||||
|
<Card HeaderText=@HeaderText class=@("w-100 h-100")>
|
||||||
<HeaderTemplate>
|
<HeaderTemplate>
|
||||||
<div class="flex-fill">
|
<div class="flex-fill">
|
||||||
</div>
|
</div>
|
||||||
@@ -36,7 +38,7 @@
|
|||||||
|
|
||||||
</HeaderTemplate>
|
</HeaderTemplate>
|
||||||
<BodyTemplate>
|
<BodyTemplate>
|
||||||
<div style=@($"height:{HeightString};overflow-y:scroll")>
|
<div style=@($"height:calc(100% - 50px);overflow-y:scroll;flex-fill;")>
|
||||||
<Virtualize Items="CurrentMessages??new List<LogMessage>()" Context="itemMessage" ItemSize="60" OverscanCount=2>
|
<Virtualize Items="CurrentMessages??new List<LogMessage>()" Context="itemMessage" ItemSize="60" OverscanCount=2>
|
||||||
<ItemContent>
|
<ItemContent>
|
||||||
@* <Tooltip Placement="Placement.Bottom" Title=@itemMessage.Message.Substring(0, Math.Min(itemMessage.Message.Length, 500))> *@
|
@* <Tooltip Placement="Placement.Bottom" Title=@itemMessage.Message.Substring(0, Math.Min(itemMessage.Message.Length, 500))> *@
|
||||||
@@ -56,4 +58,4 @@
|
|||||||
</Card>
|
</Card>
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|||||||
@@ -33,13 +33,11 @@ public partial class LogConsole : IDisposable
|
|||||||
[Parameter]
|
[Parameter]
|
||||||
public EventCallback<LogLevel> LogLevelChanged { get; set; }
|
public EventCallback<LogLevel> LogLevelChanged { get; set; }
|
||||||
|
|
||||||
[Parameter]
|
|
||||||
public string CardStyle { get; set; } = "height: 100%;";
|
|
||||||
[Parameter]
|
[Parameter]
|
||||||
public string HeaderText { get; set; } = "Log";
|
public string HeaderText { get; set; } = "Log";
|
||||||
|
|
||||||
[Parameter]
|
[Parameter]
|
||||||
public string HeightString { get; set; } = "calc(100% - 50px)";
|
public string HeightString { get; set; } = "calc(100% - 300px)";
|
||||||
|
|
||||||
[Parameter, EditorRequired]
|
[Parameter, EditorRequired]
|
||||||
public string LogPath { get; set; }
|
public string LogPath { get; set; }
|
||||||
|
|||||||
@@ -28,11 +28,12 @@ public class DtuPlugin : PluginBase, ITcpReceivingPlugin
|
|||||||
set
|
set
|
||||||
{
|
{
|
||||||
_heartbeat = value;
|
_heartbeat = value;
|
||||||
HeartbeatByte = new ArraySegment<byte>(Encoding.UTF8.GetBytes(value));
|
if (!_heartbeat.IsNullOrEmpty())
|
||||||
|
HeartbeatByte = new ArraySegment<byte>(Encoding.UTF8.GetBytes(value));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
private string _heartbeat;
|
private string _heartbeat;
|
||||||
private ArraySegment<byte> HeartbeatByte;
|
private ArraySegment<byte> HeartbeatByte = new();
|
||||||
|
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
public async Task OnTcpReceiving(ITcpSession client, ByteBlockEventArgs e)
|
public async Task OnTcpReceiving(ITcpSession client, ByteBlockEventArgs e)
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ public static class PluginUtil
|
|||||||
Action<IPluginManager> action = a => { };
|
Action<IPluginManager> action = a => { };
|
||||||
|
|
||||||
action += GetTcpServicePlugin(channelOptions);
|
action += GetTcpServicePlugin(channelOptions);
|
||||||
if (!channelOptions.Heartbeat.IsNullOrWhiteSpace())
|
//if (!channelOptions.Heartbeat.IsNullOrWhiteSpace())
|
||||||
{
|
{
|
||||||
action += a =>
|
action += a =>
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -8,17 +8,13 @@
|
|||||||
// QQ群:605534569
|
// QQ群:605534569
|
||||||
//------------------------------------------------------------------------------
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
using System.Buffers;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
|
||||||
using ThingsGateway.NewLife.Caching;
|
using ThingsGateway.NewLife.Caching;
|
||||||
|
|
||||||
namespace ThingsGateway.Foundation;
|
namespace ThingsGateway.Foundation;
|
||||||
|
|
||||||
public class LogDataCache
|
|
||||||
{
|
|
||||||
public List<LogData> LogDatas { get; set; }
|
|
||||||
public long Length { get; set; }
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 日志数据
|
/// 日志数据
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -47,8 +43,19 @@ public class LogData
|
|||||||
|
|
||||||
|
|
||||||
/// <summary>日志文本文件倒序读取</summary>
|
/// <summary>日志文本文件倒序读取</summary>
|
||||||
|
|
||||||
|
public class LogDataCache
|
||||||
|
{
|
||||||
|
public List<LogData> LogDatas { get; set; }
|
||||||
|
public long Length { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>高性能日志文件读取器(支持倒序读取)</summary>
|
||||||
public class TextFileReader
|
public class TextFileReader
|
||||||
{
|
{
|
||||||
|
private static readonly MemoryCache _cache = new() { Expire = 30 };
|
||||||
|
private static readonly MemoryCache _fileLocks = new();
|
||||||
|
private static readonly ArrayPool<byte> _bytePool = ArrayPool<byte>.Shared;
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 获取指定目录下所有文件信息
|
/// 获取指定目录下所有文件信息
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -86,159 +93,167 @@ public class TextFileReader
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
static MemoryCache _cache = new() { Expire = 30 };
|
|
||||||
public static OperResult<List<LogData>> LastLog(string file, int lineCount = 200)
|
public static OperResult<List<LogData>> LastLog(string file, int lineCount = 200)
|
||||||
{
|
{
|
||||||
lock (_cache)
|
if (!File.Exists(file))
|
||||||
{
|
return new OperResult<List<LogData>>("The file path is invalid");
|
||||||
|
|
||||||
OperResult<List<LogData>> result = new(); // 初始化结果对象
|
_fileLocks.SetExpire(file, TimeSpan.FromSeconds(30));
|
||||||
|
var fileLock = _fileLocks.GetOrAdd(file, _ => new object());
|
||||||
|
lock (fileLock)
|
||||||
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (!File.Exists(file)) // 检查文件是否存在
|
var fileInfo = new FileInfo(file);
|
||||||
|
var length = fileInfo.Length;
|
||||||
|
var cacheKey = $"{nameof(TextFileReader)}_{nameof(LastLog)}_{file})";
|
||||||
|
if (_cache.TryGetValue<LogDataCache>(cacheKey, out var cachedData))
|
||||||
{
|
{
|
||||||
result.OperCode = 999;
|
if (cachedData != null && cachedData.Length == length)
|
||||||
result.ErrorMessage = "The file path is invalid";
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<string> txt = new(); // 存储读取的文本内容
|
|
||||||
long ps = 0; // 保存起始位置
|
|
||||||
var key = $"{nameof(TextFileReader)}_{nameof(LastLog)}_{file})";
|
|
||||||
long length = 0;
|
|
||||||
using (var fs = new FileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
|
|
||||||
{
|
|
||||||
length = fs.Length;
|
|
||||||
var dataCache = _cache.Get<LogDataCache>(key);
|
|
||||||
if (dataCache != null && dataCache.Length == length)
|
|
||||||
{
|
{
|
||||||
result.Content = dataCache.LogDatas;
|
return new OperResult<List<LogData>>() { Content = cachedData.LogDatas };
|
||||||
result.OperCode = 0; // 操作状态设为成功
|
|
||||||
return result; // 返回解析结果
|
|
||||||
}
|
}
|
||||||
|
else
|
||||||
if (ps <= 0) // 如果起始位置小于等于0,将起始位置设置为文件长度
|
|
||||||
ps = length - 1;
|
|
||||||
|
|
||||||
// 循环读取指定行数的文本内容
|
|
||||||
for (int i = 0; i < lineCount; i++)
|
|
||||||
{
|
{
|
||||||
ps = InverseReadRow(fs, ps, out var value); // 使用逆序读取
|
_cache.Remove(cacheKey);
|
||||||
txt.Add(value);
|
|
||||||
if (ps <= 0) // 如果已经读取到文件开头则跳出循环
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 使用单次 LINQ 操作进行过滤和解析
|
using var fs = new FileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite, 4096, FileOptions.SequentialScan);
|
||||||
result.Content = txt
|
var result = ReadLogsInverse(fs, lineCount, fileInfo.Length);
|
||||||
.Select(a => ParseCSV(a))
|
|
||||||
.Where(data => data.Count >= 3)
|
|
||||||
.Select(data =>
|
|
||||||
{
|
|
||||||
var log = new LogData
|
|
||||||
{
|
|
||||||
LogTime = data[0].Trim(),
|
|
||||||
LogLevel = Enum.TryParse(data[1].Trim(), out LogLevel level) ? level : LogLevel.Info,
|
|
||||||
Message = data[2].Trim(),
|
|
||||||
ExceptionString = data.Count > 3 ? data[3].Trim() : null
|
|
||||||
};
|
|
||||||
return log;
|
|
||||||
})
|
|
||||||
.ToList();
|
|
||||||
|
|
||||||
result.OperCode = 0; // 操作状态设为成功
|
_cache.Set(cacheKey, new LogDataCache
|
||||||
var data = _cache.Set<LogDataCache>(key, new LogDataCache() { Length = length, LogDatas = result.Content });
|
{
|
||||||
|
LogDatas = result,
|
||||||
|
Length = fileInfo.Length,
|
||||||
|
});
|
||||||
|
|
||||||
return result; // 返回解析结果
|
return new OperResult<List<LogData>>() { Content = result };
|
||||||
}
|
}
|
||||||
catch (Exception ex) // 捕获异常
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
result = new(ex); // 创建包含异常信息的结果对象
|
return new OperResult<List<LogData>>(ex);
|
||||||
return result; // 返回异常结果
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static List<LogData> ReadLogsInverse(FileStream fs, int lineCount, long length)
|
||||||
|
{
|
||||||
|
length = fs.Length;
|
||||||
|
long ps = 0; // 保存起始位置
|
||||||
|
List<string> txt = new(); // 存储读取的文本内容
|
||||||
|
|
||||||
|
if (ps <= 0) // 如果起始位置小于等于0,将起始位置设置为文件长度
|
||||||
|
ps = length - 1;
|
||||||
|
|
||||||
|
// 循环读取指定行数的文本内容
|
||||||
|
for (int i = 0; i < lineCount; i++)
|
||||||
|
{
|
||||||
|
ps = InverseReadRow(fs, ps, out var value); // 使用逆序读取
|
||||||
|
txt.Add(value);
|
||||||
|
if (ps <= 0) // 如果已经读取到文件开头则跳出循环
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 使用单次 LINQ 操作进行过滤和解析
|
||||||
|
var result = txt
|
||||||
|
.Select(a => ParseCSV(a))
|
||||||
|
.Where(data => data.Count >= 3)
|
||||||
|
.Select(data =>
|
||||||
|
{
|
||||||
|
var log = new LogData
|
||||||
|
{
|
||||||
|
LogTime = data[0].Trim(),
|
||||||
|
LogLevel = Enum.TryParse(data[1].Trim(), out LogLevel level) ? level : LogLevel.Info,
|
||||||
|
Message = data[2].Trim(),
|
||||||
|
ExceptionString = data.Count > 3 ? data[3].Trim() : null
|
||||||
|
};
|
||||||
|
return log;
|
||||||
|
})
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
|
||||||
|
return result; // 返回解析结果
|
||||||
|
}
|
||||||
|
|
||||||
private static long InverseReadRow(FileStream fs, long position, out string value, int maxRead = 102400)
|
private static long InverseReadRow(FileStream fs, long position, out string value, int maxRead = 102400)
|
||||||
{
|
{
|
||||||
byte n = 0xD; // 换行符
|
byte n = 0xD;
|
||||||
byte a = 0xA; // 回车符
|
byte a = 0xA;
|
||||||
value = string.Empty;
|
value = string.Empty;
|
||||||
if (fs.Length == 0) return 0; // 若文件长度为0,则直接返回0作为新的位置
|
|
||||||
|
if (fs.Length == 0) return 0;
|
||||||
|
|
||||||
var newPos = position;
|
var newPos = position;
|
||||||
List<byte> buffer = new List<byte>(maxRead); // 缓存读取的数据
|
byte[] buffer = _bytePool.Rent(maxRead); // 从池中租借字节数组
|
||||||
|
int index = 0;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var readLength = 0;
|
while (true)
|
||||||
|
|
||||||
while (true) // 循环读取一行数据,TextFileLogger.Separator行判定
|
|
||||||
{
|
{
|
||||||
readLength++;
|
|
||||||
if (newPos <= 0)
|
if (newPos <= 0)
|
||||||
newPos = 0;
|
newPos = 0;
|
||||||
|
|
||||||
fs.Position = newPos;
|
fs.Position = newPos;
|
||||||
int byteRead = fs.ReadByte();
|
int byteRead = fs.ReadByte();
|
||||||
|
|
||||||
if (byteRead == -1) break; // 到达文件开头时跳出循环
|
if (byteRead == -1) break;
|
||||||
|
|
||||||
buffer.Add((byte)byteRead);
|
if (index >= maxRead)
|
||||||
|
|
||||||
if (byteRead == n || byteRead == a)//判断当前字符是换行符 // TextFileLogger.Separator
|
|
||||||
{
|
|
||||||
if (MatchSeparator(buffer))
|
|
||||||
{
|
|
||||||
// 去掉匹配的指定字符串
|
|
||||||
buffer.RemoveRange(buffer.Count - TextFileLogger.SeparatorBytes.Length, TextFileLogger.SeparatorBytes.Length);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (buffer.Count > maxRead) // 超过最大字节数限制时丢弃数据
|
|
||||||
{
|
{
|
||||||
newPos = -1;
|
newPos = -1;
|
||||||
return newPos;
|
return newPos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
buffer[index++] = (byte)byteRead;
|
||||||
|
|
||||||
|
if (byteRead == n || byteRead == a)
|
||||||
|
{
|
||||||
|
if (MatchSeparator(buffer, index))
|
||||||
|
{
|
||||||
|
index -= TextFileLogger.SeparatorBytes.Length;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
newPos--;
|
newPos--;
|
||||||
if (newPos <= -1)
|
if (newPos <= -1)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (buffer.Count >= 10)
|
if (index >= 10)
|
||||||
{
|
{
|
||||||
buffer.Reverse();
|
Array.Reverse(buffer, 0, index); // 倒序
|
||||||
value = Encoding.UTF8.GetString(buffer.ToArray()); // 转换为字符串
|
value = Encoding.UTF8.GetString(buffer, 0, index);
|
||||||
}
|
}
|
||||||
|
|
||||||
return newPos; // 返回新的读取位置
|
return newPos;
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
|
_bytePool.Return(buffer); // 归还数组
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static bool MatchSeparator(List<byte> arr)
|
private static bool MatchSeparator(byte[] arr, int length)
|
||||||
{
|
{
|
||||||
if (arr.Count < TextFileLogger.SeparatorBytes.Length)
|
if (length < TextFileLogger.SeparatorBytes.Length)
|
||||||
{
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
var pos = arr.Count - 1;
|
int pos = length - 1;
|
||||||
for (int i = 0; i < TextFileLogger.SeparatorBytes.Length; i++)
|
for (int i = 0; i < TextFileLogger.SeparatorBytes.Length; i++)
|
||||||
{
|
{
|
||||||
if (arr[pos] != TextFileLogger.SeparatorBytes[i])
|
if (arr[pos] != TextFileLogger.SeparatorBytes[i])
|
||||||
{
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
pos--;
|
pos--;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static List<string> ParseCSV(string data)
|
private static List<string> ParseCSV(string data)
|
||||||
{
|
{
|
||||||
List<string> items = new List<string>();
|
List<string> items = new List<string>();
|
||||||
|
|||||||
@@ -0,0 +1,81 @@
|
|||||||
|
//------------------------------------------------------------------------------
|
||||||
|
// 此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
|
||||||
|
// 此代码版权(除特别声明外的代码)归作者本人Diego所有
|
||||||
|
// 源代码使用协议遵循本仓库的开源协议及附加协议
|
||||||
|
// Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
|
||||||
|
// Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
|
||||||
|
// 使用文档:https://thingsgateway.cn/
|
||||||
|
// QQ群:605534569
|
||||||
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
namespace ThingsGateway.Gateway.Application;
|
||||||
|
|
||||||
|
public class SmartTriggerScheduler
|
||||||
|
{
|
||||||
|
private readonly object _lock = new(); // 锁对象,保证线程安全
|
||||||
|
private readonly Func<Task> _action; // 实际要执行的操作
|
||||||
|
private readonly TimeSpan _delay; // 执行间隔(冷却时间)
|
||||||
|
|
||||||
|
private bool _isRunning = false; // 当前是否有调度任务在运行
|
||||||
|
private bool _hasPending = false; // 在等待期间是否有新的触发
|
||||||
|
|
||||||
|
// 构造函数,传入要执行的方法和最小执行间隔
|
||||||
|
public SmartTriggerScheduler(Func<Task> action, TimeSpan minimumInterval)
|
||||||
|
{
|
||||||
|
_action = action ?? throw new ArgumentNullException(nameof(action));
|
||||||
|
_delay = minimumInterval;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 外部调用的触发方法(高频调用的地方调用这个)
|
||||||
|
public void Trigger()
|
||||||
|
{
|
||||||
|
lock (_lock)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (_isRunning)
|
||||||
|
{
|
||||||
|
// 如果正在执行中,标记为“等待处理”,之后再执行一次
|
||||||
|
_hasPending = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 否则启动执行任务
|
||||||
|
_isRunning = true;
|
||||||
|
_ = Task.Run(ExecuteLoop); // 开启异步执行循环(非阻塞)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 实际执行动作的循环逻辑
|
||||||
|
private async Task ExecuteLoop()
|
||||||
|
{
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
Func<Task> actionToRun = null;
|
||||||
|
|
||||||
|
// 拷贝 _action,并清除等待标记
|
||||||
|
lock (_lock)
|
||||||
|
{
|
||||||
|
_hasPending = false; // 当前这一轮已经处理了触发
|
||||||
|
actionToRun = _action; // 拷贝要执行的逻辑(避免锁内执行)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 执行外部提供的方法
|
||||||
|
await actionToRun().ConfigureAwait(false);
|
||||||
|
|
||||||
|
// 等待 delay 时间,进入冷却期
|
||||||
|
await Task.Delay(_delay).ConfigureAwait(false);
|
||||||
|
|
||||||
|
lock (_lock)
|
||||||
|
{
|
||||||
|
// 冷却期后检查是否在这段时间内有新的触发
|
||||||
|
if (!_hasPending)
|
||||||
|
{
|
||||||
|
// 没有新的触发了,结束执行循环
|
||||||
|
_isRunning = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -352,7 +352,12 @@ public abstract class DriverBase : DisposableObject, IDriver
|
|||||||
|
|
||||||
public string GetAuthString()
|
public string GetAuthString()
|
||||||
{
|
{
|
||||||
return PluginServiceUtil.IsEducation(GetType()) ? ThingsGateway.Authentication.ProAuthentication.TryGetAuthorizeInfo(out _) ? Localizer["Authorized"] : Localizer["Unauthorized"] : string.Empty;
|
if (PluginServiceUtil.IsEducation(GetType()))
|
||||||
|
{
|
||||||
|
ThingsGateway.Authentication.ProAuthentication.TryGetAuthorizeInfo(out var authorizeInfo);
|
||||||
|
return authorizeInfo.Auth ? Localizer["Authorized"] : Localizer["Unauthorized"];
|
||||||
|
}
|
||||||
|
return string.Empty;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -8,6 +8,8 @@
|
|||||||
// QQ群:605534569
|
// QQ群:605534569
|
||||||
//------------------------------------------------------------------------------
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
using BootstrapBlazor.Components;
|
||||||
|
|
||||||
using Mapster;
|
using Mapster;
|
||||||
|
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
@@ -205,6 +207,30 @@ public static class GlobalData
|
|||||||
|
|
||||||
#region 单例服务
|
#region 单例服务
|
||||||
|
|
||||||
|
|
||||||
|
private static IDispatchService<ChannelRuntime> channelRuntimeDispatchService;
|
||||||
|
public static IDispatchService<ChannelRuntime> ChannelDeviceRuntimeDispatchService
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if (channelRuntimeDispatchService == null)
|
||||||
|
channelRuntimeDispatchService = App.GetService<IDispatchService<ChannelRuntime>>();
|
||||||
|
|
||||||
|
return channelRuntimeDispatchService;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
private static IDispatchService<VariableRuntime> variableRuntimeDispatchService;
|
||||||
|
public static IDispatchService<VariableRuntime> VariableRuntimeDispatchService
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if (variableRuntimeDispatchService == null)
|
||||||
|
variableRuntimeDispatchService = App.GetService<IDispatchService<VariableRuntime>>();
|
||||||
|
|
||||||
|
return variableRuntimeDispatchService;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private static ISysUserService sysUserService;
|
private static ISysUserService sysUserService;
|
||||||
public static ISysUserService SysUserService
|
public static ISysUserService SysUserService
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,464 +0,0 @@
|
|||||||
//------------------------------------------------------------------------------
|
|
||||||
// 此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
|
|
||||||
// 此代码版权(除特别声明外的代码)归作者本人Diego所有
|
|
||||||
// 源代码使用协议遵循本仓库的开源协议及附加协议
|
|
||||||
// Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
|
|
||||||
// Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
|
|
||||||
// 使用文档:https://thingsgateway.cn/
|
|
||||||
// QQ群:605534569
|
|
||||||
//------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
using System.Collections;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Gateway.Application;
|
|
||||||
|
|
||||||
public class ThreadSafeStringDictionary<T> : IDictionary<string, T>, IReadOnlyDictionary<string, T>
|
|
||||||
{
|
|
||||||
private const int DEFAULT_PARTITIONS = 128;
|
|
||||||
private readonly Dictionary<string, T>[] _partitions;
|
|
||||||
private readonly object[] _partitionLocks;
|
|
||||||
private readonly IEqualityComparer<string> _comparer;
|
|
||||||
|
|
||||||
public ThreadSafeStringDictionary() : this(DEFAULT_PARTITIONS, null) { }
|
|
||||||
|
|
||||||
public ThreadSafeStringDictionary(int partitionCount, IEqualityComparer<string> comparer)
|
|
||||||
{
|
|
||||||
if (partitionCount < 1)
|
|
||||||
throw new ArgumentOutOfRangeException(nameof(partitionCount));
|
|
||||||
|
|
||||||
_partitions = new Dictionary<string, T>[partitionCount];
|
|
||||||
_partitionLocks = new object[partitionCount];
|
|
||||||
_comparer = comparer ?? StringComparer.Ordinal;
|
|
||||||
|
|
||||||
for (int i = 0; i < partitionCount; i++)
|
|
||||||
{
|
|
||||||
_partitions[i] = new Dictionary<string, T>(_comparer);
|
|
||||||
_partitionLocks[i] = new object();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private int GetPartitionIndex(string key)
|
|
||||||
{
|
|
||||||
if (key == null) throw new ArgumentNullException(nameof(key));
|
|
||||||
return Math.Abs(_comparer.GetHashCode(key)) % _partitions.Length;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 基本操作
|
|
||||||
public T this[string key]
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index][key];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
set
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
_partitions[index][key] = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public ICollection<string> Keys => GetAllItems().Select(kv => kv.Key).ToList();
|
|
||||||
public ICollection<T> Values => GetAllItems().Select(kv => kv.Value).ToList();
|
|
||||||
|
|
||||||
public int Count
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
int count = 0;
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
count += _partitions[i].Count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsReadOnly => false;
|
|
||||||
|
|
||||||
IEnumerable<string> IReadOnlyDictionary<string, T>.Keys => Keys;
|
|
||||||
|
|
||||||
IEnumerable<T> IReadOnlyDictionary<string, T>.Values => Values;
|
|
||||||
|
|
||||||
public void Add(string key, T value)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
_partitions[index].Add(key, value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Add(KeyValuePair<string, T> item) => Add(item.Key, item.Value);
|
|
||||||
|
|
||||||
public void Clear()
|
|
||||||
{
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
_partitions[i].Clear();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Contains(KeyValuePair<string, T> item)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(item.Key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].TryGetValue(item.Key, out var value) &&
|
|
||||||
EqualityComparer<T>.Default.Equals(value, item.Value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool ContainsKey(string key)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].ContainsKey(key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Remove(string key)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].Remove(key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Remove(KeyValuePair<string, T> item)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(item.Key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
if (_partitions[index].TryGetValue(item.Key, out var value) &&
|
|
||||||
EqualityComparer<T>.Default.Equals(value, item.Value))
|
|
||||||
{
|
|
||||||
return _partitions[index].Remove(item.Key);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetValue(string key, out T value)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].TryGetValue(key, out value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void CopyTo(KeyValuePair<string, T>[] array, int arrayIndex)
|
|
||||||
{
|
|
||||||
if (array == null) throw new ArgumentNullException(nameof(array));
|
|
||||||
if (arrayIndex < 0) throw new ArgumentOutOfRangeException(nameof(arrayIndex));
|
|
||||||
if (array.Length - arrayIndex < Count) throw new ArgumentException("Target array too small");
|
|
||||||
|
|
||||||
foreach (var item in GetAllItems())
|
|
||||||
{
|
|
||||||
array[arrayIndex++] = item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 枚举器实现
|
|
||||||
public IEnumerator<KeyValuePair<string, T>> GetEnumerator()
|
|
||||||
{
|
|
||||||
return GetAllItems().GetEnumerator();
|
|
||||||
}
|
|
||||||
|
|
||||||
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
|
||||||
|
|
||||||
public void AddRange(IEnumerable<KeyValuePair<string, T>> items)
|
|
||||||
{
|
|
||||||
var grouped = items.GroupBy(item => GetPartitionIndex(item.Key));
|
|
||||||
|
|
||||||
foreach (var group in grouped)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[group.Key])
|
|
||||||
{
|
|
||||||
foreach (var item in group)
|
|
||||||
{
|
|
||||||
_partitions[group.Key][item.Key] = item.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Dictionary<string, T> GetSnapshot()
|
|
||||||
{
|
|
||||||
var snapshot = new Dictionary<string, T>(_comparer);
|
|
||||||
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
foreach (var kvp in _partitions[i])
|
|
||||||
{
|
|
||||||
snapshot[kvp.Key] = kvp.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return snapshot;
|
|
||||||
}
|
|
||||||
|
|
||||||
private IEnumerable<KeyValuePair<string, T>> GetAllItems()
|
|
||||||
{
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
foreach (var item in _partitions[i]) // 直接枚举原字典
|
|
||||||
{
|
|
||||||
yield return item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class ThreadSafeLongDictionary<T> : IDictionary<long, T>, IReadOnlyDictionary<long, T>
|
|
||||||
{
|
|
||||||
private const int DEFAULT_PARTITIONS = 128;
|
|
||||||
private readonly Dictionary<long, T>[] _partitions;
|
|
||||||
private readonly object[] _partitionLocks;
|
|
||||||
|
|
||||||
public ThreadSafeLongDictionary() : this(DEFAULT_PARTITIONS) { }
|
|
||||||
|
|
||||||
public ThreadSafeLongDictionary(int partitionCount)
|
|
||||||
{
|
|
||||||
if (partitionCount < 1)
|
|
||||||
throw new ArgumentOutOfRangeException(nameof(partitionCount));
|
|
||||||
|
|
||||||
_partitions = new Dictionary<long, T>[partitionCount];
|
|
||||||
_partitionLocks = new object[partitionCount];
|
|
||||||
|
|
||||||
for (int i = 0; i < partitionCount; i++)
|
|
||||||
{
|
|
||||||
_partitions[i] = new Dictionary<long, T>();
|
|
||||||
_partitionLocks[i] = new object();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private int GetPartitionIndex(long key)
|
|
||||||
{
|
|
||||||
// 使用混合哈希算法减少碰撞
|
|
||||||
uint hash = (uint)key;
|
|
||||||
hash = ((hash >> 16) ^ hash) * 0x45d9f3b;
|
|
||||||
hash = ((hash >> 16) ^ hash) * 0x45d9f3b;
|
|
||||||
hash = (hash >> 16) ^ hash;
|
|
||||||
return (int)(hash % _partitions.Length);
|
|
||||||
}
|
|
||||||
|
|
||||||
public T this[long key]
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index][key];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
set
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
_partitions[index][key] = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public ICollection<long> Keys => GetAllItems().Select(kv => kv.Key).ToList();
|
|
||||||
public ICollection<T> Values => GetAllItems().Select(kv => kv.Value).ToList();
|
|
||||||
|
|
||||||
public int Count
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
int count = 0;
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
count += _partitions[i].Count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsReadOnly => false;
|
|
||||||
|
|
||||||
IEnumerable<long> IReadOnlyDictionary<long, T>.Keys => Keys;
|
|
||||||
|
|
||||||
IEnumerable<T> IReadOnlyDictionary<long, T>.Values => Values;
|
|
||||||
|
|
||||||
public void Add(long key, T value)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
_partitions[index].Add(key, value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Add(KeyValuePair<long, T> item) => Add(item.Key, item.Value);
|
|
||||||
|
|
||||||
public void Clear()
|
|
||||||
{
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
_partitions[i].Clear();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Contains(KeyValuePair<long, T> item)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(item.Key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].TryGetValue(item.Key, out var value) &&
|
|
||||||
EqualityComparer<T>.Default.Equals(value, item.Value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool ContainsKey(long key)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].ContainsKey(key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Remove(long key)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].Remove(key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool Remove(KeyValuePair<long, T> item)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(item.Key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
if (_partitions[index].TryGetValue(item.Key, out var value) &&
|
|
||||||
EqualityComparer<T>.Default.Equals(value, item.Value))
|
|
||||||
{
|
|
||||||
return _partitions[index].Remove(item.Key);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetValue(long key, out T value)
|
|
||||||
{
|
|
||||||
int index = GetPartitionIndex(key);
|
|
||||||
lock (_partitionLocks[index])
|
|
||||||
{
|
|
||||||
return _partitions[index].TryGetValue(key, out value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void CopyTo(KeyValuePair<long, T>[] array, int arrayIndex)
|
|
||||||
{
|
|
||||||
if (array == null) throw new ArgumentNullException(nameof(array));
|
|
||||||
if (arrayIndex < 0) throw new ArgumentOutOfRangeException(nameof(arrayIndex));
|
|
||||||
if (array.Length - arrayIndex < Count) throw new ArgumentException("Target array too small");
|
|
||||||
|
|
||||||
foreach (var item in GetAllItems())
|
|
||||||
{
|
|
||||||
array[arrayIndex++] = item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public IEnumerator<KeyValuePair<long, T>> GetEnumerator()
|
|
||||||
{
|
|
||||||
return GetAllItems().GetEnumerator();
|
|
||||||
}
|
|
||||||
|
|
||||||
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
|
||||||
|
|
||||||
public void AddRange(IEnumerable<KeyValuePair<long, T>> items)
|
|
||||||
{
|
|
||||||
var grouped = items.GroupBy(item => GetPartitionIndex(item.Key));
|
|
||||||
|
|
||||||
foreach (var group in grouped)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[group.Key])
|
|
||||||
{
|
|
||||||
foreach (var item in group)
|
|
||||||
{
|
|
||||||
_partitions[group.Key][item.Key] = item.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Dictionary<long, T> GetSnapshot()
|
|
||||||
{
|
|
||||||
var snapshot = new Dictionary<long, T>();
|
|
||||||
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
foreach (var kvp in _partitions[i])
|
|
||||||
{
|
|
||||||
snapshot[kvp.Key] = kvp.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return snapshot;
|
|
||||||
}
|
|
||||||
|
|
||||||
private IEnumerable<KeyValuePair<long, T>> GetAllItems()
|
|
||||||
{
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
foreach (var item in _partitions[i])
|
|
||||||
{
|
|
||||||
yield return item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public string GetPartitionStats()
|
|
||||||
{
|
|
||||||
var stats = new System.Text.StringBuilder();
|
|
||||||
for (int i = 0; i < _partitions.Length; i++)
|
|
||||||
{
|
|
||||||
lock (_partitionLocks[i])
|
|
||||||
{
|
|
||||||
stats.AppendLine($"Partition {i}: {_partitions[i].Count} items");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return stats.ToString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -241,7 +241,7 @@ public class DeviceRuntime : Device, IDisposable
|
|||||||
|
|
||||||
ChannelRuntime = channelRuntime;
|
ChannelRuntime = channelRuntime;
|
||||||
ChannelRuntime?.DeviceRuntimes?.TryRemove(Id, out _);
|
ChannelRuntime?.DeviceRuntimes?.TryRemove(Id, out _);
|
||||||
ChannelRuntime.DeviceRuntimes.TryAdd(Id, this);
|
ChannelRuntime?.DeviceRuntimes?.TryAdd(Id, this);
|
||||||
|
|
||||||
GlobalData.IdDevices.TryRemove(Id, out _);
|
GlobalData.IdDevices.TryRemove(Id, out _);
|
||||||
GlobalData.IdDevices.TryAdd(Id, this);
|
GlobalData.IdDevices.TryAdd(Id, this);
|
||||||
|
|||||||
@@ -17,8 +17,6 @@ using Microsoft.Extensions.Logging;
|
|||||||
|
|
||||||
using ThingsGateway.NewLife;
|
using ThingsGateway.NewLife;
|
||||||
|
|
||||||
using TouchSocket.Core;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Gateway.Application;
|
namespace ThingsGateway.Gateway.Application;
|
||||||
|
|
||||||
public class ChannelRuntimeService : IChannelRuntimeService
|
public class ChannelRuntimeService : IChannelRuntimeService
|
||||||
@@ -199,9 +197,7 @@ public class ChannelRuntimeService : IChannelRuntimeService
|
|||||||
|
|
||||||
public async Task RestartChannelAsync(IEnumerable<ChannelRuntime> oldChannelRuntimes)
|
public async Task RestartChannelAsync(IEnumerable<ChannelRuntime> oldChannelRuntimes)
|
||||||
{
|
{
|
||||||
oldChannelRuntimes.SelectMany(a => a.DeviceRuntimes.SelectMany(a => a.Value.VariableRuntimes)).ParallelForEach(a => a.Value.SafeDispose());
|
RuntimeServiceHelper.RemoveOldChannelRuntimes(oldChannelRuntimes);
|
||||||
oldChannelRuntimes.SelectMany(a => a.DeviceRuntimes).ParallelForEach(a => a.Value.SafeDispose());
|
|
||||||
oldChannelRuntimes.ParallelForEach(a => a.SafeDispose());
|
|
||||||
var ids = oldChannelRuntimes.Select(a => a.Id).ToHashSet();
|
var ids = oldChannelRuntimes.Select(a => a.Id).ToHashSet();
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -229,4 +225,5 @@ public class ChannelRuntimeService : IChannelRuntimeService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -33,16 +33,6 @@ namespace ThingsGateway.Gateway.Application;
|
|||||||
|
|
||||||
internal sealed class ChannelService : BaseService<Channel>, IChannelService
|
internal sealed class ChannelService : BaseService<Channel>, IChannelService
|
||||||
{
|
{
|
||||||
private readonly IDispatchService<Channel> _dispatchService;
|
|
||||||
|
|
||||||
/// <inheritdoc cref="IChannelService"/>
|
|
||||||
public ChannelService(
|
|
||||||
IDispatchService<Channel>? dispatchService
|
|
||||||
)
|
|
||||||
{
|
|
||||||
_dispatchService = dispatchService;
|
|
||||||
}
|
|
||||||
|
|
||||||
#region CURD
|
#region CURD
|
||||||
|
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
@@ -181,7 +171,6 @@ internal sealed class ChannelService : BaseService<Channel>, IChannelService
|
|||||||
public void DeleteChannelFromCache()
|
public void DeleteChannelFromCache()
|
||||||
{
|
{
|
||||||
App.CacheService.Remove(ThingsGatewayCacheConst.Cache_Channel);//删除通道缓存
|
App.CacheService.Remove(ThingsGatewayCacheConst.Cache_Channel);//删除通道缓存
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -35,15 +35,11 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService
|
|||||||
{
|
{
|
||||||
private readonly IChannelService _channelService;
|
private readonly IChannelService _channelService;
|
||||||
private readonly IPluginService _pluginService;
|
private readonly IPluginService _pluginService;
|
||||||
private readonly IDispatchService<Device> _dispatchService;
|
|
||||||
|
|
||||||
public DeviceService(
|
public DeviceService()
|
||||||
IDispatchService<Device> dispatchService
|
|
||||||
)
|
|
||||||
{
|
{
|
||||||
_channelService = App.RootServices.GetRequiredService<IChannelService>();
|
_channelService = App.RootServices.GetRequiredService<IChannelService>();
|
||||||
_pluginService = App.RootServices.GetRequiredService<IPluginService>();
|
_pluginService = App.RootServices.GetRequiredService<IPluginService>();
|
||||||
_dispatchService = dispatchService;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -195,7 +191,6 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService
|
|||||||
public void DeleteDeviceFromCache()
|
public void DeleteDeviceFromCache()
|
||||||
{
|
{
|
||||||
App.CacheService.Remove(ThingsGatewayCacheConst.Cache_Device);//删除设备缓存
|
App.CacheService.Remove(ThingsGatewayCacheConst.Cache_Device);//删除设备缓存
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -8,8 +8,6 @@
|
|||||||
// QQ群:605534569
|
// QQ群:605534569
|
||||||
//------------------------------------------------------------------------------
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
using BootstrapBlazor.Components;
|
|
||||||
|
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
@@ -21,17 +19,6 @@ namespace ThingsGateway.Gateway.Application;
|
|||||||
internal sealed class ChannelThreadManage : IChannelThreadManage
|
internal sealed class ChannelThreadManage : IChannelThreadManage
|
||||||
{
|
{
|
||||||
private ILogger _logger;
|
private ILogger _logger;
|
||||||
private IDispatchService<ChannelRuntime> channelRuntimeDispatchService;
|
|
||||||
private IDispatchService<ChannelRuntime> ChannelRuntimeDispatchService
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
if (channelRuntimeDispatchService == null)
|
|
||||||
channelRuntimeDispatchService = App.GetService<IDispatchService<ChannelRuntime>>();
|
|
||||||
|
|
||||||
return channelRuntimeDispatchService;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public ChannelThreadManage()
|
public ChannelThreadManage()
|
||||||
{
|
{
|
||||||
@@ -78,7 +65,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
|||||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||||
|
|
||||||
await PrivateRemoveChannelsAsync(Enumerable.Repeat(channelId, 1)).ConfigureAwait(false);
|
await PrivateRemoveChannelsAsync(Enumerable.Repeat(channelId, 1)).ConfigureAwait(false);
|
||||||
ChannelRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -98,7 +84,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
|||||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||||
|
|
||||||
await PrivateRemoveChannelsAsync(channelIds).ConfigureAwait(false);
|
await PrivateRemoveChannelsAsync(channelIds).ConfigureAwait(false);
|
||||||
ChannelRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -165,7 +150,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
|||||||
{
|
{
|
||||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||||
await PrivateRestartChannelAsync(Enumerable.Repeat(channelRuntime, 1)).ConfigureAwait(false);
|
await PrivateRestartChannelAsync(Enumerable.Repeat(channelRuntime, 1)).ConfigureAwait(false);
|
||||||
ChannelRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -183,7 +167,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
|||||||
{
|
{
|
||||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||||
await PrivateRestartChannelAsync(channelRuntimes).ConfigureAwait(false);
|
await PrivateRestartChannelAsync(channelRuntimes).ConfigureAwait(false);
|
||||||
ChannelRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -38,17 +38,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public static volatile int CycleInterval = ManageHelper.ChannelThreadOptions.MaxCycleInterval;
|
public static volatile int CycleInterval = ManageHelper.ChannelThreadOptions.MaxCycleInterval;
|
||||||
|
|
||||||
private IDispatchService<DeviceRuntime> devicelRuntimeDispatchService;
|
|
||||||
private IDispatchService<DeviceRuntime> DeviceRuntimeDispatchService
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
if (devicelRuntimeDispatchService == null)
|
|
||||||
devicelRuntimeDispatchService = App.GetService<IDispatchService<DeviceRuntime>>();
|
|
||||||
|
|
||||||
return devicelRuntimeDispatchService;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
static DeviceThreadManage()
|
static DeviceThreadManage()
|
||||||
{
|
{
|
||||||
Task.Factory.StartNew(async () => await SetCycleInterval().ConfigureAwait(false), TaskCreationOptions.LongRunning);
|
Task.Factory.StartNew(async () => await SetCycleInterval().ConfigureAwait(false), TaskCreationOptions.LongRunning);
|
||||||
@@ -250,7 +239,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
|||||||
{
|
{
|
||||||
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
||||||
await PrivateRestartDeviceAsync(Enumerable.Repeat(deviceRuntime, 1), deleteCache).ConfigureAwait(false);
|
await PrivateRestartDeviceAsync(Enumerable.Repeat(deviceRuntime, 1), deleteCache).ConfigureAwait(false);
|
||||||
DeviceRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -268,7 +256,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
|||||||
{
|
{
|
||||||
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
||||||
await PrivateRestartDeviceAsync(deviceRuntimes, deleteCache).ConfigureAwait(false);
|
await PrivateRestartDeviceAsync(deviceRuntimes, deleteCache).ConfigureAwait(false);
|
||||||
DeviceRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -440,7 +427,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
|||||||
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
||||||
|
|
||||||
await PrivateRemoveDevicesAsync(Enumerable.Repeat(deviceId, 1)).ConfigureAwait(false);
|
await PrivateRemoveDevicesAsync(Enumerable.Repeat(deviceId, 1)).ConfigureAwait(false);
|
||||||
DeviceRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -460,7 +446,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
|||||||
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
||||||
|
|
||||||
await PrivateRemoveDevicesAsync(deviceIds).ConfigureAwait(false);
|
await PrivateRemoveDevicesAsync(deviceIds).ConfigureAwait(false);
|
||||||
DeviceRuntimeDispatchService.Dispatch(null);
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -662,6 +647,7 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
|||||||
//传入变量
|
//传入变量
|
||||||
//newDeviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.SafeDispose());
|
//newDeviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.SafeDispose());
|
||||||
deviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.Init(newDeviceRuntime));
|
deviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.Init(newDeviceRuntime));
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
@@ -740,6 +726,8 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
|||||||
LogMessage?.LogWarning($"device {newDeviceRuntime.Name} cannot found channel with id{newDeviceRuntime.ChannelId}");
|
LogMessage?.LogWarning($"device {newDeviceRuntime.Name} cannot found channel with id{newDeviceRuntime.ChannelId}");
|
||||||
|
|
||||||
newDeviceRuntime.Init(channelRuntime);
|
newDeviceRuntime.Init(channelRuntime);
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
await channelRuntime.DeviceThreadManage.RestartDeviceAsync(newDeviceRuntime, false).ConfigureAwait(false);
|
await channelRuntime.DeviceThreadManage.RestartDeviceAsync(newDeviceRuntime, false).ConfigureAwait(false);
|
||||||
channelRuntime.DeviceThreadManage.LogMessage?.LogInformation($"Device {newDeviceRuntime.Name} switched to primary channel");
|
channelRuntime.DeviceThreadManage.LogMessage?.LogInformation($"Device {newDeviceRuntime.Name} switched to primary channel");
|
||||||
|
|
||||||
|
|||||||
@@ -71,6 +71,9 @@ internal sealed class GatewayMonitorHostedService : BackgroundService, IGatewayM
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
await ChannelThreadManage.RestartChannelAsync(channelRuntimes).ConfigureAwait(false);
|
await ChannelThreadManage.RestartChannelAsync(channelRuntimes).ConfigureAwait(false);
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -25,8 +25,6 @@ using ThingsGateway.NewLife;
|
|||||||
|
|
||||||
using TouchSocket.Core;
|
using TouchSocket.Core;
|
||||||
|
|
||||||
using Yitter.IdGenerator;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Gateway.Application;
|
namespace ThingsGateway.Gateway.Application;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -598,7 +596,7 @@ internal sealed class PluginService : IPluginService
|
|||||||
}
|
}
|
||||||
_ = Task.Run(() =>
|
_ = Task.Run(() =>
|
||||||
{
|
{
|
||||||
_dispatchService.Dispatch(new());
|
_dispatchService.Dispatch(null);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,6 +51,9 @@ internal static class RuntimeServiceHelper
|
|||||||
logger.LogWarning(ex, "Init Channel");
|
logger.LogWarning(ex, "Init Channel");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void Init(List<ChannelRuntime> newChannelRuntimes)
|
public static void Init(List<ChannelRuntime> newChannelRuntimes)
|
||||||
@@ -72,6 +75,7 @@ internal static class RuntimeServiceHelper
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -104,6 +108,10 @@ internal static class RuntimeServiceHelper
|
|||||||
logger.LogWarning(ex, "Init Device");
|
logger.LogWarning(ex, "Init Device");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void Init(List<DeviceRuntime> newDeviceRuntimes)
|
public static void Init(List<DeviceRuntime> newDeviceRuntimes)
|
||||||
@@ -124,6 +132,10 @@ internal static class RuntimeServiceHelper
|
|||||||
deviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.Init(newDeviceRuntime));
|
deviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.Init(newDeviceRuntime));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
}
|
}
|
||||||
public static void Init(List<VariableRuntime> newVariableRuntimes)
|
public static void Init(List<VariableRuntime> newVariableRuntimes)
|
||||||
{
|
{
|
||||||
@@ -138,10 +150,20 @@ internal static class RuntimeServiceHelper
|
|||||||
newVariableRuntime.Init(deviceRuntime);
|
newVariableRuntime.Init(deviceRuntime);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static void RemoveOldChannelRuntimes(IEnumerable<ChannelRuntime> oldChannelRuntimes)
|
||||||
|
{
|
||||||
|
oldChannelRuntimes.SelectMany(a => a.DeviceRuntimes.SelectMany(a => a.Value.VariableRuntimes)).ParallelForEach(a => a.Value.Dispose());
|
||||||
|
oldChannelRuntimes.SelectMany(a => a.DeviceRuntimes).ParallelForEach(a => a.Value.Dispose());
|
||||||
|
oldChannelRuntimes.ParallelForEach(a => a.Dispose());
|
||||||
|
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
public static async Task<List<ChannelRuntime>> GetNewChannelRuntimesAsync(HashSet<long> ids)
|
public static async Task<List<ChannelRuntime>> GetNewChannelRuntimesAsync(HashSet<long> ids)
|
||||||
{
|
{
|
||||||
@@ -179,6 +201,8 @@ internal static class RuntimeServiceHelper
|
|||||||
});
|
});
|
||||||
deviceRuntime.Dispose();
|
deviceRuntime.Dispose();
|
||||||
}
|
}
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
return changedDriver;
|
return changedDriver;
|
||||||
}
|
}
|
||||||
@@ -222,6 +246,10 @@ internal static class RuntimeServiceHelper
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
|
|
||||||
return changedDriver;
|
return changedDriver;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -332,6 +360,7 @@ internal static class RuntimeServiceHelper
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -350,6 +379,7 @@ internal static class RuntimeServiceHelper
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -202,7 +202,6 @@ public class VariableRuntimeService : IVariableRuntimeService
|
|||||||
await RuntimeServiceHelper.ChangedDriverAsync(_logger, cancellationToken).ConfigureAwait(false);
|
await RuntimeServiceHelper.ChangedDriverAsync(_logger, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
App.GetService<IDispatchService<DeviceRuntime>>().Dispatch(null);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
|
|||||||
@@ -36,20 +36,13 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
private readonly IChannelService _channelService;
|
private readonly IChannelService _channelService;
|
||||||
private readonly IDeviceService _deviceService;
|
private readonly IDeviceService _deviceService;
|
||||||
private readonly IPluginService _pluginService;
|
private readonly IPluginService _pluginService;
|
||||||
private readonly IDispatchService<bool> _allDispatchService;
|
|
||||||
private readonly IDispatchService<Variable> _dispatchService;
|
|
||||||
|
|
||||||
/// <inheritdoc cref="IVariableService"/>
|
/// <inheritdoc cref="IVariableService"/>
|
||||||
public VariableService(
|
public VariableService()
|
||||||
IDispatchService<Variable> dispatchService,
|
|
||||||
IDispatchService<bool> allDispatchService
|
|
||||||
)
|
|
||||||
{
|
{
|
||||||
_channelService = App.RootServices.GetRequiredService<IChannelService>();
|
_channelService = App.RootServices.GetRequiredService<IChannelService>();
|
||||||
_pluginService = App.RootServices.GetRequiredService<IPluginService>();
|
_pluginService = App.RootServices.GetRequiredService<IPluginService>();
|
||||||
_deviceService = App.RootServices.GetRequiredService<IDeviceService>();
|
_deviceService = App.RootServices.GetRequiredService<IDeviceService>();
|
||||||
_dispatchService = dispatchService;
|
|
||||||
_allDispatchService = allDispatchService;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#region 测试
|
#region 测试
|
||||||
@@ -230,7 +223,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
{
|
{
|
||||||
_channelService.DeleteChannelFromCache();//刷新缓存
|
_channelService.DeleteChannelFromCache();//刷新缓存
|
||||||
_deviceService.DeleteDeviceFromCache();
|
_deviceService.DeleteDeviceFromCache();
|
||||||
_allDispatchService.Dispatch(new());
|
|
||||||
DeleteVariableCache();
|
DeleteVariableCache();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -297,7 +289,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -318,7 +309,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
.ToList();
|
.ToList();
|
||||||
|
|
||||||
var result = (await db.Updateable(data).UpdateColumns(differences.Select(a => a.Key).ToArray()).ExecuteCommandAsync().ConfigureAwait(false)) > 0;
|
var result = (await db.Updateable(data).UpdateColumns(differences.Select(a => a.Key).ToArray()).ExecuteCommandAsync().ConfigureAwait(false)) > 0;
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
if (result)
|
if (result)
|
||||||
DeleteVariableCache();
|
DeleteVariableCache();
|
||||||
return result;
|
return result;
|
||||||
@@ -341,7 +331,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
|
|
||||||
if (result > 0)
|
if (result > 0)
|
||||||
DeleteVariableCache();
|
DeleteVariableCache();
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[OperDesc("DeleteVariable", isRecordPar: false, localizerType: typeof(Variable))]
|
[OperDesc("DeleteVariable", isRecordPar: false, localizerType: typeof(Variable))]
|
||||||
@@ -354,7 +343,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
.WhereIF(dataScope != null && dataScope?.Count > 0, u => dataScope.Contains(u.CreateOrgId))//在指定机构列表查询
|
.WhereIF(dataScope != null && dataScope?.Count > 0, u => dataScope.Contains(u.CreateOrgId))//在指定机构列表查询
|
||||||
.WhereIF(dataScope?.Count == 0, u => u.CreateUserId == UserManager.UserId)
|
.WhereIF(dataScope?.Count == 0, u => u.CreateUserId == UserManager.UserId)
|
||||||
.ExecuteCommandAsync().ConfigureAwait(false)) > 0;
|
.ExecuteCommandAsync().ConfigureAwait(false)) > 0;
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
|
|
||||||
if (result)
|
if (result)
|
||||||
DeleteVariableCache();
|
DeleteVariableCache();
|
||||||
@@ -428,7 +416,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
|
|
||||||
if (await base.SaveAsync(input, type).ConfigureAwait(false))
|
if (await base.SaveAsync(input, type).ConfigureAwait(false))
|
||||||
{
|
{
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
DeleteVariableCache();
|
DeleteVariableCache();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -493,7 +480,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
|||||||
using var db = GetDB();
|
using var db = GetDB();
|
||||||
await db.BulkCopyAsync(insertData, 100000).ConfigureAwait(false);
|
await db.BulkCopyAsync(insertData, 100000).ConfigureAwait(false);
|
||||||
await db.BulkUpdateAsync(upData, 100000).ConfigureAwait(false);
|
await db.BulkUpdateAsync(upData, 100000).ConfigureAwait(false);
|
||||||
_dispatchService.Dispatch(new());
|
|
||||||
DeleteVariableCache();
|
DeleteVariableCache();
|
||||||
return variables.Select(a => a.Id).ToHashSet();
|
return variables.Select(a => a.Id).ToHashSet();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,8 +10,6 @@
|
|||||||
|
|
||||||
using Microsoft.Extensions.Options;
|
using Microsoft.Extensions.Options;
|
||||||
|
|
||||||
using ThingsGateway.Authentication;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Gateway.Razor;
|
namespace ThingsGateway.Gateway.Razor;
|
||||||
|
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
@@ -27,39 +25,5 @@ public partial class GatewayAbout
|
|||||||
[NotNull]
|
[NotNull]
|
||||||
private IOptions<WebsiteOptions>? WebsiteOption { get; set; }
|
private IOptions<WebsiteOptions>? WebsiteOption { get; set; }
|
||||||
|
|
||||||
private string Password { get; set; }
|
|
||||||
private AuthorizeInfo AuthorizeInfo { get; set; }
|
|
||||||
[Inject]
|
|
||||||
ToastService ToastService { get; set; }
|
|
||||||
|
|
||||||
protected override void OnParametersSet()
|
|
||||||
{
|
|
||||||
ProAuthentication.TryGetAuthorizeInfo(out var authorizeInfo);
|
|
||||||
AuthorizeInfo = authorizeInfo;
|
|
||||||
base.OnParametersSet();
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task Register()
|
|
||||||
{
|
|
||||||
var result = ProAuthentication.TryAuthorize(Password, out var authorizeInfo);
|
|
||||||
if (result)
|
|
||||||
{
|
|
||||||
AuthorizeInfo = authorizeInfo;
|
|
||||||
await ToastService.Default();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
await ToastService.Default(false);
|
|
||||||
|
|
||||||
Password = string.Empty;
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
}
|
|
||||||
private async Task Unregister()
|
|
||||||
{
|
|
||||||
ProAuthentication.UnAuthorize();
|
|
||||||
var result = ProAuthentication.TryGetAuthorizeInfo(out var authorizeInfo);
|
|
||||||
AuthorizeInfo = authorizeInfo;
|
|
||||||
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,8 +13,6 @@ using Mapster;
|
|||||||
using ThingsGateway.Admin.Application;
|
using ThingsGateway.Admin.Application;
|
||||||
using ThingsGateway.Gateway.Application;
|
using ThingsGateway.Gateway.Application;
|
||||||
|
|
||||||
using Yitter.IdGenerator;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Gateway.Razor;
|
namespace ThingsGateway.Gateway.Razor;
|
||||||
|
|
||||||
public partial class ChannelCopyComponent
|
public partial class ChannelCopyComponent
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
|
|
||||||
<ChannelRuntimeInfo1 ChannelRuntime="ChannelRuntime" />
|
<ChannelRuntimeInfo1 ChannelRuntime="ChannelRuntime" />
|
||||||
|
|
||||||
<LogConsole CardStyle="height: calc(100% - 330px);" LogLevel=@((ChannelRuntime?.DeviceThreadManage?.LogMessage)?.LogLevel??TouchSocket.Core.LogLevel.Trace)
|
<LogConsole HeightString="calc(100% - 270px)" LogLevel=@((ChannelRuntime?.DeviceThreadManage?.LogMessage)?.LogLevel ?? TouchSocket.Core.LogLevel.Trace)
|
||||||
LogLevelChanged="(logLevel)=>
|
LogLevelChanged="(logLevel)=>
|
||||||
{
|
{
|
||||||
ChannelRuntime.DeviceThreadManage?.SetLogAsync(logLevel);
|
ChannelRuntime.DeviceThreadManage?.SetLogAsync(logLevel);
|
||||||
|
|||||||
@@ -43,10 +43,7 @@ public partial class ChannelRuntimeInfo1 : IDisposable
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(StateHasChanged);
|
||||||
{
|
|
||||||
StateHasChanged();
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ public partial class ChannelTable : IDisposable
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (table != null)
|
if (table != null)
|
||||||
await table.QueryAsync();
|
await InvokeAsync(table.QueryAsync);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -59,7 +59,7 @@ public partial class ChannelTable : IDisposable
|
|||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
await Task.Delay(1000);
|
await Task.Delay(5000);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -115,7 +115,7 @@ public partial class ChannelTable : IDisposable
|
|||||||
{
|
{
|
||||||
|
|
||||||
await Task.Run(() =>GlobalData.ChannelRuntimeService.CopyAsync(channels,devices,AutoRestartThread, default));
|
await Task.Run(() =>GlobalData.ChannelRuntimeService.CopyAsync(channels,devices,AutoRestartThread, default));
|
||||||
await table.QueryAsync();
|
await InvokeAsync(table.QueryAsync);
|
||||||
|
|
||||||
}},
|
}},
|
||||||
{nameof(ChannelCopyComponent.Model),oneModel },
|
{nameof(ChannelCopyComponent.Model),oneModel },
|
||||||
|
|||||||
@@ -17,7 +17,6 @@ using SqlSugar;
|
|||||||
|
|
||||||
using ThingsGateway.Admin.Razor;
|
using ThingsGateway.Admin.Razor;
|
||||||
using ThingsGateway.Gateway.Application;
|
using ThingsGateway.Gateway.Application;
|
||||||
using ThingsGateway.NewLife;
|
|
||||||
using ThingsGateway.NewLife.Extension;
|
using ThingsGateway.NewLife.Extension;
|
||||||
using ThingsGateway.NewLife.Json.Extension;
|
using ThingsGateway.NewLife.Json.Extension;
|
||||||
|
|
||||||
@@ -127,7 +126,7 @@ public partial class ChannelDeviceTree
|
|||||||
{nameof(ChannelEditComponent.OnValidSubmit), async () =>
|
{nameof(ChannelEditComponent.OnValidSubmit), async () =>
|
||||||
{
|
{
|
||||||
await Task.Run(() =>GlobalData.ChannelRuntimeService.SaveChannelAsync(oneModel,itemChangedType,AutoRestartThread));
|
await Task.Run(() =>GlobalData.ChannelRuntimeService.SaveChannelAsync(oneModel,itemChangedType,AutoRestartThread));
|
||||||
await Notify();
|
////await Notify();
|
||||||
}},
|
}},
|
||||||
{nameof(ChannelEditComponent.Model),oneModel },
|
{nameof(ChannelEditComponent.Model),oneModel },
|
||||||
{nameof(ChannelEditComponent.ValidateEnable),true },
|
{nameof(ChannelEditComponent.ValidateEnable),true },
|
||||||
@@ -175,7 +174,7 @@ public partial class ChannelDeviceTree
|
|||||||
{
|
{
|
||||||
|
|
||||||
await Task.Run(() =>GlobalData.ChannelRuntimeService.CopyAsync(channels,devices,AutoRestartThread, default));
|
await Task.Run(() =>GlobalData.ChannelRuntimeService.CopyAsync(channels,devices,AutoRestartThread, default));
|
||||||
await Notify();
|
//await Notify();
|
||||||
|
|
||||||
}},
|
}},
|
||||||
{nameof(ChannelCopyComponent.Model),oneModel },
|
{nameof(ChannelCopyComponent.Model),oneModel },
|
||||||
@@ -252,7 +251,7 @@ public partial class ChannelDeviceTree
|
|||||||
Spinner.SetRun(true);
|
Spinner.SetRun(true);
|
||||||
await Task.Run(() => GlobalData.ChannelRuntimeService.BatchEditAsync(changedModels, oldModel, oneModel,AutoRestartThread));
|
await Task.Run(() => GlobalData.ChannelRuntimeService.BatchEditAsync(changedModels, oldModel, oneModel,AutoRestartThread));
|
||||||
|
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
|
|
||||||
@@ -304,7 +303,7 @@ public partial class ChannelDeviceTree
|
|||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync( ()=>
|
await InvokeAsync( ()=>
|
||||||
{
|
{
|
||||||
|
|
||||||
@@ -422,7 +421,7 @@ finally
|
|||||||
Spinner.SetRun(true);
|
Spinner.SetRun(true);
|
||||||
|
|
||||||
await Task.Run(() => GlobalData.ChannelRuntimeService.DeleteChannelAsync(modelIds.Select(a => a.Id), AutoRestartThread, default));
|
await Task.Run(() => GlobalData.ChannelRuntimeService.DeleteChannelAsync(modelIds.Select(a => a.Id), AutoRestartThread, default));
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
Spinner.SetRun(false);
|
Spinner.SetRun(false);
|
||||||
@@ -466,7 +465,7 @@ finally
|
|||||||
|
|
||||||
var key = await GlobalData.GetCurrentUserChannels().ConfigureAwait(false);
|
var key = await GlobalData.GetCurrentUserChannels().ConfigureAwait(false);
|
||||||
await Task.Run(() => GlobalData.ChannelRuntimeService.DeleteChannelAsync(key.Select(a => a.Id), AutoRestartThread, default));
|
await Task.Run(() => GlobalData.ChannelRuntimeService.DeleteChannelAsync(key.Select(a => a.Id), AutoRestartThread, default));
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
Spinner.SetRun(false);
|
Spinner.SetRun(false);
|
||||||
@@ -592,7 +591,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
|
|
||||||
});
|
});
|
||||||
await Task.Run(() => GlobalData.ChannelRuntimeService.ImportChannelAsync(value, AutoRestartThread));
|
await Task.Run(() => GlobalData.ChannelRuntimeService.ImportChannelAsync(value, AutoRestartThread));
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
Spinner.SetRun(false);
|
Spinner.SetRun(false);
|
||||||
@@ -647,7 +646,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
{
|
{
|
||||||
|
|
||||||
await Task.Run(() =>GlobalData.DeviceRuntimeService.CopyAsync(devices,AutoRestartThread, default));
|
await Task.Run(() =>GlobalData.DeviceRuntimeService.CopyAsync(devices,AutoRestartThread, default));
|
||||||
await Notify();
|
//await Notify();
|
||||||
|
|
||||||
}},
|
}},
|
||||||
{nameof(DeviceCopyComponent.Model),oneModel },
|
{nameof(DeviceCopyComponent.Model),oneModel },
|
||||||
@@ -697,7 +696,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
{nameof(DeviceEditComponent.OnValidSubmit), async () =>
|
{nameof(DeviceEditComponent.OnValidSubmit), async () =>
|
||||||
{
|
{
|
||||||
await Task.Run(() =>GlobalData.DeviceRuntimeService.SaveDeviceAsync(oneModel,itemChangedType, AutoRestartThread));
|
await Task.Run(() =>GlobalData.DeviceRuntimeService.SaveDeviceAsync(oneModel,itemChangedType, AutoRestartThread));
|
||||||
await Notify();
|
//await Notify();
|
||||||
}},
|
}},
|
||||||
{nameof(DeviceEditComponent.Model),oneModel },
|
{nameof(DeviceEditComponent.Model),oneModel },
|
||||||
{nameof(DeviceEditComponent.AutoRestartThread),AutoRestartThread },
|
{nameof(DeviceEditComponent.AutoRestartThread),AutoRestartThread },
|
||||||
@@ -783,7 +782,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
|
|
||||||
});
|
});
|
||||||
await Task.Run(() =>GlobalData.DeviceRuntimeService.BatchEditAsync(changedModels,oldModel,oneModel,AutoRestartThread));
|
await Task.Run(() =>GlobalData.DeviceRuntimeService.BatchEditAsync(changedModels,oldModel,oneModel,AutoRestartThread));
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
Spinner.SetRun(false);
|
Spinner.SetRun(false);
|
||||||
@@ -832,7 +831,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync( ()=>
|
await InvokeAsync( ()=>
|
||||||
{
|
{
|
||||||
|
|
||||||
@@ -955,7 +954,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
Spinner.SetRun(true);
|
Spinner.SetRun(true);
|
||||||
|
|
||||||
await Task.Run(() => GlobalData.DeviceRuntimeService.DeleteDeviceAsync(modelIds.Select(a => a.Id), AutoRestartThread, default));
|
await Task.Run(() => GlobalData.DeviceRuntimeService.DeleteDeviceAsync(modelIds.Select(a => a.Id), AutoRestartThread, default));
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
Spinner.SetRun(false);
|
Spinner.SetRun(false);
|
||||||
@@ -1002,7 +1001,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
var data = await GlobalData.GetCurrentUserDevices().ConfigureAwait(false);
|
var data = await GlobalData.GetCurrentUserDevices().ConfigureAwait(false);
|
||||||
|
|
||||||
await Task.Run(() => GlobalData.DeviceRuntimeService.DeleteDeviceAsync(data.Select(a => a.Id), AutoRestartThread, default));
|
await Task.Run(() => GlobalData.DeviceRuntimeService.DeleteDeviceAsync(data.Select(a => a.Id), AutoRestartThread, default));
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
Spinner.SetRun(false);
|
Spinner.SetRun(false);
|
||||||
@@ -1136,7 +1135,7 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
});
|
});
|
||||||
|
|
||||||
await Task.Run(() => GlobalData.DeviceRuntimeService.ImportDeviceAsync(value, AutoRestartThread));
|
await Task.Run(() => GlobalData.DeviceRuntimeService.ImportDeviceAsync(value, AutoRestartThread));
|
||||||
await Notify();
|
//await Notify();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
Spinner.SetRun(false);
|
Spinner.SetRun(false);
|
||||||
@@ -1210,6 +1209,8 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
private ChannelDeviceTreeItem UnknownItem = new() { ChannelDevicePluginType = ChannelDevicePluginTypeEnum.PluginType, PluginType = null };
|
private ChannelDeviceTreeItem UnknownItem = new() { ChannelDevicePluginType = ChannelDevicePluginTypeEnum.PluginType, PluginType = null };
|
||||||
|
|
||||||
private TreeViewItem<ChannelDeviceTreeItem> UnknownTreeViewItem;
|
private TreeViewItem<ChannelDeviceTreeItem> UnknownTreeViewItem;
|
||||||
|
|
||||||
|
SmartTriggerScheduler? scheduler;
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
|
|
||||||
@@ -1264,8 +1265,8 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
|
|
||||||
Items = ZItem;
|
Items = ZItem;
|
||||||
ChannelRuntimeDispatchService.Subscribe(Refresh);
|
ChannelRuntimeDispatchService.Subscribe(Refresh);
|
||||||
DeviceRuntimeDispatchService.Subscribe(Refresh);
|
|
||||||
|
|
||||||
|
scheduler = new SmartTriggerScheduler(Notify, TimeSpan.FromMilliseconds(3000));
|
||||||
|
|
||||||
_ = Task.Run(async () =>
|
_ = Task.Run(async () =>
|
||||||
{
|
{
|
||||||
@@ -1289,43 +1290,20 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
await base.OnInitializedAsync();
|
await base.OnInitializedAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
private WaitLock WaitLock = new();
|
|
||||||
|
|
||||||
private volatile bool _isExecuting = false;
|
|
||||||
|
|
||||||
private async Task Notify()
|
private async Task Notify()
|
||||||
{
|
{
|
||||||
if (_isExecuting) return;
|
if (Disposed) return;
|
||||||
try
|
await OnClickSearch(SearchText);
|
||||||
|
|
||||||
|
Value = GetValue(Value);
|
||||||
|
if (ChannelDeviceChanged != null)
|
||||||
{
|
{
|
||||||
await WaitLock.WaitAsync();
|
await ChannelDeviceChanged.Invoke(Value);
|
||||||
|
|
||||||
if (_isExecuting) return;
|
|
||||||
|
|
||||||
_isExecuting = true;
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (Disposed) return;
|
|
||||||
await OnClickSearch(SearchText);
|
|
||||||
|
|
||||||
Value = GetValue(Value);
|
|
||||||
if (ChannelDeviceChanged != null)
|
|
||||||
{
|
|
||||||
await ChannelDeviceChanged.Invoke(Value);
|
|
||||||
}
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
await Task.Delay(1000);
|
|
||||||
_isExecuting = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
WaitLock.Release();
|
|
||||||
}
|
}
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static ChannelDeviceTreeItem GetValue(ChannelDeviceTreeItem channelDeviceTreeItem)
|
private static ChannelDeviceTreeItem GetValue(ChannelDeviceTreeItem channelDeviceTreeItem)
|
||||||
@@ -1354,18 +1332,15 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
private async Task Refresh(DispatchEntry<DeviceRuntime> entry)
|
private Task Refresh(DispatchEntry<ChannelRuntime> entry)
|
||||||
{
|
{
|
||||||
await Notify();
|
scheduler.Trigger();
|
||||||
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
private async Task Refresh(DispatchEntry<ChannelRuntime> entry)
|
|
||||||
{
|
|
||||||
await Notify();
|
|
||||||
}
|
|
||||||
[Inject]
|
|
||||||
private IDispatchService<DeviceRuntime> DeviceRuntimeDispatchService { get; set; }
|
|
||||||
[Inject]
|
[Inject]
|
||||||
private IDispatchService<ChannelRuntime> ChannelRuntimeDispatchService { get; set; }
|
private IDispatchService<ChannelRuntime> ChannelRuntimeDispatchService { get; set; }
|
||||||
|
|
||||||
private string SearchText;
|
private string SearchText;
|
||||||
|
|
||||||
private async Task<List<TreeViewItem<ChannelDeviceTreeItem>>> OnClickSearch(string searchText)
|
private async Task<List<TreeViewItem<ChannelDeviceTreeItem>>> OnClickSearch(string searchText)
|
||||||
@@ -1523,7 +1498,6 @@ EventCallback.Factory.Create<MouseEventArgs>(this, async e =>
|
|||||||
{
|
{
|
||||||
Disposed = true;
|
Disposed = true;
|
||||||
ChannelRuntimeDispatchService.UnSubscribe(Refresh);
|
ChannelRuntimeDispatchService.UnSubscribe(Refresh);
|
||||||
DeviceRuntimeDispatchService.UnSubscribe(Refresh);
|
|
||||||
return base.DisposeAsync(disposing);
|
return base.DisposeAsync(disposing);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -13,8 +13,6 @@ using Mapster;
|
|||||||
using ThingsGateway.Admin.Application;
|
using ThingsGateway.Admin.Application;
|
||||||
using ThingsGateway.Gateway.Application;
|
using ThingsGateway.Gateway.Application;
|
||||||
|
|
||||||
using Yitter.IdGenerator;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Gateway.Razor;
|
namespace ThingsGateway.Gateway.Razor;
|
||||||
|
|
||||||
public partial class DeviceCopyComponent
|
public partial class DeviceCopyComponent
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
|
|
||||||
<DeviceRuntimeInfo1 DeviceRuntime="DeviceRuntime" />
|
<DeviceRuntimeInfo1 DeviceRuntime="DeviceRuntime" />
|
||||||
|
|
||||||
<LogConsole CardStyle="height: calc(100% - 330px);" LogLevel=@((DeviceRuntime?.Driver?.LogMessage)?.LogLevel??TouchSocket.Core.LogLevel.Trace)
|
<LogConsole HeightString="calc(100% - 270px)" LogLevel=@((DeviceRuntime?.Driver?.LogMessage)?.LogLevel ?? TouchSocket.Core.LogLevel.Trace)
|
||||||
LogLevelChanged="(logLevel)=>
|
LogLevelChanged="(logLevel)=>
|
||||||
{
|
{
|
||||||
DeviceRuntime.Driver?.SetLogAsync(logLevel);
|
DeviceRuntime.Driver?.SetLogAsync(logLevel);
|
||||||
|
|||||||
@@ -88,9 +88,9 @@ public partial class DeviceRuntimeInfo1 : IDisposable
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
OnParametersSet();
|
||||||
await InvokeAsync(() =>
|
await InvokeAsync(() =>
|
||||||
{
|
{
|
||||||
OnParametersSet();
|
|
||||||
StateHasChanged();
|
StateHasChanged();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ public partial class DeviceTable : IDisposable
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (table != null)
|
if (table != null)
|
||||||
await table.QueryAsync();
|
await InvokeAsync(table.QueryAsync);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -59,7 +59,7 @@ public partial class DeviceTable : IDisposable
|
|||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
await Task.Delay(1000);
|
await Task.Delay(5000);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -115,7 +115,7 @@ public partial class DeviceTable : IDisposable
|
|||||||
{
|
{
|
||||||
|
|
||||||
await Task.Run(() =>GlobalData.DeviceRuntimeService.CopyAsync(devices,AutoRestartThread, default));
|
await Task.Run(() =>GlobalData.DeviceRuntimeService.CopyAsync(devices,AutoRestartThread, default));
|
||||||
await table.QueryAsync();
|
await InvokeAsync(table.QueryAsync);
|
||||||
|
|
||||||
}},
|
}},
|
||||||
{nameof(DeviceCopyComponent.Model),oneModel },
|
{nameof(DeviceCopyComponent.Model),oneModel },
|
||||||
@@ -353,7 +353,7 @@ finally
|
|||||||
await InvokeAsync(async () =>
|
await InvokeAsync(async () =>
|
||||||
{
|
{
|
||||||
await ToastService.Default();
|
await ToastService.Default();
|
||||||
await InvokeAsync(table.QueryAsync);
|
await table.QueryAsync();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,9 +24,7 @@
|
|||||||
</FirstPaneTemplate>
|
</FirstPaneTemplate>
|
||||||
<SecondPaneTemplate>
|
<SecondPaneTemplate>
|
||||||
|
|
||||||
<div class="h-100 ms-1">
|
<GatewayInfo AutoRestartThread=AutoRestartThread SelectModel=SelectModel ShowChannelRuntime=ShowChannelRuntime ShowDeviceRuntime=ShowDeviceRuntime ShowType=ShowType VariableRuntimes=VariableRuntimes ChannelRuntimes="ChannelRuntimes" DeviceRuntimes="DeviceRuntimes" />
|
||||||
<GatewayInfo AutoRestartThread=AutoRestartThread SelectModel=SelectModel ShowChannelRuntime=ShowChannelRuntime ShowDeviceRuntime=ShowDeviceRuntime ShowType=ShowType VariableRuntimes=VariableRuntimes ChannelRuntimes="ChannelRuntimes" DeviceRuntimes="DeviceRuntimes" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</SecondPaneTemplate>
|
</SecondPaneTemplate>
|
||||||
</Split>
|
</Split>
|
||||||
|
|||||||
@@ -13,8 +13,6 @@ using Mapster;
|
|||||||
using ThingsGateway.Admin.Application;
|
using ThingsGateway.Admin.Application;
|
||||||
using ThingsGateway.Gateway.Application;
|
using ThingsGateway.Gateway.Application;
|
||||||
|
|
||||||
using Yitter.IdGenerator;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Gateway.Razor;
|
namespace ThingsGateway.Gateway.Razor;
|
||||||
|
|
||||||
public partial class VariableCopyComponent
|
public partial class VariableCopyComponent
|
||||||
|
|||||||
@@ -21,6 +21,7 @@
|
|||||||
ShowExtendButtons=true
|
ShowExtendButtons=true
|
||||||
ShowToolbar="true"
|
ShowToolbar="true"
|
||||||
ShowExportButton
|
ShowExportButton
|
||||||
|
IsAutoRefresh
|
||||||
ShowDefaultButtons=true
|
ShowDefaultButtons=true
|
||||||
ShowSearch=false
|
ShowSearch=false
|
||||||
ExtendButtonColumnWidth=220
|
ExtendButtonColumnWidth=220
|
||||||
|
|||||||
@@ -35,15 +35,29 @@ public partial class VariableRuntimeInfo : IDisposable
|
|||||||
|
|
||||||
[Parameter]
|
[Parameter]
|
||||||
public IEnumerable<VariableRuntime>? Items { get; set; } = Enumerable.Empty<VariableRuntime>();
|
public IEnumerable<VariableRuntime>? Items { get; set; } = Enumerable.Empty<VariableRuntime>();
|
||||||
|
private IEnumerable<VariableRuntime>? _previousItemsRef;
|
||||||
|
protected override async Task OnParametersSetAsync()
|
||||||
|
{
|
||||||
|
if (!ReferenceEquals(_previousItemsRef, Items))
|
||||||
|
{
|
||||||
|
_previousItemsRef = Items;
|
||||||
|
await Refresh(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void Dispose()
|
public void Dispose()
|
||||||
{
|
{
|
||||||
Disposed = true;
|
Disposed = true;
|
||||||
|
VariableRuntimeDispatchService.UnSubscribe(Refresh);
|
||||||
GC.SuppressFinalize(this);
|
GC.SuppressFinalize(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void OnInitialized()
|
protected override void OnInitialized()
|
||||||
{
|
{
|
||||||
|
VariableRuntimeDispatchService.Subscribe(Refresh);
|
||||||
|
|
||||||
|
scheduler = new SmartTriggerScheduler(Notify, TimeSpan.FromMilliseconds(1000));
|
||||||
|
|
||||||
_ = RunTimerAsync();
|
_ = RunTimerAsync();
|
||||||
base.OnInitialized();
|
base.OnInitialized();
|
||||||
}
|
}
|
||||||
@@ -62,6 +76,23 @@ public partial class VariableRuntimeInfo : IDisposable
|
|||||||
}
|
}
|
||||||
return Task.FromResult(ret);
|
return Task.FromResult(ret);
|
||||||
}
|
}
|
||||||
|
[Inject]
|
||||||
|
private IDispatchService<VariableRuntime> VariableRuntimeDispatchService { get; set; }
|
||||||
|
private SmartTriggerScheduler scheduler;
|
||||||
|
|
||||||
|
|
||||||
|
private Task Refresh(DispatchEntry<VariableRuntime> entry)
|
||||||
|
{
|
||||||
|
scheduler.Trigger();
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task Notify()
|
||||||
|
{
|
||||||
|
if (Disposed) return;
|
||||||
|
if (table != null)
|
||||||
|
await InvokeAsync(table.QueryAsync);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
private async Task RunTimerAsync()
|
private async Task RunTimerAsync()
|
||||||
@@ -70,8 +101,10 @@ public partial class VariableRuntimeInfo : IDisposable
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (table != null)
|
//if (table != null)
|
||||||
await table.QueryAsync();
|
// await table.QueryAsync();
|
||||||
|
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -40,7 +40,7 @@
|
|||||||
</label>
|
</label>
|
||||||
|
|
||||||
<label class="form-control">
|
<label class="form-control">
|
||||||
@(AuthorizeInfo != null ? Localizer["Authorized"] : Localizer["Unauthorized"])
|
@(AuthorizeInfo?.Auth==true ? Localizer["Authorized"] : Localizer["Unauthorized"])
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -50,6 +50,7 @@
|
|||||||
@if (AuthorizeInfo != null)
|
@if (AuthorizeInfo != null)
|
||||||
{
|
{
|
||||||
<div class="row g-3 form-inline">
|
<div class="row g-3 form-inline">
|
||||||
|
|
||||||
<div class="col-12 col-sm-12">
|
<div class="col-12 col-sm-12">
|
||||||
|
|
||||||
<label class="form-label">
|
<label class="form-label">
|
||||||
@@ -70,7 +71,7 @@
|
|||||||
</label>
|
</label>
|
||||||
|
|
||||||
<label class="form-control">
|
<label class="form-control">
|
||||||
@AuthorizeInfo?.ExpireTime
|
@AuthorizeInfo?.RealExpireTime
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -55,7 +55,7 @@ public partial class Authentication
|
|||||||
private async Task Unregister()
|
private async Task Unregister()
|
||||||
{
|
{
|
||||||
ProAuthentication.UnAuthorize();
|
ProAuthentication.UnAuthorize();
|
||||||
var result = ProAuthentication.TryGetAuthorizeInfo(out var authorizeInfo);
|
_ = ProAuthentication.TryGetAuthorizeInfo(out var authorizeInfo);
|
||||||
AuthorizeInfo = authorizeInfo;
|
AuthorizeInfo = authorizeInfo;
|
||||||
|
|
||||||
await InvokeAsync(StateHasChanged);
|
await InvokeAsync(StateHasChanged);
|
||||||
|
|||||||
@@ -31,8 +31,14 @@ public partial class RedundancyOptionsHeader : IDisposable
|
|||||||
{
|
{
|
||||||
while (!Disposed)
|
while (!Disposed)
|
||||||
{
|
{
|
||||||
await InvokeAsync(StateHasChanged);
|
try
|
||||||
await Task.Delay(3000);
|
{
|
||||||
|
await InvokeAsync(StateHasChanged);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
await Task.Delay(3000);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
return base.OnInitializedAsync();
|
return base.OnInitializedAsync();
|
||||||
|
|||||||
@@ -15,13 +15,13 @@
|
|||||||
|
|
||||||
<EditComponent ItemsPerRow=1 Model="Model" OnSave="OnSaveRedundancy" />
|
<EditComponent ItemsPerRow=1 Model="Model" OnSave="OnSaveRedundancy" />
|
||||||
|
|
||||||
<Button IsDisabled=@(Model.IsMaster) OnClick="ForcedSync">@RedundancyLocalizer["ForcedSync"]</Button>
|
<Button IsDisabled=@(Model.IsMaster||(!Model.Enable)) OnClick="ForcedSync">@RedundancyLocalizer["ForcedSync"]</Button>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-12 col-md-6 h-100">
|
<div class="col-12 col-md-6 h-100">
|
||||||
|
|
||||||
@if (Logger != null)
|
@if (Logger != null)
|
||||||
{
|
{
|
||||||
<LogConsole LogLevel=@(Logger.LogLevel) LogLevelChanged="(a)=>{
|
<LogConsole HeightString="calc(100% - 100px)" LogLevel=@(Logger.LogLevel) LogLevelChanged="(a)=>{
|
||||||
Logger.LogLevel=a;
|
Logger.LogLevel=a;
|
||||||
}" LogPath=@LogPath HeaderText=@HeaderText></LogConsole>
|
}" LogPath=@LogPath HeaderText=@HeaderText></LogConsole>
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,26 +5,34 @@
|
|||||||
|
|
||||||
<div class="h-100">
|
<div class="h-100">
|
||||||
<DefaultTable TItem="UpdateZipFile"
|
<DefaultTable TItem="UpdateZipFile"
|
||||||
AutoGenerateColumns="true"
|
AutoGenerateColumns="true"
|
||||||
ShowDefaultButtons=false
|
ShowDefaultButtons=false
|
||||||
EditDialogSize="Size.Large"
|
EditDialogSize="Size.Large"
|
||||||
AllowResizing="true"
|
AllowResizing="true"
|
||||||
IsFixedHeader=true
|
IsFixedHeader=true
|
||||||
ShowAddButton="false"
|
ShowAddButton="false"
|
||||||
ShowRefresh="true"
|
ShowRefresh="true"
|
||||||
ShowEmpty="true"
|
ShowEmpty="true"
|
||||||
ShowSearch="false"
|
ShowSearch="false"
|
||||||
IsMultipleSelect=false
|
IsMultipleSelect=false
|
||||||
ShowExtendEditButton=false
|
ShowExtendEditButton=false
|
||||||
ShowExtendDeleteButton=false
|
ShowExtendDeleteButton=false
|
||||||
ShowExtendButtons=true
|
ShowExtendButtons=true
|
||||||
ShowDeleteButton="false"
|
ShowDeleteButton="false"
|
||||||
ShowEditButton="false"
|
ShowEditButton="false"
|
||||||
ShowAdvancedSearch=false
|
ShowAdvancedSearch=false
|
||||||
OnQueryAsync="OnQueryAsync">
|
OnQueryAsync="OnQueryAsync">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@* <TableToolbarTemplate>
|
||||||
|
<TableToolbarButton TItem="UpdateZipFile" Color="Color.Info" Icon="fa fa-plus" Text="test"
|
||||||
|
OnClickCallback="a=>ShowInfo(a.FirstOrDefault())" />
|
||||||
|
|
||||||
|
</TableToolbarTemplate> *@
|
||||||
|
|
||||||
<RowButtonTemplate>
|
<RowButtonTemplate>
|
||||||
<TableCellButton Size="Size.ExtraSmall" Color="Color.Success" Icon="fa-solid fa-people-roof" Text="@ManagementLocalizer["Info"]" OnClick="()=>ShowInfo(context)" />
|
<TableCellButton Size="Size.ExtraSmall" Color="Color.Success" Icon="fa-solid fa-people-roof" Text="@ManagementLocalizer["Info"]" OnClick="() => ShowInfo(context)" />
|
||||||
</RowButtonTemplate>
|
</RowButtonTemplate>
|
||||||
|
|
||||||
</DefaultTable>
|
</DefaultTable>
|
||||||
|
|||||||
@@ -167,15 +167,15 @@ public partial class RulesPage
|
|||||||
|
|
||||||
private async Task Notify()
|
private async Task Notify()
|
||||||
{
|
{
|
||||||
var current = ExecutionContext.Capture();
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
ExecutionContext.Restore(context);
|
if (table != null)
|
||||||
|
await table.QueryAsync();
|
||||||
await InvokeAsync(StateHasChanged);
|
await InvokeAsync(StateHasChanged);
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
ExecutionContext.Restore(current);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -11,9 +11,9 @@
|
|||||||
|
|
||||||
@if (_rules != null)
|
@if (_rules != null)
|
||||||
{
|
{
|
||||||
<LogConsole LogLevel=@(_rules.Log.LogLevel) LogLevelChanged="(a)=>{
|
<LogConsole HeightString="100%" LogLevel=@(_rules.Log.LogLevel) LogLevelChanged="(a)=>{
|
||||||
_rules.Log.LogLevel=a;
|
_rules.Log.LogLevel=a;
|
||||||
}" LogPath=@_rules.Log.LogPath HeaderText="@_rules.Rules.Name" HeightString="calc(100vh - var(--bs-header-height) - var(--bb-layout-header-height) - 130px)"></LogConsole>
|
}" LogPath=@_rules.Log.LogPath HeaderText="@_rules.Rules.Name"></LogConsole>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -35,9 +35,12 @@ internal sealed class RulesEngineHostedService : BackgroundService, IRulesEngine
|
|||||||
internal const string LogPathFormat = "Logs/RulesEngineLog/{0}";
|
internal const string LogPathFormat = "Logs/RulesEngineLog/{0}";
|
||||||
internal const string LogDir = "Logs/RulesEngineLog";
|
internal const string LogDir = "Logs/RulesEngineLog";
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger _logger;
|
||||||
|
private IDispatchService<Rules> dispatchService;
|
||||||
|
|
||||||
/// <inheritdoc cref="RulesEngineHostedService"/>
|
/// <inheritdoc cref="RulesEngineHostedService"/>
|
||||||
public RulesEngineHostedService(ILogger<RulesEngineHostedService> logger, IStringLocalizer<RulesEngineHostedService> localizer)
|
public RulesEngineHostedService(ILogger<RulesEngineHostedService> logger, IStringLocalizer<RulesEngineHostedService> localizer)
|
||||||
{
|
{
|
||||||
|
dispatchService = App.GetService<IDispatchService<Rules>>();
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
Localizer = localizer;
|
Localizer = localizer;
|
||||||
}
|
}
|
||||||
@@ -60,8 +63,7 @@ internal sealed class RulesEngineHostedService : BackgroundService, IRulesEngine
|
|||||||
{
|
{
|
||||||
var data = Init(rules);
|
var data = Init(rules);
|
||||||
await Start(data.rulesLog, data.blazorDiagram, TokenSource.Token).ConfigureAwait(false);
|
await Start(data.rulesLog, data.blazorDiagram, TokenSource.Token).ConfigureAwait(false);
|
||||||
var service = App.GetService<IDispatchService<Rules>>();
|
dispatchService.Dispatch(null);
|
||||||
service.Dispatch(new());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -89,8 +91,7 @@ internal sealed class RulesEngineHostedService : BackgroundService, IRulesEngine
|
|||||||
BlazorDiagrams.Remove(del.Key);
|
BlazorDiagrams.Remove(del.Key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var service = App.GetService<IDispatchService<Rules>>();
|
dispatchService.Dispatch(null);
|
||||||
service.Dispatch(new());
|
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
@@ -198,8 +199,8 @@ internal sealed class RulesEngineHostedService : BackgroundService, IRulesEngine
|
|||||||
var item = Init(rules);
|
var item = Init(rules);
|
||||||
await Start(item.rulesLog, item.blazorDiagram, cancellationToken).ConfigureAwait(false);
|
await Start(item.rulesLog, item.blazorDiagram, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
var service = App.GetService<IDispatchService<Rules>>();
|
dispatchService.Dispatch(null);
|
||||||
service.Dispatch(new());
|
|
||||||
|
|
||||||
_ = Task.Factory.StartNew(async () =>
|
_ = Task.Factory.StartNew(async () =>
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ public class PackHelper
|
|||||||
// 获取变量的位偏移量
|
// 获取变量的位偏移量
|
||||||
//if (item.DataType == DataTypeEnum.Boolean)
|
//if (item.DataType == DataTypeEnum.Boolean)
|
||||||
item.Index = device.GetBitOffsetDefault(address);
|
item.Index = device.GetBitOffsetDefault(address);
|
||||||
if (item.DataType == DataTypeEnum.Byte&& !(item.ArrayLength>1))
|
if (item.DataType == DataTypeEnum.Byte && !(item.ArrayLength > 1))
|
||||||
item.Index += (item.Index % 2 == 0) ? 1 : -1;
|
item.Index += (item.Index % 2 == 0) ? 1 : -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -22,8 +22,6 @@ using TouchSocket.Dmtp.Rpc;
|
|||||||
using TouchSocket.Rpc;
|
using TouchSocket.Rpc;
|
||||||
using TouchSocket.Sockets;
|
using TouchSocket.Sockets;
|
||||||
|
|
||||||
using Yitter.IdGenerator;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Plugin.Synchronization;
|
namespace ThingsGateway.Plugin.Synchronization;
|
||||||
|
|
||||||
|
|
||||||
@@ -63,11 +61,8 @@ public partial class Synchronization : BusinessBase, IRpcDriver
|
|||||||
|
|
||||||
protected override async ValueTask ProtectedExecuteAsync(CancellationToken cancellationToken)
|
protected override async ValueTask ProtectedExecuteAsync(CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
|
||||||
|
|
||||||
if (_driverPropertys.IsServer)
|
if (_driverPropertys.IsServer)
|
||||||
{
|
{
|
||||||
if (_tcpDmtpService.ServerState != ServerState.Running)
|
if (_tcpDmtpService.ServerState != ServerState.Running)
|
||||||
@@ -132,6 +127,7 @@ public partial class Synchronization : BusinessBase, IRpcDriver
|
|||||||
|
|
||||||
|
|
||||||
// 如果 online 为 true,表示设备在线
|
// 如果 online 为 true,表示设备在线
|
||||||
|
if (online)
|
||||||
{
|
{
|
||||||
var deviceRunTimes = CollectDevices.Where(a => a.Value.IsCollect == true).Select(a => a.Value).Adapt<List<DeviceDataWithValue>>();
|
var deviceRunTimes = CollectDevices.Where(a => a.Value.IsCollect == true).Select(a => a.Value).Adapt<List<DeviceDataWithValue>>();
|
||||||
|
|
||||||
@@ -308,9 +304,11 @@ public partial class Synchronization : BusinessBase, IRpcDriver
|
|||||||
a.Channel.Id = CommonUtils.GetSingleId();
|
a.Channel.Id = CommonUtils.GetSingleId();
|
||||||
a.DeviceVariables.ForEach(b =>
|
a.DeviceVariables.ForEach(b =>
|
||||||
{
|
{
|
||||||
b.Device.Id = 0;
|
b.Device.ChannelId = a.Channel.Id;
|
||||||
|
b.Device.Id = CommonUtils.GetSingleId();
|
||||||
b.Variables.ForEach(c =>
|
b.Variables.ForEach(c =>
|
||||||
{
|
{
|
||||||
|
c.DeviceId = b.Device.Id;
|
||||||
c.Id = 0;
|
c.Id = 0;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -340,12 +338,12 @@ public partial class Synchronization : BusinessBase, IRpcDriver
|
|||||||
{
|
{
|
||||||
if (deviceDatas.TryGetValue(item.Key.DeviceName ?? string.Empty, out var variableDatas))
|
if (deviceDatas.TryGetValue(item.Key.DeviceName ?? string.Empty, out var variableDatas))
|
||||||
{
|
{
|
||||||
variableDatas.Add(item.Key.Name, item.Value?.ToString() ?? string.Empty);
|
variableDatas.TryAdd(item.Key.Name, item.Value?.ToString() ?? string.Empty);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
deviceDatas.Add(item.Key.DeviceName ?? string.Empty, new());
|
deviceDatas.TryAdd(item.Key.DeviceName ?? string.Empty, new());
|
||||||
deviceDatas[item.Key.DeviceName ?? string.Empty].Add(item.Key.Name, item.Value?.ToString() ?? string.Empty);
|
deviceDatas[item.Key.DeviceName ?? string.Empty].TryAdd(item.Key.Name, item.Value?.ToString() ?? string.Empty);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -402,7 +400,7 @@ public partial class Synchronization : BusinessBase, IRpcDriver
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
|
||||||
var data = await _tcpDmtpClient.GetDmtpRpcActor().InvokeTAsync<Dictionary<string, Dictionary<string, OperResult<object>>>>(
|
var data = await client.GetDmtpRpcActor().InvokeTAsync<Dictionary<string, Dictionary<string, OperResult<object>>>>(
|
||||||
nameof(ReverseCallbackServer.Rpc), waitInvoke, new Dictionary<string, Dictionary<string, string>>() { { item.Key, item.Value } }).ConfigureAwait(false);
|
nameof(ReverseCallbackServer.Rpc), waitInvoke, new Dictionary<string, Dictionary<string, string>>() { { item.Key, item.Value } }).ConfigureAwait(false);
|
||||||
|
|
||||||
dataResult.AddRange(data);
|
dataResult.AddRange(data);
|
||||||
@@ -415,7 +413,7 @@ public partial class Synchronization : BusinessBase, IRpcDriver
|
|||||||
|
|
||||||
foreach (var vItem in item.Value)
|
foreach (var vItem in item.Value)
|
||||||
{
|
{
|
||||||
dataResult[item.Key].Add(vItem.Key, new OperResult<object>(ex));
|
dataResult[item.Key].TryAdd(vItem.Key, new OperResult<object>(ex));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -426,7 +424,7 @@ public partial class Synchronization : BusinessBase, IRpcDriver
|
|||||||
|
|
||||||
foreach (var vItem in item.Value)
|
foreach (var vItem in item.Value)
|
||||||
{
|
{
|
||||||
dataResult[item.Key].Add(vItem.Key, new OperResult<object>("No online"));
|
dataResult[item.Key].TryAdd(vItem.Key, new OperResult<object>("No online"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ namespace ThingsGateway.Plugin.Synchronization;
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// <inheritdoc/>
|
/// <inheritdoc/>
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class SynchronizationProperty : BusinessPropertyBase
|
public class SynchronizationProperty : BusinessPropertyBase, IBusinessPropertyAllVariableBase
|
||||||
{
|
{
|
||||||
[DynamicProperty]
|
[DynamicProperty]
|
||||||
public bool IsServer { get; set; } = true;
|
public bool IsServer { get; set; } = true;
|
||||||
|
|||||||
@@ -37,7 +37,6 @@ using ThingsGateway.Debug;
|
|||||||
using ThingsGateway.Extension;
|
using ThingsGateway.Extension;
|
||||||
using ThingsGateway.Gateway.Application;
|
using ThingsGateway.Gateway.Application;
|
||||||
using ThingsGateway.NewLife.Caching;
|
using ThingsGateway.NewLife.Caching;
|
||||||
using ThingsGateway.NewLife.Extension;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Server;
|
namespace ThingsGateway.Server;
|
||||||
|
|
||||||
@@ -138,7 +137,8 @@ public class Startup : AppStartup
|
|||||||
{
|
{
|
||||||
options.WriteFilter = (logMsg) =>
|
options.WriteFilter = (logMsg) =>
|
||||||
{
|
{
|
||||||
if (logMsg.Message.IsNullOrEmpty()) return false;
|
if (App.HostApplicationLifetime.ApplicationStopping.IsCancellationRequested && logMsg.LogLevel >= LogLevel.Warning) return false;
|
||||||
|
if (string.IsNullOrEmpty(logMsg.Message)) return false;
|
||||||
else return true;
|
else return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -11,6 +11,7 @@
|
|||||||
|
|
||||||
// nuget动态加载的程序集
|
// nuget动态加载的程序集
|
||||||
"SupportPackageNamePrefixs": [
|
"SupportPackageNamePrefixs": [
|
||||||
|
"ThingsGateway.SqlSugar",
|
||||||
"ThingsGateway.Admin.Application",
|
"ThingsGateway.Admin.Application",
|
||||||
"ThingsGateway.Admin.Razor",
|
"ThingsGateway.Admin.Razor",
|
||||||
"ThingsGateway.Gateway.Application",
|
"ThingsGateway.Gateway.Application",
|
||||||
|
|||||||
@@ -11,6 +11,7 @@
|
|||||||
|
|
||||||
// nuget动态加载的程序集
|
// nuget动态加载的程序集
|
||||||
"SupportPackageNamePrefixs": [
|
"SupportPackageNamePrefixs": [
|
||||||
|
"ThingsGateway.SqlSugar",
|
||||||
"ThingsGateway.Admin.Application",
|
"ThingsGateway.Admin.Application",
|
||||||
"ThingsGateway.Admin.Razor",
|
"ThingsGateway.Admin.Razor",
|
||||||
"ThingsGateway.Gateway.Application",
|
"ThingsGateway.Gateway.Application",
|
||||||
|
|||||||
@@ -41,6 +41,7 @@ public class SingleFilePublish : ISingleFilePublish
|
|||||||
"ThingsGateway.Razor",
|
"ThingsGateway.Razor",
|
||||||
"ThingsGateway.Admin.Razor" ,
|
"ThingsGateway.Admin.Razor" ,
|
||||||
"ThingsGateway.Admin.Application",
|
"ThingsGateway.Admin.Application",
|
||||||
|
"ThingsGateway.SqlSugar",
|
||||||
|
|
||||||
"ThingsGateway.Management",
|
"ThingsGateway.Management",
|
||||||
"ThingsGateway.RulesEngine",
|
"ThingsGateway.RulesEngine",
|
||||||
|
|||||||
@@ -29,7 +29,6 @@ using ThingsGateway.Admin.Application;
|
|||||||
using ThingsGateway.Admin.Razor;
|
using ThingsGateway.Admin.Razor;
|
||||||
using ThingsGateway.Extension;
|
using ThingsGateway.Extension;
|
||||||
using ThingsGateway.NewLife.Caching;
|
using ThingsGateway.NewLife.Caching;
|
||||||
using ThingsGateway.NewLife.Extension;
|
|
||||||
|
|
||||||
namespace ThingsGateway.Server;
|
namespace ThingsGateway.Server;
|
||||||
|
|
||||||
@@ -162,7 +161,8 @@ public class Startup : AppStartup
|
|||||||
{
|
{
|
||||||
options.WriteFilter = (logMsg) =>
|
options.WriteFilter = (logMsg) =>
|
||||||
{
|
{
|
||||||
if (logMsg.Message.IsNullOrEmpty()) return false;
|
if (App.HostApplicationLifetime.ApplicationStopping.IsCancellationRequested && logMsg.LogLevel >= LogLevel.Warning) return false;
|
||||||
|
if (string.IsNullOrEmpty(logMsg.Message)) return false;
|
||||||
else return true;
|
else return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -101,6 +101,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ThingsGateway.UpgradeServer
|
|||||||
EndProject
|
EndProject
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ThingsGateway.Plugin.Synchronization", "Plugin\ThingsGateway.Plugin.Synchronization\ThingsGateway.Plugin.Synchronization.csproj", "{438B86D4-0CAE-DCC3-E952-90CE77BB8661}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ThingsGateway.Plugin.Synchronization", "Plugin\ThingsGateway.Plugin.Synchronization\ThingsGateway.Plugin.Synchronization.csproj", "{438B86D4-0CAE-DCC3-E952-90CE77BB8661}"
|
||||||
EndProject
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ThingsGateway.SqlSugar", "Admin\ThingsGateway.SqlSugar\ThingsGateway.SqlSugar.csproj", "{544EDA9F-978F-84F7-48BF-FA5888F52FFB}"
|
||||||
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Debug|Any CPU = Debug|Any CPU
|
Debug|Any CPU = Debug|Any CPU
|
||||||
@@ -267,6 +269,10 @@ Global
|
|||||||
{438B86D4-0CAE-DCC3-E952-90CE77BB8661}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{438B86D4-0CAE-DCC3-E952-90CE77BB8661}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{438B86D4-0CAE-DCC3-E952-90CE77BB8661}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{438B86D4-0CAE-DCC3-E952-90CE77BB8661}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{438B86D4-0CAE-DCC3-E952-90CE77BB8661}.Release|Any CPU.Build.0 = Release|Any CPU
|
{438B86D4-0CAE-DCC3-E952-90CE77BB8661}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{544EDA9F-978F-84F7-48BF-FA5888F52FFB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{544EDA9F-978F-84F7-48BF-FA5888F52FFB}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{544EDA9F-978F-84F7-48BF-FA5888F52FFB}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{544EDA9F-978F-84F7-48BF-FA5888F52FFB}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
HideSolutionNode = FALSE
|
HideSolutionNode = FALSE
|
||||||
@@ -310,10 +316,11 @@ Global
|
|||||||
{7D5E01DE-D6D7-E45D-58FD-E01B38A312B2} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
|
{7D5E01DE-D6D7-E45D-58FD-E01B38A312B2} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
|
||||||
{29DCAC9C-2D0F-E251-E907-F07D804CA117} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
|
{29DCAC9C-2D0F-E251-E907-F07D804CA117} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
|
||||||
{438B86D4-0CAE-DCC3-E952-90CE77BB8661} = {36510D70-161F-4241-B8D0-781E21032816}
|
{438B86D4-0CAE-DCC3-E952-90CE77BB8661} = {36510D70-161F-4241-B8D0-781E21032816}
|
||||||
|
{544EDA9F-978F-84F7-48BF-FA5888F52FFB} = {72C65578-92A5-4E99-9779-27835B12B32F}
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||||
RESX_Rules = {"EnabledRules":[]}
|
|
||||||
RESX_NeutralResourcesLanguage = zh-Hans
|
|
||||||
SolutionGuid = {199B1B96-4F56-4828-9531-813BA02DB282}
|
SolutionGuid = {199B1B96-4F56-4828-9531-813BA02DB282}
|
||||||
|
RESX_NeutralResourcesLanguage = zh-Hans
|
||||||
|
RESX_Rules = {"EnabledRules":[]}
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
EndGlobal
|
EndGlobal
|
||||||
|
|||||||
@@ -41,6 +41,7 @@ public class SingleFilePublish : ISingleFilePublish
|
|||||||
"ThingsGateway.Razor",
|
"ThingsGateway.Razor",
|
||||||
"ThingsGateway.Admin.Razor" ,
|
"ThingsGateway.Admin.Razor" ,
|
||||||
"ThingsGateway.Admin.Application",
|
"ThingsGateway.Admin.Application",
|
||||||
|
"ThingsGateway.SqlSugar",
|
||||||
|
|
||||||
"ThingsGateway.Foundation.Razor" ,
|
"ThingsGateway.Foundation.Razor" ,
|
||||||
"ThingsGateway.UpgradeServer" ,
|
"ThingsGateway.UpgradeServer" ,
|
||||||
|
|||||||
@@ -157,6 +157,7 @@ public class Startup : AppStartup
|
|||||||
{
|
{
|
||||||
options.WriteFilter = (logMsg) =>
|
options.WriteFilter = (logMsg) =>
|
||||||
{
|
{
|
||||||
|
if (App.HostApplicationLifetime.ApplicationStopping.IsCancellationRequested && logMsg.LogLevel >= LogLevel.Warning) return false;
|
||||||
if (string.IsNullOrEmpty(logMsg.Message)) return false;
|
if (string.IsNullOrEmpty(logMsg.Message)) return false;
|
||||||
else return true;
|
else return true;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<Project>
|
<Project>
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<Version>10.6.29</Version>
|
<Version>10.7.0</Version>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|||||||
Reference in New Issue
Block a user