mirror of
https://gitee.com/ThingsGateway/ThingsGateway.git
synced 2025-10-26 13:25:18 +08:00
Compare commits
24 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b64ac0539e | ||
|
|
541c60b363 | ||
|
|
824e95f7cb | ||
|
|
38f7850196 | ||
|
|
bef9de88e2 | ||
|
|
48cd5e7c7f | ||
|
|
3b44fda51c | ||
|
|
dbfc9a5bb4 | ||
|
|
1b758aa41a | ||
|
|
43bdc70899 | ||
|
|
fadda000a6 | ||
|
|
45a8c91a5a | ||
|
|
8e938f18be | ||
|
|
ab1b364c54 | ||
|
|
5ec65b2fb0 | ||
|
|
926eced724 | ||
|
|
f7f8802272 | ||
|
|
c6910dff02 | ||
|
|
ad299d0dbb | ||
|
|
8b124d1050 | ||
|
|
ff41080dbd | ||
|
|
0e28606e3d | ||
|
|
6a025ceee5 | ||
|
|
6b2e53d6dc |
3
.gitignore
vendored
3
.gitignore
vendored
@@ -365,4 +365,5 @@ FodyWeavers.xsd
|
||||
/src/*Pro*/
|
||||
/src/*Pro*
|
||||
/src/*pro*
|
||||
/src/*pro*/
|
||||
/src/*pro*/
|
||||
/src/ThingsGateway.Server/Configuration/GiteeOAuthSettings.json
|
||||
|
||||
@@ -0,0 +1,21 @@
|
||||
//------------------------------------------------------------------------------
|
||||
// 此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
|
||||
// 此代码版权(除特别声明外的代码)归作者本人Diego所有
|
||||
// 源代码使用协议遵循本仓库的开源协议及附加协议
|
||||
// Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
|
||||
// Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
|
||||
// 使用文档:https://thingsgateway.cn/
|
||||
// QQ群:605534569
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
|
||||
[AttributeUsage(AttributeTargets.Method)]
|
||||
public sealed class LoginLogAttribute : Attribute
|
||||
{
|
||||
}
|
||||
[AttributeUsage(AttributeTargets.Method)]
|
||||
public sealed class LogoutLogAttribute : Attribute
|
||||
{
|
||||
}
|
||||
@@ -8,6 +8,7 @@
|
||||
// QQ群:605534569
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
using Microsoft.AspNetCore.Authentication;
|
||||
using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
|
||||
@@ -27,14 +28,31 @@ public class AuthController : ControllerBase
|
||||
|
||||
[HttpPost("login")]
|
||||
[AllowAnonymous]
|
||||
[LoginLog]
|
||||
public Task<LoginOutput> LoginAsync([FromBody] LoginInput input)
|
||||
{
|
||||
|
||||
return _authService.LoginAsync(input);
|
||||
|
||||
}
|
||||
|
||||
[HttpGet("oauth-login")]
|
||||
[AllowAnonymous]
|
||||
[SuppressRequestAudit]
|
||||
public IActionResult OAuthLogin(string scheme = "Gitee", string returnUrl = "/")
|
||||
{
|
||||
var props = new AuthenticationProperties
|
||||
{
|
||||
RedirectUri = returnUrl
|
||||
};
|
||||
return Challenge(props, scheme);
|
||||
}
|
||||
|
||||
|
||||
[HttpPost("logout")]
|
||||
[Authorize]
|
||||
[IgnoreRolePermission]
|
||||
[LogoutLog]
|
||||
public Task LogoutAsync()
|
||||
{
|
||||
return _authService.LoginOutAsync();
|
||||
|
||||
@@ -8,6 +8,8 @@
|
||||
// QQ群:605534569
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
using System.Reflection;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
public class RequestAuditData
|
||||
@@ -94,5 +96,6 @@ public class RequestAuditData
|
||||
/// 验证错误信息
|
||||
/// </summary>
|
||||
public Validation Validation { get; set; }
|
||||
public MethodInfo MethodInfo { get; set; }
|
||||
}
|
||||
|
||||
|
||||
@@ -21,29 +21,36 @@ using System.Logging;
|
||||
using ThingsGateway.FriendlyException;
|
||||
using ThingsGateway.Logging;
|
||||
using ThingsGateway.NewLife.Json.Extension;
|
||||
using ThingsGateway.UnifyResult;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
public class RequestAuditFilter : IAsyncActionFilter
|
||||
public class RequestAuditFilter : IAsyncActionFilter, IOrderedFilter
|
||||
{
|
||||
private const int FilterOrder = -3000;
|
||||
public int Order => FilterOrder;
|
||||
|
||||
public async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
|
||||
{
|
||||
var timeOperation = Stopwatch.StartNew();
|
||||
var resultContext = await next().ConfigureAwait(false);
|
||||
// 计算接口执行时间
|
||||
timeOperation.Stop();
|
||||
|
||||
var controllerActionDescriptor = (context.ActionDescriptor as ControllerActionDescriptor);
|
||||
// 获取动作方法描述器
|
||||
var actionMethod = controllerActionDescriptor?.MethodInfo;
|
||||
|
||||
|
||||
// 处理 Blazor Server
|
||||
if (actionMethod == null)
|
||||
{
|
||||
_ = await next.Invoke().ConfigureAwait(false);
|
||||
return;
|
||||
}
|
||||
|
||||
// 排除 WebSocket 请求处理
|
||||
if (context.HttpContext.IsWebSocketRequest())
|
||||
{
|
||||
_ = await next().ConfigureAwait(false);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -51,7 +58,6 @@ public class RequestAuditFilter : IAsyncActionFilter
|
||||
if (actionMethod.IsDefined(typeof(SuppressRequestAuditAttribute), true)
|
||||
|| actionMethod.DeclaringType.IsDefined(typeof(SuppressRequestAuditAttribute), true))
|
||||
{
|
||||
_ = await next().ConfigureAwait(false);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -65,10 +71,7 @@ public class RequestAuditFilter : IAsyncActionFilter
|
||||
return;
|
||||
}
|
||||
|
||||
// 计算接口执行时间
|
||||
var timeOperation = Stopwatch.StartNew();
|
||||
var resultContext = await next().ConfigureAwait(false);
|
||||
timeOperation.Stop();
|
||||
|
||||
|
||||
|
||||
var logData = new RequestAuditData();
|
||||
@@ -88,21 +91,29 @@ public class RequestAuditFilter : IAsyncActionFilter
|
||||
var requestUrl = Uri.UnescapeDataString(httpRequest.GetRequestUrlAddress());
|
||||
logData.RequestUrl = requestUrl;
|
||||
|
||||
|
||||
object returnValue = null;
|
||||
Type finalReturnType;
|
||||
var result = resultContext.Result as IActionResult;
|
||||
var data = result switch
|
||||
// 解析返回值
|
||||
if (UnifyContext.CheckVaildResult(result, out var data))
|
||||
{
|
||||
// 处理内容结果
|
||||
ContentResult content => content.Content,
|
||||
// 处理对象结果
|
||||
ObjectResult obj => obj.Value,
|
||||
// 处理 JSON 对象
|
||||
JsonResult json => json.Value,
|
||||
_ => null,
|
||||
};
|
||||
logData.ReturnInformation = data;
|
||||
|
||||
returnValue = data;
|
||||
finalReturnType = data?.GetType();
|
||||
}
|
||||
// 处理文件类型
|
||||
else if (result is FileResult fresult)
|
||||
{
|
||||
returnValue = new
|
||||
{
|
||||
FileName = fresult.FileDownloadName,
|
||||
fresult.ContentType,
|
||||
Length = fresult is FileContentResult cresult ? (object)cresult.FileContents.Length : null
|
||||
};
|
||||
finalReturnType = fresult?.GetType();
|
||||
}
|
||||
else finalReturnType = result?.GetType();
|
||||
|
||||
logData.ReturnInformation = returnValue;
|
||||
|
||||
//获取客户端信息
|
||||
var client = App.GetService<IAppService>().UserAgent;
|
||||
@@ -112,12 +123,12 @@ public class RequestAuditFilter : IAsyncActionFilter
|
||||
var desc = App.CreateLocalizerByType(controllerActionDescriptor.ControllerTypeInfo.AsType())[actionMethod.Name];
|
||||
//获取特性
|
||||
|
||||
|
||||
logData.CateGory = desc.Value;//传操作名称
|
||||
logData.Operation = desc.Value;//传操作名称
|
||||
logData.Client = client;
|
||||
logData.Path = httpContext.Request.Path.Value;//请求地址
|
||||
logData.Method = httpContext.Request.Method;//请求方法
|
||||
logData.MethodInfo = actionMethod;//请求方法
|
||||
|
||||
logData.ControllerName = controllerActionDescriptor.ControllerName;
|
||||
logData.ActionName = controllerActionDescriptor.ActionName;
|
||||
|
||||
@@ -0,0 +1,289 @@
|
||||
using Microsoft.AspNetCore.Authentication;
|
||||
using Microsoft.AspNetCore.Authentication.OAuth;
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.AspNetCore.WebUtilities;
|
||||
using Microsoft.Extensions.DependencyInjection;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using Microsoft.Extensions.Options;
|
||||
|
||||
using System.Collections.Concurrent;
|
||||
using System.Net.Http.Headers;
|
||||
using System.Security.Claims;
|
||||
using System.Text;
|
||||
using System.Text.Encodings.Web;
|
||||
using System.Text.Json;
|
||||
|
||||
using ThingsGateway.Extension;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
/// <summary>
|
||||
/// 只适合 Demo 登录,会直接授权超管的权限
|
||||
/// </summary>
|
||||
public class AdminOAuthHandler<TOptions>(
|
||||
IVerificatInfoService verificatInfoService,
|
||||
IAppService appService,
|
||||
ISysUserService sysUserService,
|
||||
ISysDictService configService,
|
||||
IOptionsMonitor<TOptions> options,
|
||||
ILoggerFactory logger,
|
||||
IUserAgentService userAgentService,
|
||||
UrlEncoder encoder
|
||||
) : OAuthHandler<TOptions>(options, logger, encoder)
|
||||
where TOptions : AdminOAuthOptions, new()
|
||||
{
|
||||
private async Task<LoginEvent> GetLogin()
|
||||
{
|
||||
var sysUser = await sysUserService.GetUserByIdAsync(RoleConst.SuperAdminId).ConfigureAwait(false);//获取用户信息
|
||||
|
||||
var appConfig = await configService.GetAppConfigAsync().ConfigureAwait(false);
|
||||
|
||||
|
||||
var expire = appConfig.LoginPolicy.VerificatExpireTime;
|
||||
|
||||
var loginEvent = new LoginEvent
|
||||
{
|
||||
Ip = appService.RemoteIpAddress,
|
||||
Device = appService.UserAgent?.Platform,
|
||||
Expire = expire,
|
||||
SysUser = sysUser,
|
||||
VerificatId = CommonUtils.GetSingleId()
|
||||
};
|
||||
|
||||
//获取verificat列表
|
||||
var tokenTimeout = loginEvent.DateTime.AddMinutes(loginEvent.Expire);
|
||||
//生成verificat信息
|
||||
var verificatInfo = new VerificatInfo
|
||||
{
|
||||
Device = loginEvent.Device,
|
||||
Expire = loginEvent.Expire,
|
||||
VerificatTimeout = tokenTimeout,
|
||||
Id = loginEvent.VerificatId,
|
||||
UserId = loginEvent.SysUser.Id,
|
||||
LoginIp = loginEvent.Ip,
|
||||
LoginTime = loginEvent.DateTime
|
||||
};
|
||||
|
||||
|
||||
//添加到verificat列表
|
||||
verificatInfoService.Add(verificatInfo);
|
||||
|
||||
return loginEvent;
|
||||
}
|
||||
/// <summary>
|
||||
/// 登录事件
|
||||
/// </summary>
|
||||
/// <param name="loginEvent"></param>
|
||||
/// <returns></returns>
|
||||
private async Task UpdateUser(LoginEvent loginEvent)
|
||||
{
|
||||
var sysUser = loginEvent.SysUser;
|
||||
|
||||
#region 登录/密码策略
|
||||
|
||||
var key = CacheConst.Cache_LoginErrorCount + sysUser.Account;//获取登录错误次数Key值
|
||||
App.CacheService.Remove(key);//移除登录错误次数
|
||||
|
||||
//获取用户verificat列表
|
||||
var userToken = verificatInfoService.GetOne(loginEvent.VerificatId);
|
||||
|
||||
#endregion 登录/密码策略
|
||||
|
||||
#region 重新赋值属性,设置本次登录信息为最新的信息
|
||||
|
||||
sysUser.LastLoginIp = sysUser.LatestLoginIp;
|
||||
sysUser.LastLoginTime = sysUser.LatestLoginTime;
|
||||
sysUser.LatestLoginIp = loginEvent.Ip;
|
||||
sysUser.LatestLoginTime = loginEvent.DateTime;
|
||||
|
||||
#endregion 重新赋值属性,设置本次登录信息为最新的信息
|
||||
|
||||
using var db = DbContext.Db.GetConnectionScopeWithAttr<SysUser>().CopyNew();
|
||||
//更新用户登录信息
|
||||
if (await db.Updateable(sysUser).UpdateColumns(it => new
|
||||
{
|
||||
it.LastLoginIp,
|
||||
it.LastLoginTime,
|
||||
it.LatestLoginIp,
|
||||
it.LatestLoginTime,
|
||||
}).ExecuteCommandAsync().ConfigureAwait(false) > 0)
|
||||
App.CacheService.HashAdd(CacheConst.Cache_SysUser, sysUser.Id.ToString(), sysUser);//更新Cache信息
|
||||
}
|
||||
|
||||
|
||||
static AdminOAuthHandler()
|
||||
{
|
||||
Task.Factory.StartNew(Insertable, TaskCreationOptions.LongRunning);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// 日志消息队列(线程安全)
|
||||
/// </summary>
|
||||
protected static readonly ConcurrentQueue<SysOperateLog> _operateLogMessageQueue = new();
|
||||
|
||||
/// <summary>
|
||||
/// 创建访问日志
|
||||
/// </summary>
|
||||
private static async Task Insertable()
|
||||
{
|
||||
var db = DbContext.Db.GetConnectionScopeWithAttr<SysOperateLog>().CopyNew();
|
||||
var appLifetime = App.RootServices!.GetService<IHostApplicationLifetime>()!;
|
||||
while (!appLifetime.ApplicationStopping.IsCancellationRequested)
|
||||
{
|
||||
try
|
||||
{
|
||||
var data = _operateLogMessageQueue.ToListWithDequeue(); // 从日志队列中获取数据
|
||||
if (data.Count > 0)
|
||||
{
|
||||
await db.InsertableWithAttr(data).ExecuteCommandAsync(appLifetime.ApplicationStopping).ConfigureAwait(false);//入库
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
NewLife.Log.XTrace.WriteException(ex);
|
||||
}
|
||||
finally
|
||||
{
|
||||
await Task.Delay(3000, appLifetime.ApplicationStopping).ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
|
||||
protected override async Task<AuthenticationTicket> CreateTicketAsync(
|
||||
ClaimsIdentity identity,
|
||||
AuthenticationProperties properties,
|
||||
OAuthTokenResponse tokens)
|
||||
{
|
||||
properties.RedirectUri = Options.HomePath;
|
||||
properties.IsPersistent = true;
|
||||
|
||||
if (!string.IsNullOrEmpty(tokens.ExpiresIn) && int.TryParse(tokens.ExpiresIn, out var result))
|
||||
{
|
||||
properties.ExpiresUtc = TimeProvider.System.GetUtcNow().AddSeconds(result);
|
||||
}
|
||||
var user = await HandleUserInfoAsync(tokens).ConfigureAwait(false);
|
||||
|
||||
var loginEvent = await GetLogin().ConfigureAwait(false);
|
||||
await UpdateUser(loginEvent).ConfigureAwait(false);
|
||||
identity.AddClaim(new Claim(ClaimConst.VerificatId, loginEvent.VerificatId.ToString()));
|
||||
identity.AddClaim(new Claim(ClaimConst.UserId, RoleConst.SuperAdminId.ToString()));
|
||||
|
||||
identity.AddClaim(new Claim(ClaimConst.SuperAdmin, "true"));
|
||||
identity.AddClaim(new Claim(ClaimConst.OrgId, RoleConst.DefaultTenantId.ToString()));
|
||||
identity.AddClaim(new Claim(ClaimConst.TenantId, RoleConst.DefaultTenantId.ToString()));
|
||||
|
||||
|
||||
var context = new OAuthCreatingTicketContext(
|
||||
new ClaimsPrincipal(identity),
|
||||
properties,
|
||||
Context,
|
||||
Scheme,
|
||||
Options,
|
||||
Backchannel,
|
||||
tokens,
|
||||
user
|
||||
);
|
||||
|
||||
context.RunClaimActions();
|
||||
await Events.CreatingTicket(context).ConfigureAwait(false);
|
||||
|
||||
var httpContext = context.HttpContext;
|
||||
UserAgent? userAgent = null;
|
||||
var str = httpContext?.Request?.Headers?.UserAgent;
|
||||
if (!string.IsNullOrEmpty(str))
|
||||
{
|
||||
userAgent = userAgentService.Parse(str);
|
||||
}
|
||||
|
||||
var sysOperateLog = new SysOperateLog()
|
||||
{
|
||||
Name = this.Scheme.Name,
|
||||
Category = LogCateGoryEnum.Login,
|
||||
ExeStatus = true,
|
||||
OpIp = httpContext.GetRemoteIpAddressToIPv4(),
|
||||
OpBrowser = userAgent?.Browser,
|
||||
OpOs = userAgent?.Platform,
|
||||
OpTime = DateTime.Now,
|
||||
VerificatId = loginEvent.VerificatId,
|
||||
OpAccount = Options.GetName(user),
|
||||
|
||||
ReqMethod = "OAuth",
|
||||
ReqUrl = string.Empty,
|
||||
ResultJson = string.Empty,
|
||||
ClassName = nameof(AdminOAuthHandler<TOptions>),
|
||||
MethodName = string.Empty,
|
||||
ParamJson = string.Empty,
|
||||
};
|
||||
_operateLogMessageQueue.Enqueue(sysOperateLog);
|
||||
return new AuthenticationTicket(context.Principal, context.Properties, Scheme.Name);
|
||||
}
|
||||
|
||||
/// <summary>刷新 Token 方法</summary>
|
||||
protected virtual async Task<OAuthTokenResponse> RefreshTokenAsync(OAuthTokenResponse oAuthToken)
|
||||
{
|
||||
var query = new Dictionary<string, string>
|
||||
{
|
||||
{ "refresh_token", oAuthToken.RefreshToken },
|
||||
{ "grant_type", "refresh_token" }
|
||||
};
|
||||
|
||||
var request = new HttpRequestMessage(HttpMethod.Post, QueryHelpers.AddQueryString(Options.TokenEndpoint, query));
|
||||
request.Headers.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
|
||||
|
||||
var response = await Backchannel.SendAsync(request, Context.RequestAborted).ConfigureAwait(false);
|
||||
|
||||
var content = await response.Content.ReadAsStringAsync().ConfigureAwait(false);
|
||||
|
||||
if (response.IsSuccessStatusCode)
|
||||
{
|
||||
return OAuthTokenResponse.Success(JsonDocument.Parse(content));
|
||||
}
|
||||
|
||||
return OAuthTokenResponse.Failed(new OAuthTokenException($"OAuth token endpoint failure: {await Display(response).ConfigureAwait(false)}"));
|
||||
}
|
||||
|
||||
/// <summary>处理用户信息方法</summary>
|
||||
protected virtual async Task<JsonElement> HandleUserInfoAsync(OAuthTokenResponse tokens)
|
||||
{
|
||||
var request = new HttpRequestMessage(HttpMethod.Get, BuildUserInfoUrl(tokens));
|
||||
request.Headers.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
|
||||
|
||||
var response = await Backchannel.SendAsync(request, Context.RequestAborted).ConfigureAwait(false);
|
||||
|
||||
var content = await response.Content.ReadAsStringAsync().ConfigureAwait(false);
|
||||
|
||||
if (response.IsSuccessStatusCode)
|
||||
{
|
||||
return JsonDocument.Parse(content).RootElement;
|
||||
}
|
||||
|
||||
throw new OAuthTokenException($"OAuth user info endpoint failure: {await Display(response).ConfigureAwait(false)}");
|
||||
}
|
||||
|
||||
/// <summary>生成用户信息请求地址方法</summary>
|
||||
protected virtual string BuildUserInfoUrl(OAuthTokenResponse tokens)
|
||||
{
|
||||
return QueryHelpers.AddQueryString(Options.UserInformationEndpoint, new Dictionary<string, string>
|
||||
{
|
||||
{ "access_token", tokens.AccessToken }
|
||||
});
|
||||
}
|
||||
|
||||
/// <summary>生成错误信息方法</summary>
|
||||
protected static async Task<string> Display(HttpResponseMessage response)
|
||||
{
|
||||
var output = new StringBuilder();
|
||||
output.Append($"Status: {response.StatusCode}; ");
|
||||
output.Append($"Headers: {response.Headers}; ");
|
||||
output.Append($"Body: {await response.Content.ReadAsStringAsync().ConfigureAwait(false)};");
|
||||
|
||||
return output.ToString();
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>自定义 Token 异常</summary>
|
||||
public class OAuthTokenException(string message) : Exception(message);
|
||||
@@ -0,0 +1,40 @@
|
||||
using Microsoft.AspNetCore.Authentication.OAuth;
|
||||
|
||||
using System.Text.Json;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
/// <summary>OAuthOptions 配置类</summary>
|
||||
public abstract class AdminOAuthOptions : OAuthOptions
|
||||
{
|
||||
/// <summary>默认构造函数</summary>
|
||||
protected AdminOAuthOptions()
|
||||
{
|
||||
ConfigureClaims();
|
||||
this.Events.OnRemoteFailure = context =>
|
||||
{
|
||||
var redirectUri = string.IsNullOrEmpty(HomePath) ? "/" : HomePath;
|
||||
context.Response.Redirect(redirectUri);
|
||||
context.HandleResponse();
|
||||
return Task.CompletedTask;
|
||||
};
|
||||
|
||||
|
||||
}
|
||||
|
||||
/// <summary>配置 Claims 映射</summary>
|
||||
protected virtual void ConfigureClaims()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
public virtual string GetName(JsonElement element)
|
||||
{
|
||||
JsonElement.ObjectEnumerator target = element.EnumerateObject();
|
||||
return target.TryGetValue("name");
|
||||
}
|
||||
|
||||
/// <summary>获得/设置 登陆后首页</summary>
|
||||
public string HomePath { get; set; } = "/";
|
||||
|
||||
}
|
||||
@@ -0,0 +1,77 @@
|
||||
using Microsoft.AspNetCore.Authentication;
|
||||
using Microsoft.AspNetCore.Authentication.OAuth;
|
||||
using Microsoft.AspNetCore.WebUtilities;
|
||||
|
||||
using System.Net.Http.Headers;
|
||||
using System.Text.Json;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
public class GiteeOAuthOptions : AdminOAuthOptions
|
||||
{
|
||||
|
||||
public GiteeOAuthOptions() : base()
|
||||
{
|
||||
this.SignInScheme = ClaimConst.Scheme;
|
||||
this.AuthorizationEndpoint = "https://gitee.com/oauth/authorize";
|
||||
this.TokenEndpoint = "https://gitee.com/oauth/token";
|
||||
this.UserInformationEndpoint = "https://gitee.com/api/v5/user";
|
||||
this.HomePath = "/";
|
||||
this.CallbackPath = "/signin-gitee";
|
||||
Scope.Add("user_info");
|
||||
Scope.Add("projects");
|
||||
|
||||
Events.OnCreatingTicket = async context =>
|
||||
{
|
||||
await HandlerGiteeStarredUrl(context).ConfigureAwait(false);
|
||||
};
|
||||
|
||||
Events.OnRedirectToAuthorizationEndpoint = context =>
|
||||
{
|
||||
//context.RedirectUri = context.RedirectUri.Replace("http%3A%2F%2F", "https%3A%2F%2F"); // 强制替换
|
||||
context.Response.Redirect(context.RedirectUri);
|
||||
return Task.CompletedTask;
|
||||
};
|
||||
}
|
||||
|
||||
public override string GetName(JsonElement element)
|
||||
{
|
||||
JsonElement.ObjectEnumerator target = element.EnumerateObject();
|
||||
return target.TryGetValue("name");
|
||||
}
|
||||
|
||||
private static async Task HandlerGiteeStarredUrl(OAuthCreatingTicketContext context, string repoFullName = "ThingsGateway/ThingsGateway")
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(context.AccessToken))
|
||||
throw new InvalidOperationException("Access token is missing.");
|
||||
|
||||
var uri = $"https://gitee.com/api/v5/user/starred/{repoFullName}";
|
||||
|
||||
var queryString = new Dictionary<string, string>
|
||||
{
|
||||
{ "access_token", context.AccessToken }
|
||||
};
|
||||
|
||||
var request = new HttpRequestMessage(HttpMethod.Put, QueryHelpers.AddQueryString(uri, queryString))
|
||||
{
|
||||
Headers = { Accept = { new MediaTypeWithQualityHeaderValue("application/json") } }
|
||||
};
|
||||
|
||||
var response = await context.Backchannel.SendAsync(request, context.HttpContext.RequestAborted).ConfigureAwait(false);
|
||||
|
||||
if (!response.IsSuccessStatusCode)
|
||||
{
|
||||
var content = await response.Content.ReadAsStringAsync().ConfigureAwait(false);
|
||||
throw new Exception($"Failed to star repository: {response.StatusCode}, {content}");
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
protected override void ConfigureClaims()
|
||||
{
|
||||
ClaimActions.MapJsonKey(ClaimConst.AvatarUrl, "avatar_url");
|
||||
ClaimActions.MapJsonKey(ClaimConst.Account, "name");
|
||||
|
||||
base.ConfigureClaims();
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,7 @@
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
public class GiteeOAuthSettings
|
||||
{
|
||||
public string ClientId { get; set; }
|
||||
public string ClientSecret { get; set; }
|
||||
}
|
||||
@@ -0,0 +1,12 @@
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
public class GiteeOAuthUser
|
||||
{
|
||||
public string Id { get; set; }
|
||||
|
||||
public string Login { get; set; }
|
||||
|
||||
public string Name { get; set; }
|
||||
|
||||
public string Avatar_Url { get; set; }
|
||||
}
|
||||
@@ -0,0 +1,22 @@
|
||||
using System.Text.Json;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
public static class OAuthUserExtensions
|
||||
{
|
||||
public static GiteeOAuthUser ToAuthUser(this JsonElement element)
|
||||
{
|
||||
GiteeOAuthUser authUser = new GiteeOAuthUser();
|
||||
JsonElement.ObjectEnumerator target = element.EnumerateObject();
|
||||
authUser.Id = target.TryGetValue("id");
|
||||
authUser.Login = target.TryGetValue("login");
|
||||
authUser.Name = target.TryGetValue("name");
|
||||
authUser.Avatar_Url = target.TryGetValue("avatar_url");
|
||||
return authUser;
|
||||
}
|
||||
|
||||
public static string TryGetValue(this JsonElement.ObjectEnumerator target, string propertyName)
|
||||
{
|
||||
return target.FirstOrDefault<JsonProperty>((Func<JsonProperty, bool>)(t => t.Name.Equals(propertyName, StringComparison.OrdinalIgnoreCase))).Value.ToString() ?? string.Empty;
|
||||
}
|
||||
}
|
||||
@@ -11,6 +11,7 @@
|
||||
using SqlSugar;
|
||||
|
||||
using System.Collections.Concurrent;
|
||||
using System.Reflection;
|
||||
|
||||
using ThingsGateway.Extension;
|
||||
using ThingsGateway.FriendlyException;
|
||||
@@ -52,13 +53,18 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter
|
||||
var client = requestAuditData.Client;//获取客户端信息
|
||||
var path = requestAuditData.Path;//获取操作名称
|
||||
var method = requestAuditData.Method;//获取方法
|
||||
var methodInfo = requestAuditData.MethodInfo;
|
||||
var login = methodInfo.GetCustomAttribute(typeof(LoginLogAttribute));
|
||||
var logout = methodInfo.GetCustomAttribute(typeof(LogoutLogAttribute));
|
||||
|
||||
//表示访问日志
|
||||
if (path == "/api/auth/login" || path == "/api/auth/logout")
|
||||
if (login != null || logout != null)
|
||||
{
|
||||
//如果没有异常信息
|
||||
if (requestAuditData.Exception == null)
|
||||
{
|
||||
save = await CreateVisitLog(operation, path, requestAuditData, client, flush).ConfigureAwait(false);//添加到访问日志
|
||||
LogCateGoryEnum logCateGoryEnum = login != null ? LogCateGoryEnum.Login : LogCateGoryEnum.Logout;
|
||||
save = await CreateVisitLog(operation, path, requestAuditData, client, logCateGoryEnum, flush).ConfigureAwait(false);//添加到访问日志
|
||||
}
|
||||
else
|
||||
{
|
||||
@@ -152,18 +158,20 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter
|
||||
/// <param name="path"></param>
|
||||
/// <param name="requestAuditData">requestAuditData</param>
|
||||
/// <param name="userAgent">客户端信息</param>
|
||||
/// <param name="logCateGoryEnum">logCateGory</param>
|
||||
/// <param name="flush"></param>
|
||||
private async Task<bool> CreateVisitLog(string operation, string path, RequestAuditData requestAuditData, UserAgent userAgent, bool flush)
|
||||
private async Task<bool> CreateVisitLog(string operation, string path, RequestAuditData requestAuditData, UserAgent userAgent, LogCateGoryEnum logCateGoryEnum, bool flush)
|
||||
{
|
||||
long verificatId = 0;//验证Id
|
||||
var opAccount = "";//用户账号
|
||||
if (path == "/api/auth/login")
|
||||
if (logCateGoryEnum == LogCateGoryEnum.Login)
|
||||
{
|
||||
//如果是登录,用户信息就从返回值里拿
|
||||
var result = requestAuditData.ReturnInformation?.ToSystemTextJsonString();//返回值转json
|
||||
var userInfo = result.FromJsonNetString<UnifyResult<LoginOutput>>();//格式化成user表
|
||||
opAccount = userInfo.Data.Account;//赋值账号
|
||||
verificatId = userInfo.Data.VerificatId;
|
||||
if (requestAuditData.ReturnInformation is UnifyResult<LoginOutput> userInfo)
|
||||
{
|
||||
opAccount = userInfo.Data.Account;//赋值账号
|
||||
verificatId = userInfo.Data.VerificatId;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
@@ -175,7 +183,7 @@ public class DatabaseLoggingWriter : IDatabaseLoggingWriter
|
||||
var sysLogVisit = new SysOperateLog
|
||||
{
|
||||
Name = operation,
|
||||
Category = path == "/api/auth/login" ? LogCateGoryEnum.Login : LogCateGoryEnum.Logout,
|
||||
Category = logCateGoryEnum,
|
||||
ExeStatus = true,
|
||||
OpIp = requestAuditData.RemoteIPv4,
|
||||
OpBrowser = userAgent?.Browser,
|
||||
|
||||
@@ -20,9 +20,11 @@ namespace ThingsGateway.Admin.Application;
|
||||
public class AppService : IAppService
|
||||
{
|
||||
private readonly IUserAgentService UserAgentService;
|
||||
public AppService(IUserAgentService userAgentService)
|
||||
private readonly IClaimsPrincipalService ClaimsPrincipalService;
|
||||
public AppService(IUserAgentService userAgentService, IClaimsPrincipalService claimsPrincipalService)
|
||||
{
|
||||
UserAgentService = userAgentService;
|
||||
ClaimsPrincipalService = claimsPrincipalService;
|
||||
}
|
||||
public string GetReturnUrl(string returnUrl)
|
||||
{
|
||||
@@ -70,7 +72,7 @@ public class AppService : IAppService
|
||||
ExpiresUtc = diffTime,
|
||||
}).ConfigureAwait(false);
|
||||
}
|
||||
public ClaimsPrincipal? User => App.User;
|
||||
public ClaimsPrincipal? User => ClaimsPrincipalService.User;
|
||||
|
||||
public string? RemoteIpAddress => App.HttpContext?.GetRemoteIpAddressToIPv4();
|
||||
|
||||
|
||||
@@ -0,0 +1,24 @@
|
||||
//------------------------------------------------------------------------------
|
||||
// 此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
|
||||
// 此代码版权(除特别声明外的代码)归作者本人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 HybridClaimsPrincipalService : IClaimsPrincipalService
|
||||
{
|
||||
HybridAppService _hybridAppService;
|
||||
public HybridClaimsPrincipalService(HybridAppService hybridAppService)
|
||||
{
|
||||
_hybridAppService = hybridAppService;
|
||||
}
|
||||
public ClaimsPrincipal? User => _hybridAppService.User;
|
||||
|
||||
}
|
||||
@@ -12,8 +12,6 @@ using Microsoft.AspNetCore.Authentication.Cookies;
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.Extensions.Localization;
|
||||
|
||||
using SqlSugar;
|
||||
|
||||
using System.Security.Claims;
|
||||
|
||||
using ThingsGateway.DataEncryption;
|
||||
@@ -64,6 +62,10 @@ public class AuthService : IAuthService
|
||||
{
|
||||
throw Oops.Bah(appConfig.WebsitePolicy.CloseTip);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
string? password = input.Password;
|
||||
if (isCookie) //openApi登录不再需要解密
|
||||
{
|
||||
@@ -237,25 +239,20 @@ public class AuthService : IAuthService
|
||||
var logingEvent = new LoginEvent
|
||||
{
|
||||
Ip = _appService.RemoteIpAddress,
|
||||
Device = App.GetService<IAppService>().UserAgent?.Platform,
|
||||
Device = _appService.UserAgent?.Platform,
|
||||
Expire = expire,
|
||||
SysUser = sysUser,
|
||||
VerificatId = verificatId
|
||||
};
|
||||
await WriteTokenToCache(loginPolicy, logingEvent).ConfigureAwait(false);//写入verificat到cache
|
||||
await UpdateUser(logingEvent).ConfigureAwait(false);
|
||||
if (sysUser.Account == RoleConst.SuperAdmin)
|
||||
{
|
||||
var modules = (await _sysResourceService.GetAllAsync().ConfigureAwait(false)).Where(a => a.Category == ResourceCategoryEnum.Module).OrderBy(a => a.SortCode);//获取模块列表
|
||||
sysUser.ModuleList = modules.ToList();//模块列表赋值给用户
|
||||
}
|
||||
|
||||
//返回结果
|
||||
return new LoginOutput
|
||||
{
|
||||
VerificatId = verificatId,
|
||||
Account = sysUser.Account,
|
||||
Id = sysUser.Id,
|
||||
ModuleList = sysUser.ModuleList,
|
||||
AccessToken = accessToken,
|
||||
RefreshToken = refreshToken
|
||||
};
|
||||
|
||||
@@ -11,8 +11,6 @@
|
||||
using Microsoft.AspNetCore.Http.Connections.Features;
|
||||
using Microsoft.AspNetCore.SignalR;
|
||||
|
||||
using Yitter.IdGenerator;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
/// <summary>
|
||||
@@ -28,7 +26,7 @@ public class UserIdProvider : IUserIdProvider
|
||||
|
||||
if (UserId > 0)
|
||||
{
|
||||
return $"{UserId}{SysHub.Separate}{YitIdHelper.NextId()}";//返回用户ID
|
||||
return $"{UserId}{SysHub.Separate}{CommonUtils.GetSingleId()}";//返回用户ID
|
||||
}
|
||||
|
||||
return connection.ConnectionId;
|
||||
|
||||
@@ -466,7 +466,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService
|
||||
var exist = await GetUserByIdAsync(input.Id).ConfigureAwait(false);//获取用户信息
|
||||
if (exist != null)
|
||||
{
|
||||
var isSuperAdmin = exist.Account == RoleConst.SuperAdmin;//判断是否有超管
|
||||
var isSuperAdmin = exist.Id == RoleConst.SuperAdminId;//判断是否有超管
|
||||
if (isSuperAdmin && !UserManager.SuperAdmin)
|
||||
throw Oops.Bah(Localizer["CanotEditAdminUser"]);
|
||||
|
||||
@@ -540,7 +540,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService
|
||||
await CheckApiDataScopeAsync(sysUser.OrgId, sysUser.CreateUserId).ConfigureAwait(false);
|
||||
if (sysUser != null)
|
||||
{
|
||||
var isSuperAdmin = (sysUser.Account == RoleConst.SuperAdmin || input.GrantInfoList.Any(a => a == RoleConst.SuperAdminRoleId)) && !UserManager.SuperAdmin;//判断是否有超管
|
||||
var isSuperAdmin = (sysUser.Id == RoleConst.SuperAdminId || input.GrantInfoList.Any(a => a == RoleConst.SuperAdminRoleId)) && !UserManager.SuperAdmin;//判断是否有超管
|
||||
if (isSuperAdmin)
|
||||
throw Oops.Bah(Localizer["CanotGrantAdmin"]);
|
||||
|
||||
@@ -660,7 +660,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService
|
||||
public async Task<bool> DeleteUserAsync(IEnumerable<long> ids)
|
||||
{
|
||||
using var db = GetDB();
|
||||
var containsSuperAdmin = await db.Queryable<SysUser>().Where(it => it.Account == RoleConst.SuperAdmin && ids.Contains(it.Id)).AnyAsync().ConfigureAwait(false);//判断是否有超管
|
||||
var containsSuperAdmin = await db.Queryable<SysUser>().Where(it => it.Id == RoleConst.SuperAdminId && ids.Contains(it.Id)).AnyAsync().ConfigureAwait(false);//判断是否有超管
|
||||
if (containsSuperAdmin)
|
||||
throw Oops.Bah(Localizer["CanotDeleteAdminUser"]);
|
||||
if (ids.Contains(UserManager.UserId))
|
||||
@@ -899,7 +899,7 @@ internal sealed class SysUserService : BaseService<SysUser>, ISysUserService
|
||||
var tenantId = await _sysOrgService.GetTenantIdByOrgIdAsync(sysUser.OrgId, sysOrgList).ConfigureAwait(false);
|
||||
sysUser.TenantId = tenantId;
|
||||
|
||||
if (sysUser.Account == RoleConst.SuperAdmin)
|
||||
if (sysUser.Id == RoleConst.SuperAdminId)
|
||||
{
|
||||
var modules = (await _sysResourceService.GetAllAsync().ConfigureAwait(false)).Where(a => a.Category == ResourceCategoryEnum.Module).OrderBy(a => a.SortCode);
|
||||
sysUser.ModuleList = modules.ToList();//模块列表赋值给用户
|
||||
|
||||
@@ -13,8 +13,6 @@ using BootstrapBlazor.Components;
|
||||
using Microsoft.AspNetCore.Builder;
|
||||
using Microsoft.Extensions.DependencyInjection;
|
||||
|
||||
using SqlSugar;
|
||||
|
||||
using System.Reflection;
|
||||
|
||||
using ThingsGateway.UnifyResult;
|
||||
@@ -28,19 +26,12 @@ public class Startup : AppStartup
|
||||
{
|
||||
Directory.CreateDirectory("DB");
|
||||
|
||||
services.AddConfigurableOptions<SqlSugarOptions>();
|
||||
services.AddConfigurableOptions<AdminLogOptions>();
|
||||
services.AddConfigurableOptions<TenantOptions>();
|
||||
|
||||
services.AddSingleton(typeof(IDataService<>), typeof(BaseService<>));
|
||||
services.AddSingleton<ISugarAopService, SugarAopService>();
|
||||
services.AddSingleton<ISugarConfigAopService, SugarConfigAopService>();
|
||||
|
||||
services.AddSingleton<IUserAgentService, UserAgentService>();
|
||||
services.AddSingleton<IAppService, AppService>();
|
||||
|
||||
StaticConfig.EnableAllWhereIF = true;
|
||||
|
||||
services.AddConfigurableOptions<EmailOptions>();
|
||||
services.AddConfigurableOptions<HardwareInfoOptions>();
|
||||
|
||||
@@ -57,7 +48,6 @@ public class Startup : AppStartup
|
||||
|
||||
services.AddSingleton<IVerificatInfoService, VerificatInfoService>();
|
||||
services.AddSingleton<IUserCenterService, UserCenterService>();
|
||||
services.AddSingleton<ISugarAopService, SugarAopService>();
|
||||
services.AddSingleton<ISysDictService, SysDictService>();
|
||||
services.AddSingleton<ISysOperateLogService, SysOperateLogService>();
|
||||
services.AddSingleton<IRelationService, RelationService>();
|
||||
@@ -98,6 +88,21 @@ public class Startup : AppStartup
|
||||
CodeFirstUtils.CodeFirst(fullName!);//CodeFirst
|
||||
|
||||
|
||||
try
|
||||
{
|
||||
using var db = DbContext.GetDB<SysOperateLog>();
|
||||
if (db.CurrentConnectionConfig.DbType == SqlSugar.DbType.Sqlite)
|
||||
{
|
||||
if (!db.DbMaintenance.IsAnyIndex("idx_operatelog_optime_date"))
|
||||
{
|
||||
var indexsql = "CREATE INDEX idx_operatelog_optime_date ON sys_operatelog(strftime('%Y-%m-%d', OpTime));";
|
||||
db.Ado.ExecuteCommand(indexsql);
|
||||
}
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
|
||||
//删除在线用户统计
|
||||
var verificatInfoService = App.RootServices.GetService<IVerificatInfoService>();
|
||||
verificatInfoService.RemoveAllClientId();
|
||||
|
||||
@@ -18,9 +18,7 @@
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="BootstrapBlazor.TableExport" Version="9.2.5" />
|
||||
<PackageReference Include="Rougamo.Fody" Version="5.0.0" />
|
||||
<PackageReference Include="SqlSugarCore" Version="5.1.4.193" />
|
||||
</ItemGroup>
|
||||
<ItemGroup Condition=" '$(TargetFramework)' == 'net8.0' ">
|
||||
<PackageReference Include="Microsoft.Extensions.Hosting.Abstractions" Version="8.0.1" />
|
||||
@@ -49,6 +47,7 @@
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\ThingsGateway.Razor\ThingsGateway.Razor.csproj" />
|
||||
<ProjectReference Include="..\ThingsGateway.SqlSugar\ThingsGateway.SqlSugar.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
||||
@@ -40,6 +40,8 @@ public class BlazorAppContext
|
||||
/// </summary>
|
||||
public SysUser CurrentUser { get; private set; }
|
||||
|
||||
public string? Avatar => UserManager.AvatarUrl.IsNullOrEmpty() ? CurrentUser.Avatar : UserManager.AvatarUrl;
|
||||
|
||||
/// <summary>
|
||||
/// 用户个人菜单
|
||||
/// </summary>
|
||||
|
||||
@@ -38,7 +38,7 @@ public partial class UserCenterPage
|
||||
protected override async Task OnParametersSetAsync()
|
||||
{
|
||||
SysUser = AppContext.CurrentUser.Adapt<SysUser>();
|
||||
SysUser.Avatar = AppContext.CurrentUser.Avatar;
|
||||
SysUser.Avatar = AppContext.Avatar;
|
||||
WorkbenchInfo = (await UserCenterService.GetLoginWorkbenchAsync(SysUser.Id)).Adapt<WorkbenchInfo>();
|
||||
|
||||
await base.OnParametersSetAsync();
|
||||
|
||||
@@ -11,6 +11,7 @@
|
||||
|
||||
// nuget动态加载的程序集
|
||||
"SupportPackageNamePrefixs": [
|
||||
"ThingsGateway.SqlSugar",
|
||||
"ThingsGateway.Admin.Application",
|
||||
"ThingsGateway.Admin.Razor",
|
||||
"ThingsGateway.Razor"
|
||||
|
||||
@@ -11,6 +11,7 @@
|
||||
|
||||
// nuget动态加载的程序集
|
||||
"SupportPackageNamePrefixs": [
|
||||
"ThingsGateway.SqlSugar",
|
||||
"ThingsGateway.Admin.Application",
|
||||
"ThingsGateway.Admin.Razor",
|
||||
"ThingsGateway.Razor"
|
||||
|
||||
12
src/Admin/ThingsGateway.AdminServer/GlobalUsings.cs
Normal file
12
src/Admin/ThingsGateway.AdminServer/GlobalUsings.cs
Normal file
@@ -0,0 +1,12 @@
|
||||
// ------------------------------------------------------------------------
|
||||
// 版权信息
|
||||
// 版权归百小僧及百签科技(广东)有限公司所有。
|
||||
// 所有权利保留。
|
||||
// 官方网站:https://baiqian.com
|
||||
//
|
||||
// 许可证信息
|
||||
// 项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。
|
||||
// 许可证的完整文本可以在源代码树根目录中的 LICENSE-APACHE 和 LICENSE-MIT 文件中找到。
|
||||
// ------------------------------------------------------------------------
|
||||
|
||||
global using ThingsGateway.Admin.Application;
|
||||
@@ -18,7 +18,6 @@ using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.AspNetCore.Components;
|
||||
using Microsoft.Extensions.Localization;
|
||||
|
||||
using ThingsGateway.Admin.Application;
|
||||
using ThingsGateway.Admin.Razor;
|
||||
using ThingsGateway.Extension;
|
||||
|
||||
|
||||
@@ -13,8 +13,6 @@ using Microsoft.Extensions.Localization;
|
||||
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
|
||||
using ThingsGateway.Admin.Application;
|
||||
|
||||
namespace ThingsGateway.AdminServer;
|
||||
|
||||
public partial class AccessDenied
|
||||
|
||||
@@ -20,11 +20,11 @@ using Microsoft.Extensions.Options;
|
||||
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
|
||||
using ThingsGateway.Admin.Application;
|
||||
using ThingsGateway.DataEncryption;
|
||||
using ThingsGateway.NewLife.Extension;
|
||||
using ThingsGateway.Razor;
|
||||
|
||||
|
||||
namespace ThingsGateway.AdminServer;
|
||||
|
||||
public partial class Login
|
||||
|
||||
@@ -48,7 +48,7 @@
|
||||
<CultureChooser />
|
||||
</div>
|
||||
|
||||
<Logout ImageUrl="@(AppContext.CurrentUser.Avatar??$"{WebsiteConst.DefaultResourceUrl}images/defaultUser.svg")" ShowUserName=false DisplayName="@UserManager.UserAccount" UserName="@UserManager.VerificatId.ToString()" PrefixUserNameText=@AdminLocalizer["CurrentVerificat"]>
|
||||
<Logout ImageUrl="@(AppContext.Avatar??$"{WebsiteConst.DefaultResourceUrl}images/defaultUser.svg")" ShowUserName=false DisplayName="@UserManager.UserAccount" UserName="@UserManager.VerificatId.ToString()" PrefixUserNameText=@AdminLocalizer["CurrentVerificat"]>
|
||||
<LinkTemplate>
|
||||
<a href=@("/") class="h6"><i class="fa-solid fa-suitcase me-2"></i>@Localizer["系统首页"]</a>
|
||||
|
||||
|
||||
@@ -17,7 +17,6 @@ using Microsoft.Extensions.Options;
|
||||
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
|
||||
using ThingsGateway.Admin.Application;
|
||||
using ThingsGateway.Admin.Razor;
|
||||
using ThingsGateway.Razor;
|
||||
|
||||
@@ -27,38 +26,6 @@ public partial class MainLayout : IDisposable
|
||||
{
|
||||
[Inject]
|
||||
IStringLocalizer<ThingsGateway.Razor._Imports> RazorLocalizer { get; set; }
|
||||
private Task OnRefresh(ContextMenuItem item, object? context)
|
||||
{
|
||||
if (context is TabItem tabItem)
|
||||
{
|
||||
_tab.Refresh(tabItem);
|
||||
}
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
|
||||
private async Task OnClose(ContextMenuItem item, object? context)
|
||||
{
|
||||
if (context is TabItem tabItem)
|
||||
{
|
||||
await _tab.RemoveTab(tabItem);
|
||||
}
|
||||
}
|
||||
|
||||
private Task OnCloseOther(ContextMenuItem item, object? context)
|
||||
{
|
||||
if (context is TabItem tabItem)
|
||||
{
|
||||
_tab.ActiveTab(tabItem);
|
||||
}
|
||||
_tab.CloseOtherTabs();
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
|
||||
private Task OnCloseAll(ContextMenuItem item, object? context)
|
||||
{
|
||||
_tab.CloseAllTabs();
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
|
||||
#region 全局通知
|
||||
|
||||
|
||||
@@ -40,7 +40,8 @@ public class SingleFilePublish : ISingleFilePublish
|
||||
"ThingsGateway.NewLife.X",
|
||||
"ThingsGateway.Razor",
|
||||
"ThingsGateway.Admin.Razor" ,
|
||||
"ThingsGateway.Admin.Application"
|
||||
"ThingsGateway.Admin.Application",
|
||||
"ThingsGateway.SqlSugar",
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
@@ -25,11 +25,9 @@ using System.Text;
|
||||
using System.Text.Encodings.Web;
|
||||
using System.Text.Unicode;
|
||||
|
||||
using ThingsGateway.Admin.Application;
|
||||
using ThingsGateway.Admin.Razor;
|
||||
using ThingsGateway.Extension;
|
||||
using ThingsGateway.NewLife.Caching;
|
||||
using ThingsGateway.NewLife.Extension;
|
||||
|
||||
namespace ThingsGateway.AdminServer;
|
||||
|
||||
@@ -88,6 +86,7 @@ public class Startup : AppStartup
|
||||
}
|
||||
;
|
||||
|
||||
services.AddMvcFilter<RequestAuditFilter>();
|
||||
services.AddControllers()
|
||||
.AddNewtonsoftJson(options => SetNewtonsoftJsonSetting(options.SerializerSettings))
|
||||
//.AddXmlSerializerFormatters()
|
||||
@@ -160,7 +159,8 @@ public class Startup : AppStartup
|
||||
{
|
||||
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;
|
||||
};
|
||||
|
||||
@@ -237,7 +237,6 @@ public class Startup : AppStartup
|
||||
// logContext.Set(LoggingConst.Method, httpContext.Request.Method);//请求方法
|
||||
// });
|
||||
//});
|
||||
services.AddMvcFilter<RequestAuditFilter>();
|
||||
|
||||
//日志写入数据库配置
|
||||
services.AddDatabaseLogging<DatabaseLoggingWriter>(options =>
|
||||
@@ -369,12 +368,6 @@ public class Startup : AppStartup
|
||||
app.UseStaticFiles(new StaticFileOptions { ContentTypeProvider = provider });
|
||||
app.UseStaticFiles();
|
||||
|
||||
app.Use(async (context, next) =>
|
||||
{
|
||||
context.Response.Headers.Append("ThingsGateway", "ThingsGateway");
|
||||
await next().ConfigureAwait(false);
|
||||
});
|
||||
|
||||
|
||||
// 特定文件类型(文件后缀)处理
|
||||
var contentTypeProvider = GetFileExtensionContentTypeProvider();
|
||||
|
||||
@@ -71,13 +71,25 @@ public static class App
|
||||
/// </summary>
|
||||
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;
|
||||
|
||||
/// <summary>
|
||||
/// 本地化服务工厂
|
||||
/// </summary>
|
||||
public static IStringLocalizerFactory? StringLocalizerFactory
|
||||
|
||||
{
|
||||
get
|
||||
{
|
||||
|
||||
@@ -25,7 +25,7 @@ public static class ILoggerExtensions
|
||||
/// <param name="logger"></param>
|
||||
/// <param name="properties">建议使用 ConcurrentDictionary 类型</param>
|
||||
/// <returns></returns>
|
||||
public static IDisposable ScopeContext(this ILogger logger, IDictionary<object, object> properties)
|
||||
public static IDisposable ScopeContext(this ILogger logger, IDictionary<string, object> properties)
|
||||
{
|
||||
if (logger == null) throw new ArgumentNullException(nameof(logger));
|
||||
|
||||
|
||||
@@ -26,11 +26,11 @@ public static class LogContextExtensions
|
||||
/// <param name="key">键</param>
|
||||
/// <param name="value">值</param>
|
||||
/// <returns></returns>
|
||||
public static LogContext Set(this LogContext logContext, object key, object value)
|
||||
public static LogContext Set(this LogContext logContext, string key, object value)
|
||||
{
|
||||
if (logContext == null || key == null) return logContext;
|
||||
|
||||
logContext.Properties ??= new Dictionary<object, object>();
|
||||
logContext.Properties ??= new Dictionary<string, object>();
|
||||
|
||||
logContext.Properties.Remove(key);
|
||||
logContext.Properties.Add(key, value);
|
||||
@@ -43,7 +43,7 @@ public static class LogContextExtensions
|
||||
/// <param name="logContext"></param>
|
||||
/// <param name="properties"></param>
|
||||
/// <returns></returns>
|
||||
public static LogContext SetRange(this LogContext logContext, IDictionary<object, object> properties)
|
||||
public static LogContext SetRange(this LogContext logContext, IDictionary<string, object> properties)
|
||||
{
|
||||
if (logContext == null
|
||||
|| properties == null
|
||||
@@ -63,7 +63,7 @@ public static class LogContextExtensions
|
||||
/// <param name="logContext"></param>
|
||||
/// <param name="key">键</param>
|
||||
/// <returns></returns>
|
||||
public static object Get(this LogContext logContext, object key)
|
||||
public static object Get(this LogContext logContext, string key)
|
||||
{
|
||||
if (logContext == null
|
||||
|| key == null
|
||||
@@ -80,7 +80,7 @@ public static class LogContextExtensions
|
||||
/// <param name="logContext"></param>
|
||||
/// <param name="key">键</param>
|
||||
/// <returns></returns>
|
||||
public static T Get<T>(this LogContext logContext, object key)
|
||||
public static T Get<T>(this LogContext logContext, string key)
|
||||
{
|
||||
var value = logContext.Get(key);
|
||||
return value.ChangeType<T>();
|
||||
|
||||
@@ -84,7 +84,7 @@ public static class StringLoggingExtensions
|
||||
/// <param name="message"></param>
|
||||
/// <param name="properties">建议使用 ConcurrentDictionary 类型</param>
|
||||
/// <returns></returns>
|
||||
public static StringLoggingPart ScopeContext(this string message, IDictionary<object, object> properties)
|
||||
public static StringLoggingPart ScopeContext(this string message, IDictionary<string, object> properties)
|
||||
{
|
||||
return StringLoggingPart.Default().SetMessage(message).ScopeContext(properties);
|
||||
}
|
||||
|
||||
@@ -20,7 +20,7 @@ public sealed class LogContext : IDisposable
|
||||
/// <summary>
|
||||
/// 日志上下文数据
|
||||
/// </summary>
|
||||
public IDictionary<object, object> Properties { get; set; }
|
||||
public IDictionary<string, object> Properties { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// 原生日志上下文数据
|
||||
|
||||
@@ -96,7 +96,7 @@ public sealed partial class StringLoggingPart
|
||||
/// </summary>
|
||||
/// <param name="properties">建议使用 ConcurrentDictionary 类型</param>
|
||||
/// <returns></returns>
|
||||
public StringLoggingPart ScopeContext(IDictionary<object, object> properties)
|
||||
public StringLoggingPart ScopeContext(IDictionary<string, object> properties)
|
||||
{
|
||||
if (properties == null) return this;
|
||||
LogContext = new LogContext { Properties = properties };
|
||||
|
||||
@@ -59,7 +59,7 @@ public static class Log
|
||||
/// </summary>
|
||||
/// <param name="properties">建议使用 ConcurrentDictionary 类型</param>
|
||||
/// <returns></returns>
|
||||
public static (ILogger logger, IDisposable scope) ScopeContext(IDictionary<object, object> properties)
|
||||
public static (ILogger logger, IDisposable scope) ScopeContext(IDictionary<string, object> properties)
|
||||
{
|
||||
return GetLogger(StringLoggingPart.Default().ScopeContext(properties));
|
||||
}
|
||||
|
||||
@@ -39,7 +39,7 @@
|
||||
<PackageReference Include="System.Text.RegularExpressions" Version="4.3.1" />
|
||||
<PackageReference Include="Mapster" Version="7.4.0" />
|
||||
<PackageReference Include="MiniProfiler.AspNetCore.Mvc" Version="4.5.4" />
|
||||
<PackageReference Include="Swashbuckle.AspNetCore" Version="8.1.2" />
|
||||
<PackageReference Include="Swashbuckle.AspNetCore" Version="8.1.3" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup Condition=" '$(TargetFramework)' == 'net8.0' ">
|
||||
|
||||
@@ -349,7 +349,7 @@ public static class UnifyContext
|
||||
/// <param name="result"></param>
|
||||
/// <param name="data"></param>
|
||||
/// <returns></returns>
|
||||
internal static bool CheckVaildResult(IActionResult result, out object data)
|
||||
public static bool CheckVaildResult(IActionResult result, out object data)
|
||||
{
|
||||
data = default;
|
||||
|
||||
|
||||
@@ -37,7 +37,8 @@ public static class SystemTextJsonExtension
|
||||
{
|
||||
Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping,
|
||||
WriteIndented = true, // 缩进
|
||||
DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull // 忽略 null
|
||||
DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull, // 忽略 null
|
||||
NumberHandling = JsonNumberHandling.AllowNamedFloatingPointLiterals,
|
||||
};
|
||||
// 如有自定义Converter,这里添加
|
||||
// IndentedOptions.Converters.Add(new ByteArrayJsonConverter());
|
||||
@@ -50,7 +51,8 @@ public static class SystemTextJsonExtension
|
||||
{
|
||||
Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping,
|
||||
WriteIndented = false, // 不缩进
|
||||
DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull
|
||||
DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull,
|
||||
NumberHandling = JsonNumberHandling.AllowNamedFloatingPointLiterals,
|
||||
};
|
||||
NoneIndentedOptions.Converters.Add(new ByteArrayToNumberArrayConverterSystemTextJson());
|
||||
NoneIndentedOptions.Converters.Add(new JTokenSystemTextJsonConverter());
|
||||
|
||||
@@ -27,6 +27,13 @@ public class Startup : AppStartup
|
||||
{
|
||||
services.AddBootstrapBlazor(
|
||||
option => option.JSModuleVersion = Random.Shared.Next(10000).ToString()
|
||||
, jsonLocalizationOptions =>
|
||||
{
|
||||
jsonLocalizationOptions.DisableGetLocalizerFromResourceManager = true;
|
||||
jsonLocalizationOptions.DisableGetLocalizerFromService = true;
|
||||
jsonLocalizationOptions.IgnoreLocalizerMissing = true;
|
||||
jsonLocalizationOptions.UseKeyWhenValueIsNull = true;
|
||||
}
|
||||
);
|
||||
services.AddConfigurableOptions<MenuOptions>();
|
||||
services.ConfigureIconThemeOptions(options => options.ThemeKey = "fa");
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<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" />
|
||||
</ItemGroup>
|
||||
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
//下载文件
|
||||
export function blazor_downloadFile(url, fileName, dtoObject) {
|
||||
export async function blazor_downloadFile(url, fileName, dtoObject) {
|
||||
const params = new URLSearchParams();
|
||||
|
||||
// 将 dtoObject 的属性添加到 URLSearchParams 中
|
||||
@@ -12,97 +12,92 @@ export function blazor_downloadFile(url, fileName, dtoObject) {
|
||||
// 构建完整的 URL
|
||||
const fullUrl = `${url}?${params.toString()}`;
|
||||
|
||||
// 发起 fetch 请求
|
||||
fetch(fullUrl)
|
||||
.then(response => {
|
||||
// 获取响应头中的 content-disposition
|
||||
const dispositionHeader = response.headers.get('content-disposition');
|
||||
let resolvedFileName = fileName;
|
||||
try {
|
||||
// 发起 fetch 请求
|
||||
const response = await fetch(fullUrl);
|
||||
|
||||
if (dispositionHeader) {
|
||||
// 解析出文件名
|
||||
const match = /filename[^;=\n]*=((['"]).*?\2|[^;\n]*)/.exec(dispositionHeader);
|
||||
const serverFileName = match && match[1] ? match[1].replace(/['"]/g, '') : null;
|
||||
if (serverFileName) {
|
||||
resolvedFileName = serverFileName;
|
||||
}
|
||||
// 获取响应头中的 content-disposition
|
||||
const dispositionHeader = response.headers.get('content-disposition');
|
||||
let resolvedFileName = fileName;
|
||||
|
||||
if (dispositionHeader) {
|
||||
// 解析出文件名
|
||||
const match = /filename[^;=\n]*=((['"]).*?\2|[^;\n]*)/.exec(dispositionHeader);
|
||||
const serverFileName = match && match[1] ? match[1].replace(/['"]/g, '') : null;
|
||||
if (serverFileName) {
|
||||
resolvedFileName = serverFileName;
|
||||
}
|
||||
}
|
||||
|
||||
// 将响应转换为 blob 对象
|
||||
return response.blob().then(blob => {
|
||||
// 创建临时的文件 URL
|
||||
const fileUrl = window.URL.createObjectURL(blob);
|
||||
// 将响应转换为 blob 对象
|
||||
const blob = await response.blob();
|
||||
|
||||
// 创建一个 <a> 元素并设置下载链接和文件名
|
||||
const anchorElement = document.createElement('a');
|
||||
anchorElement.href = fileUrl;
|
||||
anchorElement.download = resolvedFileName;
|
||||
anchorElement.style.display = 'none';
|
||||
// 创建临时的文件 URL
|
||||
const fileUrl = window.URL.createObjectURL(blob);
|
||||
|
||||
// 将 <a> 元素添加到 body 中并触发下载
|
||||
document.body.appendChild(anchorElement);
|
||||
anchorElement.click();
|
||||
document.body.removeChild(anchorElement);
|
||||
// 创建一个 <a> 元素并设置下载链接和文件名
|
||||
const anchorElement = document.createElement('a');
|
||||
anchorElement.href = fileUrl;
|
||||
anchorElement.download = resolvedFileName;
|
||||
anchorElement.style.display = 'none';
|
||||
|
||||
// 撤销临时的文件 URL
|
||||
window.URL.revokeObjectURL(fileUrl);
|
||||
});
|
||||
})
|
||||
.catch(error => {
|
||||
console.error('DownFile error ', error);
|
||||
});
|
||||
// 将 <a> 元素添加到 body 中并触发下载
|
||||
document.body.appendChild(anchorElement);
|
||||
anchorElement.click();
|
||||
document.body.removeChild(anchorElement);
|
||||
|
||||
// 撤销临时的文件 URL
|
||||
window.URL.revokeObjectURL(fileUrl);
|
||||
|
||||
return true;
|
||||
} catch (error) {
|
||||
console.error('DownFile error ', error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
//下载文件
|
||||
export function postJson_downloadFile(url, fileName, jsonBody) {
|
||||
const params = new URLSearchParams();
|
||||
export async function postJson_downloadFile(url, fileName, jsonBody) {
|
||||
|
||||
|
||||
// 发起 fetch 请求
|
||||
fetch(url, {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json'
|
||||
},
|
||||
body: jsonBody
|
||||
})
|
||||
.then(response => {
|
||||
// 获取响应头中的 content-disposition
|
||||
const dispositionHeader = response.headers.get('content-disposition');
|
||||
let resolvedFileName = fileName;
|
||||
|
||||
if (dispositionHeader) {
|
||||
// 解析出文件名
|
||||
const match = /filename[^;=\n]*=((['"]).*?\2|[^;\n]*)/.exec(dispositionHeader);
|
||||
const serverFileName = match && match[1] ? match[1].replace(/['"]/g, '') : null;
|
||||
if (serverFileName) {
|
||||
resolvedFileName = serverFileName;
|
||||
}
|
||||
}
|
||||
|
||||
// 将响应转换为 blob 对象
|
||||
return response.blob().then(blob => {
|
||||
// 创建临时的文件 URL
|
||||
const fileUrl = window.URL.createObjectURL(blob);
|
||||
|
||||
// 创建一个 <a> 元素并设置下载链接和文件名
|
||||
const anchorElement = document.createElement('a');
|
||||
anchorElement.href = fileUrl;
|
||||
anchorElement.download = resolvedFileName;
|
||||
anchorElement.style.display = 'none';
|
||||
|
||||
// 将 <a> 元素添加到 body 中并触发下载
|
||||
document.body.appendChild(anchorElement);
|
||||
anchorElement.click();
|
||||
document.body.removeChild(anchorElement);
|
||||
|
||||
// 撤销临时的文件 URL
|
||||
window.URL.revokeObjectURL(fileUrl);
|
||||
});
|
||||
})
|
||||
.catch(error => {
|
||||
console.error('downfile error ', error);
|
||||
try {
|
||||
const response = await fetch(url, {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json'
|
||||
},
|
||||
body: jsonBody
|
||||
});
|
||||
|
||||
const dispositionHeader = response.headers.get('content-disposition');
|
||||
let resolvedFileName = fileName;
|
||||
|
||||
if (dispositionHeader) {
|
||||
const match = /filename[^;=\n]*=((['"]).*?\2|[^;\n]*)/.exec(dispositionHeader);
|
||||
const serverFileName = match && match[1] ? match[1].replace(/['"]/g, '') : null;
|
||||
if (serverFileName) {
|
||||
resolvedFileName = serverFileName;
|
||||
}
|
||||
}
|
||||
|
||||
const blob = await response.blob();
|
||||
const fileUrl = window.URL.createObjectURL(blob);
|
||||
|
||||
const anchorElement = document.createElement('a');
|
||||
anchorElement.href = fileUrl;
|
||||
anchorElement.download = resolvedFileName;
|
||||
anchorElement.style.display = 'none';
|
||||
|
||||
document.body.appendChild(anchorElement);
|
||||
anchorElement.click();
|
||||
document.body.removeChild(anchorElement);
|
||||
|
||||
window.URL.revokeObjectURL(fileUrl);
|
||||
|
||||
return true; // 唯一新增的返回值
|
||||
} catch (error) {
|
||||
console.error('downfile error ', error);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -31,6 +31,11 @@ public class ClaimConst
|
||||
/// </summary>
|
||||
public const string UserId = "UserId";
|
||||
|
||||
/// <summary>
|
||||
/// AvatarUrl
|
||||
/// </summary>
|
||||
public const string AvatarUrl = "AvatarUrl";
|
||||
|
||||
/// <summary>
|
||||
/// 验证Id
|
||||
/// </summary>
|
||||
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
|
||||
{
|
||||
private IAppService _appService;
|
||||
public SugarAopService(IAppService appService)
|
||||
private IClaimsPrincipalService _claimsPrincipalService;
|
||||
public SugarAopService(IClaimsPrincipalService appService)
|
||||
{
|
||||
_appService = appService;
|
||||
_claimsPrincipalService = appService;
|
||||
}
|
||||
/// <summary>
|
||||
/// Aop设置
|
||||
@@ -85,7 +85,7 @@ public class SugarAopService : ISugarAopService
|
||||
if (entityInfo.PropertyName == nameof(BaseEntity.CreateTime))
|
||||
entityInfo.SetValue(DateTime.Now);
|
||||
|
||||
if (_appService.User != null)
|
||||
if (_claimsPrincipalService.User != null)
|
||||
{
|
||||
//创建人
|
||||
if (entityInfo.PropertyName == nameof(BaseEntity.CreateUserId))
|
||||
@@ -103,7 +103,7 @@ public class SugarAopService : ISugarAopService
|
||||
if (entityInfo.PropertyName == nameof(BaseEntity.UpdateTime))
|
||||
entityInfo.SetValue(DateTime.Now);
|
||||
//更新人
|
||||
if (_appService.User != null)
|
||||
if (_claimsPrincipalService.User != null)
|
||||
{
|
||||
if (entityInfo.PropertyName == nameof(BaseEntity.UpdateUserId))
|
||||
entityInfo.SetValue(UserManager.UserId);
|
||||
@@ -117,6 +117,25 @@ public class SugarAopService : ISugarAopService
|
||||
db.Aop.DataExecuted = (value, entity) =>
|
||||
{
|
||||
};
|
||||
|
||||
|
||||
db.Aop.OnLogExecuted = (sql, pars) =>
|
||||
{
|
||||
//执行时间超过1秒
|
||||
if (db.Ado.SqlExecutionTime.TotalSeconds > 1)
|
||||
{
|
||||
//代码CS文件名
|
||||
var fileName = db.Ado.SqlStackTrace.FirstFileName;
|
||||
//代码行数
|
||||
var fileLine = db.Ado.SqlStackTrace.FirstLine;
|
||||
//方法名
|
||||
var FirstMethodName = db.Ado.SqlStackTrace.FirstMethodName;
|
||||
|
||||
DbContext.WriteLog($"{fileName}-{FirstMethodName}-{fileLine} 执行时间超过1秒");
|
||||
DbContext.WriteLogWithSql(UtilMethods.GetNativeSql(sql, pars));
|
||||
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
}
|
||||
@@ -37,7 +37,7 @@ public sealed class SqlSugarOption : ConnectionConfig
|
||||
/// <summary>
|
||||
/// 是否控制台显示Sql语句
|
||||
/// </summary>
|
||||
public bool IsShowSql { get; set; }
|
||||
public bool? IsShowSql { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// 更新数据
|
||||
44
src/Admin/ThingsGateway.SqlSugar/Startup.cs
Normal file
44
src/Admin/ThingsGateway.SqlSugar/Startup.cs
Normal file
@@ -0,0 +1,44 @@
|
||||
//------------------------------------------------------------------------------
|
||||
// 此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
|
||||
// 此代码版权(除特别声明外的代码)归作者本人Diego所有
|
||||
// 源代码使用协议遵循本仓库的开源协议及附加协议
|
||||
// Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
|
||||
// Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
|
||||
// 使用文档:https://thingsgateway.cn/
|
||||
// QQ群:605534569
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
using BootstrapBlazor.Components;
|
||||
|
||||
using Microsoft.AspNetCore.Builder;
|
||||
using Microsoft.Extensions.DependencyInjection;
|
||||
|
||||
using SqlSugar;
|
||||
|
||||
namespace ThingsGateway.Admin.Application;
|
||||
|
||||
[AppStartup(1000000000)]
|
||||
public class Startup : AppStartup
|
||||
{
|
||||
public void Configure(IServiceCollection services)
|
||||
{
|
||||
services.AddConfigurableOptions<SqlSugarOptions>();
|
||||
|
||||
services.AddSingleton(typeof(IDataService<>), typeof(BaseService<>));
|
||||
services.AddSingleton<ISugarAopService, SugarAopService>();
|
||||
services.AddSingleton<ISugarConfigAopService, SugarConfigAopService>();
|
||||
|
||||
services.AddSingleton<IClaimsPrincipalService, ClaimsPrincipalService>();
|
||||
|
||||
StaticConfig.EnableAllWhereIF = true;
|
||||
|
||||
services.AddSingleton<ISugarAopService, SugarAopService>();
|
||||
|
||||
}
|
||||
|
||||
public void Use(IApplicationBuilder applicationBuilder)
|
||||
{
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
@@ -17,33 +17,39 @@ namespace ThingsGateway.Admin.Application;
|
||||
/// </summary>
|
||||
public static class UserManager
|
||||
{
|
||||
private static readonly IAppService _appService;
|
||||
private static readonly IClaimsPrincipalService _claimsPrincipalService;
|
||||
static UserManager()
|
||||
{
|
||||
_appService = App.RootServices.GetService<IAppService>();
|
||||
_claimsPrincipalService = App.RootServices.GetService<IClaimsPrincipalService>();
|
||||
}
|
||||
/// <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>
|
||||
public static string UserAccount => _appService.User?.FindFirst(ClaimConst.Account)?.Value;
|
||||
public static string UserAccount => _claimsPrincipalService.User?.FindFirst(ClaimConst.Account)?.Value;
|
||||
|
||||
/// <summary>
|
||||
/// AvatarUrl
|
||||
/// </summary>
|
||||
public static string AvatarUrl => (_claimsPrincipalService.User?.FindFirst(ClaimConst.AvatarUrl)?.Value);
|
||||
|
||||
/// <summary>
|
||||
/// 当前用户Id
|
||||
/// </summary>
|
||||
public static long UserId => (_appService.User?.FindFirst(ClaimConst.UserId)?.Value).ToLong();
|
||||
public static long UserId => (_claimsPrincipalService.User?.FindFirst(ClaimConst.UserId)?.Value).ToLong();
|
||||
|
||||
/// <summary>
|
||||
/// 当前验证Id
|
||||
/// </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.195" />
|
||||
<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>
|
||||
|
||||
<PropertyGroup>
|
||||
<PluginVersion>10.6.21</PluginVersion>
|
||||
<ProPluginVersion>10.6.21</ProPluginVersion>
|
||||
<AuthenticationVersion>2.1.7</AuthenticationVersion>
|
||||
<PluginVersion>10.7.20</PluginVersion>
|
||||
<ProPluginVersion>10.7.20</ProPluginVersion>
|
||||
<AuthenticationVersion>2.4.0</AuthenticationVersion>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
|
||||
@@ -6,7 +6,9 @@
|
||||
@using BootstrapBlazor.Components
|
||||
@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>
|
||||
<div class="flex-fill">
|
||||
</div>
|
||||
@@ -36,7 +38,7 @@
|
||||
|
||||
</HeaderTemplate>
|
||||
<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>
|
||||
<ItemContent>
|
||||
@* <Tooltip Placement="Placement.Bottom" Title=@itemMessage.Message.Substring(0, Math.Min(itemMessage.Message.Length, 500))> *@
|
||||
@@ -56,4 +58,4 @@
|
||||
</Card>
|
||||
|
||||
|
||||
|
||||
</div>
|
||||
|
||||
@@ -33,13 +33,11 @@ public partial class LogConsole : IDisposable
|
||||
[Parameter]
|
||||
public EventCallback<LogLevel> LogLevelChanged { get; set; }
|
||||
|
||||
[Parameter]
|
||||
public string CardStyle { get; set; } = "height: 100%;";
|
||||
[Parameter]
|
||||
public string HeaderText { get; set; } = "Log";
|
||||
|
||||
[Parameter]
|
||||
public string HeightString { get; set; } = "calc(100% - 50px)";
|
||||
public string HeightString { get; set; } = "calc(100% - 300px)";
|
||||
|
||||
[Parameter, EditorRequired]
|
||||
public string LogPath { get; set; }
|
||||
|
||||
@@ -28,11 +28,12 @@ public class DtuPlugin : PluginBase, ITcpReceivingPlugin
|
||||
set
|
||||
{
|
||||
_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 ArraySegment<byte> HeartbeatByte;
|
||||
private ArraySegment<byte> HeartbeatByte = new();
|
||||
|
||||
/// <inheritdoc/>
|
||||
public async Task OnTcpReceiving(ITcpSession client, ByteBlockEventArgs e)
|
||||
|
||||
@@ -61,14 +61,14 @@ internal sealed class HeartbeatAndReceivePlugin : PluginBase, ITcpConnectedPlugi
|
||||
{
|
||||
return;//此处可判断,如果为服务器,则不用使用心跳。
|
||||
}
|
||||
|
||||
if (HeartbeatTime > 0)
|
||||
SendHeartbeat = true;
|
||||
HeartbeatTime = Math.Max(HeartbeatTime, 1000);
|
||||
|
||||
if (DtuId.IsNullOrWhiteSpace()) return;
|
||||
|
||||
if (client is ITcpClient tcpClient)
|
||||
{
|
||||
SendHeartbeat = true;
|
||||
await tcpClient.SendAsync(DtuIdByte).ConfigureAwait(false);
|
||||
|
||||
if (Task == null)
|
||||
|
||||
@@ -47,7 +47,7 @@ public static class PluginUtil
|
||||
Action<IPluginManager> action = a => { };
|
||||
|
||||
action += GetTcpServicePlugin(channelOptions);
|
||||
if (!channelOptions.Heartbeat.IsNullOrWhiteSpace())
|
||||
//if (!channelOptions.Heartbeat.IsNullOrWhiteSpace())
|
||||
{
|
||||
action += a =>
|
||||
{
|
||||
|
||||
@@ -8,17 +8,13 @@
|
||||
// QQ群:605534569
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
using System.Buffers;
|
||||
using System.Text;
|
||||
|
||||
using ThingsGateway.NewLife.Caching;
|
||||
|
||||
namespace ThingsGateway.Foundation;
|
||||
|
||||
public class LogDataCache
|
||||
{
|
||||
public List<LogData> LogDatas { get; set; }
|
||||
public long Length { get; set; }
|
||||
}
|
||||
/// <summary>
|
||||
/// 日志数据
|
||||
/// </summary>
|
||||
@@ -47,8 +43,19 @@ public class LogData
|
||||
|
||||
|
||||
/// <summary>日志文本文件倒序读取</summary>
|
||||
|
||||
public class LogDataCache
|
||||
{
|
||||
public List<LogData> LogDatas { get; set; }
|
||||
public long Length { get; set; }
|
||||
}
|
||||
|
||||
/// <summary>高性能日志文件读取器(支持倒序读取)</summary>
|
||||
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>
|
||||
@@ -86,159 +93,167 @@ public class TextFileReader
|
||||
return result;
|
||||
}
|
||||
|
||||
static MemoryCache _cache = new() { Expire = 30 };
|
||||
|
||||
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
|
||||
{
|
||||
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;
|
||||
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)
|
||||
if (cachedData != null && cachedData.Length == length)
|
||||
{
|
||||
result.Content = dataCache.LogDatas;
|
||||
result.OperCode = 0; // 操作状态设为成功
|
||||
return result; // 返回解析结果
|
||||
return new OperResult<List<LogData>>() { Content = cachedData.LogDatas };
|
||||
}
|
||||
|
||||
if (ps <= 0) // 如果起始位置小于等于0,将起始位置设置为文件长度
|
||||
ps = length - 1;
|
||||
|
||||
// 循环读取指定行数的文本内容
|
||||
for (int i = 0; i < lineCount; i++)
|
||||
else
|
||||
{
|
||||
ps = InverseReadRow(fs, ps, out var value); // 使用逆序读取
|
||||
txt.Add(value);
|
||||
if (ps <= 0) // 如果已经读取到文件开头则跳出循环
|
||||
break;
|
||||
_cache.Remove(cacheKey);
|
||||
}
|
||||
}
|
||||
|
||||
// 使用单次 LINQ 操作进行过滤和解析
|
||||
result.Content = 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();
|
||||
using var fs = new FileStream(file, FileMode.Open, FileAccess.Read, FileShare.ReadWrite, 4096, FileOptions.SequentialScan);
|
||||
var result = ReadLogsInverse(fs, lineCount, fileInfo.Length);
|
||||
|
||||
result.OperCode = 0; // 操作状态设为成功
|
||||
var data = _cache.Set<LogDataCache>(key, new LogDataCache() { Length = length, LogDatas = result.Content });
|
||||
_cache.Set(cacheKey, new LogDataCache
|
||||
{
|
||||
LogDatas = result,
|
||||
Length = fileInfo.Length,
|
||||
});
|
||||
|
||||
return result; // 返回解析结果
|
||||
return new OperResult<List<LogData>>() { Content = result };
|
||||
}
|
||||
catch (Exception ex) // 捕获异常
|
||||
catch (Exception ex)
|
||||
{
|
||||
result = new(ex); // 创建包含异常信息的结果对象
|
||||
return result; // 返回异常结果
|
||||
return new OperResult<List<LogData>>(ex);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
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)
|
||||
{
|
||||
byte n = 0xD; // 换行符
|
||||
byte a = 0xA; // 回车符
|
||||
byte n = 0xD;
|
||||
byte a = 0xA;
|
||||
value = string.Empty;
|
||||
if (fs.Length == 0) return 0; // 若文件长度为0,则直接返回0作为新的位置
|
||||
|
||||
if (fs.Length == 0) return 0;
|
||||
|
||||
var newPos = position;
|
||||
List<byte> buffer = new List<byte>(maxRead); // 缓存读取的数据
|
||||
byte[] buffer = _bytePool.Rent(maxRead); // 从池中租借字节数组
|
||||
int index = 0;
|
||||
|
||||
try
|
||||
{
|
||||
var readLength = 0;
|
||||
|
||||
while (true) // 循环读取一行数据,TextFileLogger.Separator行判定
|
||||
while (true)
|
||||
{
|
||||
readLength++;
|
||||
if (newPos <= 0)
|
||||
newPos = 0;
|
||||
|
||||
fs.Position = newPos;
|
||||
int byteRead = fs.ReadByte();
|
||||
|
||||
if (byteRead == -1) break; // 到达文件开头时跳出循环
|
||||
if (byteRead == -1) break;
|
||||
|
||||
buffer.Add((byte)byteRead);
|
||||
|
||||
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) // 超过最大字节数限制时丢弃数据
|
||||
if (index >= maxRead)
|
||||
{
|
||||
newPos = -1;
|
||||
return newPos;
|
||||
}
|
||||
|
||||
buffer[index++] = (byte)byteRead;
|
||||
|
||||
if (byteRead == n || byteRead == a)
|
||||
{
|
||||
if (MatchSeparator(buffer, index))
|
||||
{
|
||||
index -= TextFileLogger.SeparatorBytes.Length;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
newPos--;
|
||||
if (newPos <= -1)
|
||||
break;
|
||||
}
|
||||
|
||||
if (buffer.Count >= 10)
|
||||
if (index >= 10)
|
||||
{
|
||||
buffer.Reverse();
|
||||
value = Encoding.UTF8.GetString(buffer.ToArray()); // 转换为字符串
|
||||
Array.Reverse(buffer, 0, index); // 倒序
|
||||
value = Encoding.UTF8.GetString(buffer, 0, index);
|
||||
}
|
||||
|
||||
return newPos; // 返回新的读取位置
|
||||
return newPos;
|
||||
}
|
||||
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;
|
||||
}
|
||||
var pos = arr.Count - 1;
|
||||
|
||||
int pos = length - 1;
|
||||
for (int i = 0; i < TextFileLogger.SeparatorBytes.Length; i++)
|
||||
{
|
||||
if (arr[pos] != TextFileLogger.SeparatorBytes[i])
|
||||
{
|
||||
return false;
|
||||
}
|
||||
pos--;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
private static List<string> ParseCSV(string data)
|
||||
{
|
||||
List<string> items = new List<string>();
|
||||
|
||||
@@ -10,8 +10,8 @@
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.Extensions.Localization.Abstractions" Version="9.0.5" />
|
||||
<PackageReference Include="TouchSocket" Version="3.1.5" />
|
||||
<PackageReference Include="TouchSocket.SerialPorts" Version="3.1.5" />
|
||||
<PackageReference Include="TouchSocket" Version="3.1.6" />
|
||||
<PackageReference Include="TouchSocket.SerialPorts" Version="3.1.6" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
|
||||
@@ -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;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -90,6 +90,7 @@ public class RuntimeInfoController : ControllerBase
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
[HttpPost("checkRealAlarm")]
|
||||
[RequestAudit]
|
||||
[DisplayName("确认实时报警")]
|
||||
public async Task CheckRealAlarm(long variableId)
|
||||
{
|
||||
|
||||
@@ -12,6 +12,8 @@ using BootstrapBlazor.Components;
|
||||
|
||||
using Microsoft.Extensions.Localization;
|
||||
|
||||
using System.Text;
|
||||
|
||||
using ThingsGateway.NewLife;
|
||||
using ThingsGateway.NewLife.Threading;
|
||||
using ThingsGateway.Razor;
|
||||
@@ -345,14 +347,36 @@ public abstract class DriverBase : DisposableObject, IDriver
|
||||
|
||||
#region 插件重写
|
||||
|
||||
public virtual bool Authentication()
|
||||
public virtual bool GetAuthentication(out DateTime? expireTime)
|
||||
{
|
||||
expireTime = null;
|
||||
return true;
|
||||
}
|
||||
|
||||
public string GetAuthString()
|
||||
{
|
||||
return PluginServiceUtil.IsEducation(GetType()) ? ThingsGateway.Authentication.ProAuthentication.TryGetAuthorizeInfo(out _) ? Localizer["Authorized"] : Localizer["Unauthorized"] : string.Empty;
|
||||
if (PluginServiceUtil.IsEducation(GetType()))
|
||||
{
|
||||
StringBuilder stringBuilder = new();
|
||||
var ret = GetAuthentication(out var expireTime);
|
||||
if (ret)
|
||||
{
|
||||
stringBuilder.Append(Localizer["Authorized"]);
|
||||
}
|
||||
else
|
||||
{
|
||||
stringBuilder.Append(Localizer["Unauthorized"]);
|
||||
}
|
||||
|
||||
stringBuilder.Append(" ");
|
||||
if (expireTime.HasValue && (DateTime.Now - expireTime.Value).TotalHours > -72)
|
||||
{
|
||||
stringBuilder.Append(Localizer["ExpireTime", expireTime.Value.ToString("yyyy-MM-dd HH")]);
|
||||
}
|
||||
|
||||
return stringBuilder.ToString();
|
||||
}
|
||||
return string.Empty;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
||||
@@ -43,6 +43,6 @@ namespace ThingsGateway.Gateway.Application
|
||||
Task SetLogAsync(LogLevel? logLevel = null, bool upDataBase = true);
|
||||
Task AfterVariablesChangedAsync(CancellationToken cancellationToken);
|
||||
string GetAuthString();
|
||||
bool Authentication();
|
||||
bool GetAuthentication(out DateTime? expireTime);
|
||||
}
|
||||
}
|
||||
@@ -8,6 +8,8 @@
|
||||
// QQ群:605534569
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
using BootstrapBlazor.Components;
|
||||
|
||||
using Mapster;
|
||||
|
||||
using System.Collections.Concurrent;
|
||||
@@ -205,6 +207,30 @@ public static class GlobalData
|
||||
|
||||
#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;
|
||||
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();
|
||||
}
|
||||
}
|
||||
@@ -234,6 +234,7 @@
|
||||
"ThingsGateway.Gateway.Application.DriverBase": {
|
||||
"Authorized": "Authorized",
|
||||
"Unauthorized": "Unauthorized",
|
||||
"ExpireTime": "ExpireTime",
|
||||
"DeviceTaskStart": "Device {0} thread started",
|
||||
"DeviceTaskStartTimeout": "Device {0} thread start timeout {1} s",
|
||||
"DeviceTaskStop": "Device {0} thread stopped",
|
||||
|
||||
@@ -240,6 +240,7 @@
|
||||
"ThingsGateway.Gateway.Application.DriverBase": {
|
||||
"Authorized": "已授权",
|
||||
"Unauthorized": "未授权",
|
||||
"ExpireTime": "过期时间",
|
||||
"DeviceTaskStart": "设备 {0} 线程开始",
|
||||
"DeviceTaskStartTimeout": "设备 {0} 线程启动超时 {1} s",
|
||||
"DeviceTaskStop": "设备 {0} 线程停止",
|
||||
|
||||
@@ -241,7 +241,7 @@ public class DeviceRuntime : Device, IDisposable
|
||||
|
||||
ChannelRuntime = channelRuntime;
|
||||
ChannelRuntime?.DeviceRuntimes?.TryRemove(Id, out _);
|
||||
ChannelRuntime.DeviceRuntimes.TryAdd(Id, this);
|
||||
ChannelRuntime?.DeviceRuntimes?.TryAdd(Id, this);
|
||||
|
||||
GlobalData.IdDevices.TryRemove(Id, out _);
|
||||
GlobalData.IdDevices.TryAdd(Id, this);
|
||||
|
||||
@@ -17,8 +17,6 @@ using Microsoft.Extensions.Logging;
|
||||
|
||||
using ThingsGateway.NewLife;
|
||||
|
||||
using TouchSocket.Core;
|
||||
|
||||
namespace ThingsGateway.Gateway.Application;
|
||||
|
||||
public class ChannelRuntimeService : IChannelRuntimeService
|
||||
@@ -199,9 +197,7 @@ public class ChannelRuntimeService : IChannelRuntimeService
|
||||
|
||||
public async Task RestartChannelAsync(IEnumerable<ChannelRuntime> oldChannelRuntimes)
|
||||
{
|
||||
oldChannelRuntimes.SelectMany(a => a.DeviceRuntimes.SelectMany(a => a.Value.VariableRuntimes)).ParallelForEach(a => a.Value.SafeDispose());
|
||||
oldChannelRuntimes.SelectMany(a => a.DeviceRuntimes).ParallelForEach(a => a.Value.SafeDispose());
|
||||
oldChannelRuntimes.ParallelForEach(a => a.SafeDispose());
|
||||
RuntimeServiceHelper.RemoveOldChannelRuntimes(oldChannelRuntimes);
|
||||
var ids = oldChannelRuntimes.Select(a => a.Id).ToHashSet();
|
||||
try
|
||||
{
|
||||
@@ -229,4 +225,5 @@ public class ChannelRuntimeService : IChannelRuntimeService
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
@@ -33,16 +33,6 @@ namespace ThingsGateway.Gateway.Application;
|
||||
|
||||
internal sealed class ChannelService : BaseService<Channel>, IChannelService
|
||||
{
|
||||
private readonly IDispatchService<Channel> _dispatchService;
|
||||
|
||||
/// <inheritdoc cref="IChannelService"/>
|
||||
public ChannelService(
|
||||
IDispatchService<Channel>? dispatchService
|
||||
)
|
||||
{
|
||||
_dispatchService = dispatchService;
|
||||
}
|
||||
|
||||
#region CURD
|
||||
|
||||
/// <inheritdoc/>
|
||||
@@ -181,7 +171,6 @@ internal sealed class ChannelService : BaseService<Channel>, IChannelService
|
||||
public void DeleteChannelFromCache()
|
||||
{
|
||||
App.CacheService.Remove(ThingsGatewayCacheConst.Cache_Channel);//删除通道缓存
|
||||
_dispatchService.Dispatch(new());
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
||||
@@ -35,15 +35,11 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService
|
||||
{
|
||||
private readonly IChannelService _channelService;
|
||||
private readonly IPluginService _pluginService;
|
||||
private readonly IDispatchService<Device> _dispatchService;
|
||||
|
||||
public DeviceService(
|
||||
IDispatchService<Device> dispatchService
|
||||
)
|
||||
public DeviceService()
|
||||
{
|
||||
_channelService = App.RootServices.GetRequiredService<IChannelService>();
|
||||
_pluginService = App.RootServices.GetRequiredService<IPluginService>();
|
||||
_dispatchService = dispatchService;
|
||||
}
|
||||
|
||||
|
||||
@@ -195,7 +191,6 @@ internal sealed class DeviceService : BaseService<Device>, IDeviceService
|
||||
public void DeleteDeviceFromCache()
|
||||
{
|
||||
App.CacheService.Remove(ThingsGatewayCacheConst.Cache_Device);//删除设备缓存
|
||||
_dispatchService.Dispatch(new());
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
||||
@@ -26,28 +26,53 @@ internal sealed class GatewayExportService : IGatewayExportService
|
||||
|
||||
private IJSRuntime JSRuntime { get; set; }
|
||||
|
||||
public async Task OnChannelExport(ExportFilter exportFilter)
|
||||
public async Task<bool> OnChannelExport(ExportFilter exportFilter)
|
||||
{
|
||||
await using var ajaxJS = await JSRuntime.InvokeAsync<IJSObjectReference>("import", $"/_content/ThingsGateway.Razor/js/downloadFile.js");
|
||||
string url = "api/gatewayExport/channel";
|
||||
string fileName = $"{DateTime.Now.ToFileDateTimeFormat()}.xlsx";
|
||||
await ajaxJS.InvokeVoidAsync("postJson_downloadFile", url, fileName, exportFilter.ToJsonString());
|
||||
try
|
||||
{
|
||||
|
||||
await using var ajaxJS = await JSRuntime.InvokeAsync<IJSObjectReference>("import", $"/_content/ThingsGateway.Razor/js/downloadFile.js");
|
||||
string url = "api/gatewayExport/channel";
|
||||
string fileName = $"{DateTime.Now.ToFileDateTimeFormat()}.xlsx";
|
||||
return await ajaxJS.InvokeAsync<bool>("postJson_downloadFile", url, fileName, exportFilter.ToJsonString());
|
||||
|
||||
}
|
||||
catch
|
||||
{
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public async Task OnDeviceExport(ExportFilter exportFilter)
|
||||
public async Task<bool> OnDeviceExport(ExportFilter exportFilter)
|
||||
{
|
||||
await using var ajaxJS = await JSRuntime.InvokeAsync<IJSObjectReference>("import", $"/_content/ThingsGateway.Razor/js/downloadFile.js");
|
||||
string url = "api/gatewayExport/device";
|
||||
string fileName = $"{DateTime.Now.ToFileDateTimeFormat()}.xlsx";
|
||||
await ajaxJS.InvokeVoidAsync("postJson_downloadFile", url, fileName, exportFilter.ToJsonString());
|
||||
try
|
||||
{
|
||||
|
||||
|
||||
await using var ajaxJS = await JSRuntime.InvokeAsync<IJSObjectReference>("import", $"/_content/ThingsGateway.Razor/js/downloadFile.js");
|
||||
string url = "api/gatewayExport/device";
|
||||
string fileName = $"{DateTime.Now.ToFileDateTimeFormat()}.xlsx";
|
||||
return await ajaxJS.InvokeAsync<bool>("postJson_downloadFile", url, fileName, exportFilter.ToJsonString());
|
||||
}
|
||||
catch
|
||||
{
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public async Task OnVariableExport(ExportFilter exportFilter)
|
||||
public async Task<bool> OnVariableExport(ExportFilter exportFilter)
|
||||
{
|
||||
await using var ajaxJS = await JSRuntime.InvokeAsync<IJSObjectReference>("import", $"/_content/ThingsGateway.Razor/js/downloadFile.js");
|
||||
string url = "api/gatewayExport/variable";
|
||||
string fileName = $"{DateTime.Now.ToFileDateTimeFormat()}.xlsx";
|
||||
await ajaxJS.InvokeVoidAsync("postJson_downloadFile", url, fileName, exportFilter.ToJsonString());
|
||||
try
|
||||
{
|
||||
await using var ajaxJS = await JSRuntime.InvokeAsync<IJSObjectReference>("import", $"/_content/ThingsGateway.Razor/js/downloadFile.js");
|
||||
string url = "api/gatewayExport/variable";
|
||||
string fileName = $"{DateTime.Now.ToFileDateTimeFormat()}.xlsx";
|
||||
return await ajaxJS.InvokeAsync<bool>("postJson_downloadFile", url, fileName, exportFilter.ToJsonString());
|
||||
}
|
||||
catch
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
@@ -35,18 +35,30 @@ public sealed class HybridGatewayExportService : IGatewayExportService
|
||||
|
||||
}
|
||||
|
||||
public async Task OnChannelExport(ExportFilter exportFilter)
|
||||
public async Task<bool> OnChannelExport(ExportFilter exportFilter)
|
||||
{
|
||||
exportFilter.QueryPageOptions.IsPage = false;
|
||||
exportFilter.QueryPageOptions.IsVirtualScroll = false;
|
||||
try
|
||||
{
|
||||
|
||||
var sheets = await _channelService.ExportChannelAsync(exportFilter).ConfigureAwait(false);
|
||||
var path = await _importExportService.CreateFileAsync<Device>(sheets, "Channel", false).ConfigureAwait(false);
|
||||
|
||||
Open(path);
|
||||
exportFilter.QueryPageOptions.IsPage = false;
|
||||
exportFilter.QueryPageOptions.IsVirtualScroll = false;
|
||||
|
||||
var sheets = await _channelService.ExportChannelAsync(exportFilter).ConfigureAwait(false);
|
||||
var path = await _importExportService.CreateFileAsync<Device>(sheets, "Channel", false).ConfigureAwait(false);
|
||||
|
||||
Open(path);
|
||||
return true;
|
||||
|
||||
}
|
||||
catch
|
||||
{
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private static void Open(string path)
|
||||
private static bool Open(string path)
|
||||
{
|
||||
path = System.IO.Path.GetDirectoryName(path); // Ensure the path is absolute
|
||||
|
||||
@@ -63,25 +75,47 @@ public sealed class HybridGatewayExportService : IGatewayExportService
|
||||
{
|
||||
System.Diagnostics.Process.Start("open", path);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public async Task OnDeviceExport(ExportFilter exportFilter)
|
||||
public async Task<bool> OnDeviceExport(ExportFilter exportFilter)
|
||||
{
|
||||
exportFilter.QueryPageOptions.IsPage = false;
|
||||
exportFilter.QueryPageOptions.IsVirtualScroll = false;
|
||||
var sheets = await _deviceService.ExportDeviceAsync(exportFilter).ConfigureAwait(false);
|
||||
var path = await _importExportService.CreateFileAsync<Device>(sheets, "Device", false).ConfigureAwait(false);
|
||||
Open(path);
|
||||
try
|
||||
{
|
||||
|
||||
exportFilter.QueryPageOptions.IsPage = false;
|
||||
exportFilter.QueryPageOptions.IsVirtualScroll = false;
|
||||
var sheets = await _deviceService.ExportDeviceAsync(exportFilter).ConfigureAwait(false);
|
||||
var path = await _importExportService.CreateFileAsync<Device>(sheets, "Device", false).ConfigureAwait(false);
|
||||
Open(path);
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
catch
|
||||
{
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public async Task OnVariableExport(ExportFilter exportFilter)
|
||||
public async Task<bool> OnVariableExport(ExportFilter exportFilter)
|
||||
{
|
||||
exportFilter.QueryPageOptions.IsPage = false;
|
||||
exportFilter.QueryPageOptions.IsVirtualScroll = false;
|
||||
var sheets = await _variableService.ExportVariableAsync(exportFilter).ConfigureAwait(false);
|
||||
var path = await _importExportService.CreateFileAsync<Variable>(sheets, "Variable", false).ConfigureAwait(false);
|
||||
Open(path);
|
||||
try
|
||||
{
|
||||
|
||||
exportFilter.QueryPageOptions.IsPage = false;
|
||||
exportFilter.QueryPageOptions.IsVirtualScroll = false;
|
||||
var sheets = await _variableService.ExportVariableAsync(exportFilter).ConfigureAwait(false);
|
||||
var path = await _importExportService.CreateFileAsync<Variable>(sheets, "Variable", false).ConfigureAwait(false);
|
||||
Open(path);
|
||||
return true;
|
||||
}
|
||||
catch
|
||||
{
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -12,7 +12,7 @@ namespace ThingsGateway.Gateway.Application;
|
||||
|
||||
public interface IGatewayExportService
|
||||
{
|
||||
Task OnChannelExport(ExportFilter exportFilter);
|
||||
Task OnDeviceExport(ExportFilter exportFilter);
|
||||
Task OnVariableExport(ExportFilter exportFilter);
|
||||
Task<bool> OnChannelExport(ExportFilter exportFilter);
|
||||
Task<bool> OnDeviceExport(ExportFilter exportFilter);
|
||||
Task<bool> OnVariableExport(ExportFilter exportFilter);
|
||||
}
|
||||
|
||||
@@ -8,8 +8,6 @@
|
||||
// QQ群:605534569
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
using BootstrapBlazor.Components;
|
||||
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
using System.Collections.Concurrent;
|
||||
@@ -21,17 +19,6 @@ namespace ThingsGateway.Gateway.Application;
|
||||
internal sealed class ChannelThreadManage : IChannelThreadManage
|
||||
{
|
||||
private ILogger _logger;
|
||||
private IDispatchService<ChannelRuntime> channelRuntimeDispatchService;
|
||||
private IDispatchService<ChannelRuntime> ChannelRuntimeDispatchService
|
||||
{
|
||||
get
|
||||
{
|
||||
if (channelRuntimeDispatchService == null)
|
||||
channelRuntimeDispatchService = App.GetService<IDispatchService<ChannelRuntime>>();
|
||||
|
||||
return channelRuntimeDispatchService;
|
||||
}
|
||||
}
|
||||
|
||||
public ChannelThreadManage()
|
||||
{
|
||||
@@ -78,7 +65,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||
|
||||
await PrivateRemoveChannelsAsync(Enumerable.Repeat(channelId, 1)).ConfigureAwait(false);
|
||||
ChannelRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -98,7 +84,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||
|
||||
await PrivateRemoveChannelsAsync(channelIds).ConfigureAwait(false);
|
||||
ChannelRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -165,7 +150,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
||||
{
|
||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||
await PrivateRestartChannelAsync(Enumerable.Repeat(channelRuntime, 1)).ConfigureAwait(false);
|
||||
ChannelRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -183,7 +167,6 @@ internal sealed class ChannelThreadManage : IChannelThreadManage
|
||||
{
|
||||
await NewChannelLock.WaitAsync().ConfigureAwait(false);
|
||||
await PrivateRestartChannelAsync(channelRuntimes).ConfigureAwait(false);
|
||||
ChannelRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
|
||||
@@ -38,17 +38,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
||||
/// </summary>
|
||||
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()
|
||||
{
|
||||
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 PrivateRestartDeviceAsync(Enumerable.Repeat(deviceRuntime, 1), deleteCache).ConfigureAwait(false);
|
||||
DeviceRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -268,7 +256,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
||||
{
|
||||
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
||||
await PrivateRestartDeviceAsync(deviceRuntimes, deleteCache).ConfigureAwait(false);
|
||||
DeviceRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -440,7 +427,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
||||
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
||||
|
||||
await PrivateRemoveDevicesAsync(Enumerable.Repeat(deviceId, 1)).ConfigureAwait(false);
|
||||
DeviceRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -460,7 +446,6 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
||||
await NewDeviceLock.WaitAsync().ConfigureAwait(false);
|
||||
|
||||
await PrivateRemoveDevicesAsync(deviceIds).ConfigureAwait(false);
|
||||
DeviceRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
finally
|
||||
{
|
||||
@@ -662,6 +647,7 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
||||
//传入变量
|
||||
//newDeviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.SafeDispose());
|
||||
deviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.Init(newDeviceRuntime));
|
||||
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||
}
|
||||
|
||||
/// <inheritdoc/>
|
||||
@@ -740,6 +726,8 @@ internal sealed class DeviceThreadManage : IAsyncDisposable, IDeviceThreadManage
|
||||
LogMessage?.LogWarning($"device {newDeviceRuntime.Name} cannot found channel with id{newDeviceRuntime.ChannelId}");
|
||||
|
||||
newDeviceRuntime.Init(channelRuntime);
|
||||
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||
|
||||
await channelRuntime.DeviceThreadManage.RestartDeviceAsync(newDeviceRuntime, false).ConfigureAwait(false);
|
||||
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);
|
||||
|
||||
|
||||
|
||||
@@ -25,8 +25,6 @@ using ThingsGateway.NewLife;
|
||||
|
||||
using TouchSocket.Core;
|
||||
|
||||
using Yitter.IdGenerator;
|
||||
|
||||
namespace ThingsGateway.Gateway.Application;
|
||||
|
||||
/// <summary>
|
||||
@@ -518,7 +516,7 @@ internal sealed class PluginService : IPluginService
|
||||
{
|
||||
var fileInfo = new FileInfo(path);
|
||||
if (fileInfo.Exists)
|
||||
fileInfo.MoveTo($"{path}{YitIdHelper.NextId()}{DelEx}", true);
|
||||
fileInfo.MoveTo($"{path}{CommonUtils.GetSingleId()}{DelEx}", true);
|
||||
else
|
||||
return false;
|
||||
return true;
|
||||
@@ -598,7 +596,7 @@ internal sealed class PluginService : IPluginService
|
||||
}
|
||||
_ = Task.Run(() =>
|
||||
{
|
||||
_dispatchService.Dispatch(new());
|
||||
_dispatchService.Dispatch(null);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@@ -51,6 +51,9 @@ internal static class RuntimeServiceHelper
|
||||
logger.LogWarning(ex, "Init Channel");
|
||||
}
|
||||
}
|
||||
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||
|
||||
}
|
||||
|
||||
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");
|
||||
}
|
||||
}
|
||||
|
||||
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||
|
||||
}
|
||||
|
||||
public static void Init(List<DeviceRuntime> newDeviceRuntimes)
|
||||
@@ -124,6 +132,10 @@ internal static class RuntimeServiceHelper
|
||||
deviceRuntime.VariableRuntimes.ParallelForEach(a => a.Value.Init(newDeviceRuntime));
|
||||
}
|
||||
}
|
||||
|
||||
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||
|
||||
}
|
||||
public static void Init(List<VariableRuntime> newVariableRuntimes)
|
||||
{
|
||||
@@ -138,10 +150,20 @@ internal static class RuntimeServiceHelper
|
||||
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)
|
||||
{
|
||||
@@ -179,6 +201,8 @@ internal static class RuntimeServiceHelper
|
||||
});
|
||||
deviceRuntime.Dispose();
|
||||
}
|
||||
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||
|
||||
return changedDriver;
|
||||
}
|
||||
@@ -222,6 +246,10 @@ internal static class RuntimeServiceHelper
|
||||
|
||||
}
|
||||
|
||||
|
||||
GlobalData.ChannelDeviceRuntimeDispatchService.Dispatch(null);
|
||||
GlobalData.VariableRuntimeDispatchService.Dispatch(null);
|
||||
|
||||
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);
|
||||
}
|
||||
|
||||
App.GetService<IDispatchService<DeviceRuntime>>().Dispatch(null);
|
||||
}
|
||||
}
|
||||
finally
|
||||
|
||||
@@ -36,20 +36,13 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
||||
private readonly IChannelService _channelService;
|
||||
private readonly IDeviceService _deviceService;
|
||||
private readonly IPluginService _pluginService;
|
||||
private readonly IDispatchService<bool> _allDispatchService;
|
||||
private readonly IDispatchService<Variable> _dispatchService;
|
||||
|
||||
/// <inheritdoc cref="IVariableService"/>
|
||||
public VariableService(
|
||||
IDispatchService<Variable> dispatchService,
|
||||
IDispatchService<bool> allDispatchService
|
||||
)
|
||||
public VariableService()
|
||||
{
|
||||
_channelService = App.RootServices.GetRequiredService<IChannelService>();
|
||||
_pluginService = App.RootServices.GetRequiredService<IPluginService>();
|
||||
_deviceService = App.RootServices.GetRequiredService<IDeviceService>();
|
||||
_dispatchService = dispatchService;
|
||||
_allDispatchService = allDispatchService;
|
||||
}
|
||||
|
||||
#region 测试
|
||||
@@ -230,7 +223,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
||||
{
|
||||
_channelService.DeleteChannelFromCache();//刷新缓存
|
||||
_deviceService.DeleteDeviceFromCache();
|
||||
_allDispatchService.Dispatch(new());
|
||||
DeleteVariableCache();
|
||||
}
|
||||
else
|
||||
@@ -297,7 +289,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
||||
}
|
||||
finally
|
||||
{
|
||||
_dispatchService.Dispatch(new());
|
||||
|
||||
}
|
||||
}
|
||||
@@ -318,7 +309,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
||||
.ToList();
|
||||
|
||||
var result = (await db.Updateable(data).UpdateColumns(differences.Select(a => a.Key).ToArray()).ExecuteCommandAsync().ConfigureAwait(false)) > 0;
|
||||
_dispatchService.Dispatch(new());
|
||||
if (result)
|
||||
DeleteVariableCache();
|
||||
return result;
|
||||
@@ -341,7 +331,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
||||
|
||||
if (result > 0)
|
||||
DeleteVariableCache();
|
||||
_dispatchService.Dispatch(new());
|
||||
}
|
||||
|
||||
[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?.Count == 0, u => u.CreateUserId == UserManager.UserId)
|
||||
.ExecuteCommandAsync().ConfigureAwait(false)) > 0;
|
||||
_dispatchService.Dispatch(new());
|
||||
|
||||
if (result)
|
||||
DeleteVariableCache();
|
||||
@@ -428,7 +416,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
||||
|
||||
if (await base.SaveAsync(input, type).ConfigureAwait(false))
|
||||
{
|
||||
_dispatchService.Dispatch(new());
|
||||
DeleteVariableCache();
|
||||
return true;
|
||||
}
|
||||
@@ -493,7 +480,6 @@ internal sealed class VariableService : BaseService<Variable>, IVariableService
|
||||
using var db = GetDB();
|
||||
await db.BulkCopyAsync(insertData, 100000).ConfigureAwait(false);
|
||||
await db.BulkUpdateAsync(upData, 100000).ConfigureAwait(false);
|
||||
_dispatchService.Dispatch(new());
|
||||
DeleteVariableCache();
|
||||
return variables.Select(a => a.Id).ToHashSet();
|
||||
}
|
||||
|
||||
@@ -18,6 +18,8 @@ using SqlSugar;
|
||||
|
||||
using System.Reflection;
|
||||
|
||||
using ThingsGateway.Authentication;
|
||||
|
||||
namespace ThingsGateway.Gateway.Application;
|
||||
|
||||
[AppStartup(-100)]
|
||||
@@ -25,6 +27,9 @@ public class Startup : AppStartup
|
||||
{
|
||||
public void Configure(IServiceCollection services)
|
||||
{
|
||||
|
||||
ProAuthentication.TryGetAuthorizeInfo(out var authorizeInfo);
|
||||
|
||||
services.AddConfigurableOptions<ChannelThreadOptions>();
|
||||
services.AddConfigurableOptions<GatewayLogOptions>();
|
||||
services.AddConfigurableOptions<RpcLogOptions>();
|
||||
@@ -125,8 +130,35 @@ public class Startup : AppStartup
|
||||
}
|
||||
catch { }
|
||||
|
||||
try
|
||||
{
|
||||
using var db = DbContext.GetDB<BackendLog>();
|
||||
if (db.CurrentConnectionConfig.DbType == SqlSugar.DbType.Sqlite)
|
||||
{
|
||||
|
||||
if (!db.DbMaintenance.IsAnyIndex("idx_backendlog_logtime_date"))
|
||||
{
|
||||
var indexsql = "CREATE INDEX idx_backendlog_logtime_date ON backend_log(strftime('%Y-%m-%d', LogTime));";
|
||||
db.Ado.ExecuteCommand(indexsql);
|
||||
}
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
try
|
||||
{
|
||||
using var db = DbContext.GetDB<RpcLog>();
|
||||
if (db.CurrentConnectionConfig.DbType == SqlSugar.DbType.Sqlite)
|
||||
{
|
||||
|
||||
if (!db.DbMaintenance.IsAnyIndex("idx_rpclog_logtime_date"))
|
||||
{
|
||||
var indexsql = "CREATE INDEX idx_rpclog_logtime_date ON rpc_log(strftime('%Y-%m-%d', LogTime));";
|
||||
db.Ado.ExecuteCommand(indexsql);
|
||||
}
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
serviceProvider.GetService<IHostApplicationLifetime>().ApplicationStarted.Register(() =>
|
||||
{
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<Import Project="$(SolutionDir)Version.props" />
|
||||
<Import Project="$(SolutionDir)PackNuget.props" />
|
||||
@@ -8,9 +8,10 @@
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Portable.BouncyCastle" Version="1.9.0" />
|
||||
<PackageReference Include="Rougamo.Fody" Version="5.0.0" />
|
||||
<PackageReference Include="TouchSocket.Dmtp" Version="3.1.5" />
|
||||
<PackageReference Include="TouchSocket.WebApi.Swagger" Version="3.1.5" />
|
||||
<PackageReference Include="TouchSocket.Dmtp" Version="3.1.6" />
|
||||
<PackageReference Include="TouchSocket.WebApi.Swagger" Version="3.1.6" />
|
||||
<PackageReference Include="ThingsGateway.Authentication" Version="$(AuthenticationVersion)" />
|
||||
<!--<ProjectReference Include="..\..\PluginPro\ThingsGateway.Authentication\ThingsGateway.Authentication.csproj" />-->
|
||||
|
||||
</ItemGroup>
|
||||
|
||||
|
||||
@@ -10,8 +10,6 @@
|
||||
|
||||
using Microsoft.Extensions.Options;
|
||||
|
||||
using ThingsGateway.Authentication;
|
||||
|
||||
namespace ThingsGateway.Gateway.Razor;
|
||||
|
||||
/// <inheritdoc/>
|
||||
@@ -27,39 +25,5 @@ public partial class GatewayAbout
|
||||
[NotNull]
|
||||
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);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user