using NetAdmin.Application.Repositories; using NetAdmin.Application.Services; using NetAdmin.Domain.Dto.Dependency; using NetAdmin.Domain.Dto.Sys; using NetAdmin.Domain.Dto.Sys.LoginLog; using NetAdmin.Domain.Dto.Sys.RequestLog; using NetAdmin.SysComponent.Application.Services.Sys.Dependency; namespace NetAdmin.SysComponent.Application.Services.Sys; /// public sealed class RequestLogService(BasicRepository rpo, LoginLogService loginLogService) // : RepositoryService(rpo), IRequestLogService { /// public async Task BulkDeleteAsync(BulkReq req) { req.ThrowIfInvalid(); var ret = 0; // ReSharper disable once LoopCanBeConvertedToQuery foreach (var item in req.Items) { ret += await DeleteAsync(item).ConfigureAwait(false); } return ret; } /// public Task CountAsync(QueryReq req) { req.ThrowIfInvalid(); return QueryInternal(req) #if DBTYPE_SQLSERVER .WithLock(SqlServerLock.NoLock | SqlServerLock.NoWait) #endif .CountAsync(); } /// public async Task CreateAsync(CreateRequestLogReq req) { req.ThrowIfInvalid(); var ret = await Rpo.InsertAsync(req).ConfigureAwait(false); // 插入登录日志 if (req.ApiPathCrc32 == Chars.FLG_PATH_API_SYS_USER_LOGIN_BY_PWD.Crc32()) { _ = await loginLogService.CreateAsync(req.Adapt()).ConfigureAwait(false); } return ret.Adapt(); } /// public Task DeleteAsync(DelReq req) { req.ThrowIfInvalid(); throw new NotImplementedException(); } /// public Task ExistAsync(QueryReq req) { req.ThrowIfInvalid(); return QueryInternal(req) #if DBTYPE_SQLSERVER .WithLock(SqlServerLock.NoLock | SqlServerLock.NoWait) #endif .AnyAsync(); } /// public Task ExportAsync(QueryReq req) { req.ThrowIfInvalid(); return ExportAsync( // QueryInternal, req, Ln.请求日志导出, a => new { a.Id , Api = new { a.Api.Id } , a.CreatedClientIp , a.CreatedTime , a.Duration , a.HttpMethod , a.HttpStatusCode , Owner = new { a.Owner.UserName } }); } /// public async Task GetAsync(QueryRequestLogReq req) { req.ThrowIfInvalid(); var df = new DynamicFilterInfo { Field = nameof(QueryRequestLogReq.CreatedTime) , Operator = DynamicFilterOperators.DateRange , Value = new[] { req.CreatedTime.AddHours(-1).yyyy_MM_dd_HH_mm_ss() , req.CreatedTime.AddHours(1).yyyy_MM_dd_HH_mm_ss() }.Json() .Object() }; var ret = await QueryInternal(new QueryReq { Filter = req, DynamicFilter = df }) .Include(a => a.Detail) .ToOneAsync() .ConfigureAwait(false); return ret.Adapt(); } /// public async Task> GetBarChartAsync(QueryReq req) { req.ThrowIfInvalid(); var ret = await QueryInternal(req with { Order = Orders.None }, false) #if DBTYPE_SQLSERVER .WithLock(SqlServerLock.NoLock | SqlServerLock.NoWait) #endif .GroupBy(a => new { a.CreatedTime.Year , a.CreatedTime.Month , a.CreatedTime.Day , a.CreatedTime.Hour }) .ToListAsync(a => new GetBarChartRsp { Timestamp = new DateTime( a.Key.Year, a.Key.Month, a.Key.Day, a.Key.Hour, 0 , 0, DateTimeKind.Unspecified) , Value = a.Count() }) .ConfigureAwait(false); return ret.OrderBy(x => x.Timestamp); } /// public async Task> GetPieChartByApiSummaryAsync(QueryReq req) { req.ThrowIfInvalid(); var ret = await QueryInternal(req with { Order = Orders.None }) #if DBTYPE_SQLSERVER .WithLock(SqlServerLock.NoLock | SqlServerLock.NoWait) #endif .GroupBy(a => a.Api.Summary) .ToListAsync(a => new GetPieChartRsp { Value = a.Count(), Key = a.Key }) .ConfigureAwait(false); return ret.OrderByDescending(x => x.Value); } /// public async Task> GetPieChartByHttpStatusCodeAsync(QueryReq req) { req.ThrowIfInvalid(); var ret = await QueryInternal(req with { Order = Orders.None }, false) #if DBTYPE_SQLSERVER .WithLock(SqlServerLock.NoLock | SqlServerLock.NoWait) #endif .GroupBy(a => a.HttpStatusCode) #pragma warning disable CA1305 .ToListAsync(a => new GetPieChartRsp { Value = a.Count(), Key = a.Key.ToString() }) #pragma warning restore CA1305 .ConfigureAwait(false); return ret.Select(x => x with { Key = Enum.Parse(x.Key).ToString() }) .OrderByDescending(x => x.Value); } /// public async Task> PagedQueryAsync(PagedQueryReq req) { req.ThrowIfInvalid(); var select = QueryInternal(req with { Order = Orders.None }, false); var selectTemp = select.WithTempQuery(a => new { temp = a }); if (req.Order == Orders.Random) { selectTemp = selectTemp.OrderByRandom(); } else { selectTemp = selectTemp.OrderBy( // req.Prop?.Length > 0, $"{req.Prop} {(req.Order == Orders.Ascending ? "ASC" : "DESC")}"); if (!req.Prop?.Equals(nameof(req.Filter.CreatedTime), StringComparison.OrdinalIgnoreCase) ?? true) { selectTemp = selectTemp.OrderByDescending(a => a.temp.CreatedTime); } } var ret = await selectTemp.Page(req.Page, req.PageSize) #if DBTYPE_SQLSERVER .WithLock(SqlServerLock.NoLock | SqlServerLock.NoWait) #endif .Count(out var total) .ToListAsync(a => a.temp) .ConfigureAwait(false); return new PagedQueryRsp(req.Page, req.PageSize, total , ret.Adapt>()); } /// public async Task> QueryAsync(QueryReq req) { req.ThrowIfInvalid(); var ret = await QueryInternal(req) #if DBTYPE_SQLSERVER .WithLock(SqlServerLock.NoLock | SqlServerLock.NoWait) #endif .Take(req.Count) .ToListAsync() .ConfigureAwait(false); return ret.Adapt>(); } private ISelect QueryInternal(QueryReq req) { return QueryInternal(req, true); } private ISelect QueryInternal(QueryReq req, bool include) { var ret = Rpo.Select; if (include) { ret = ret.Include(a => a.Api).Include(a => a.Owner); } ret = ret.WhereDynamicFilter(req.DynamicFilter); if (req.Filter?.Id is not 0) { ret = ret.WhereDynamic(req.Filter); } if (req.Keywords?.Length > 0) { ret = req.Keywords.IsIpV4() ? ret.Where(a => a.CreatedClientIp == req.Keywords.IpV4ToInt32()) : ret.Where(a => a.Id == req.Keywords.Int64Try(0) || a.OwnerId == req.Keywords.Int64Try(0)); } // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault switch (req.Order) { case Orders.None: return ret; case Orders.Random: return ret.OrderByRandom(); } ret = ret.OrderByPropertyNameIf(req.Prop?.Length > 0, req.Prop, req.Order == Orders.Ascending); if (!req.Prop?.Equals(nameof(req.Filter.CreatedTime), StringComparison.OrdinalIgnoreCase) ?? true) { ret = ret.OrderByDescending(a => a.CreatedTime); } return ret; } }