2026-02-24 20:44:16 +00:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.Linq;
|
2026-06-05 19:05:57 +00:00
|
|
|
|
using System.Linq.Expressions;
|
|
|
|
|
|
using System.Text.Json;
|
2026-02-24 20:44:16 +00:00
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
using Microsoft.AspNetCore.Authorization;
|
2026-06-05 19:05:57 +00:00
|
|
|
|
using Sozsoft.Platform.Entities;
|
|
|
|
|
|
using Sozsoft.Platform.ListForms;
|
2026-02-24 20:44:16 +00:00
|
|
|
|
using Volo.Abp;
|
|
|
|
|
|
using Volo.Abp.Application.Dtos;
|
|
|
|
|
|
using Volo.Abp.Application.Services;
|
|
|
|
|
|
using Volo.Abp.AuditLogging;
|
2026-06-05 19:05:57 +00:00
|
|
|
|
using Volo.Abp.Domain.Repositories;
|
2026-02-24 20:44:16 +00:00
|
|
|
|
using Volo.Abp.Uow;
|
|
|
|
|
|
using static Sozsoft.Platform.Data.Seeds.SeedConsts;
|
|
|
|
|
|
|
|
|
|
|
|
namespace Sozsoft.Platform.AuditLogs;
|
|
|
|
|
|
|
|
|
|
|
|
public interface IAuditLogAppService
|
2026-06-05 19:05:57 +00:00
|
|
|
|
: ICrudAppService<AuditLogDto, Guid, AuditLogListRequestDto>
|
2026-02-24 20:44:16 +00:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-28 08:23:21 +00:00
|
|
|
|
[Authorize(AppCodes.IdentityManagement.AuditLogs)]
|
2026-06-05 19:05:57 +00:00
|
|
|
|
public class AuditLogAppService : CrudAppService<
|
|
|
|
|
|
AuditLog,
|
|
|
|
|
|
AuditLogDto,
|
|
|
|
|
|
Guid,
|
|
|
|
|
|
AuditLogListRequestDto>, IAuditLogAppService
|
2026-02-24 20:44:16 +00:00
|
|
|
|
{
|
2026-06-05 19:05:57 +00:00
|
|
|
|
private readonly IRepository<ListForm, Guid> _listFormRepository;
|
|
|
|
|
|
|
|
|
|
|
|
public AuditLogAppService(
|
|
|
|
|
|
IAuditLogRepository auditLogRepository,
|
|
|
|
|
|
IRepository<ListForm, Guid> listFormRepository
|
|
|
|
|
|
) : base(auditLogRepository)
|
2026-02-24 20:44:16 +00:00
|
|
|
|
{
|
2026-06-05 19:05:57 +00:00
|
|
|
|
_listFormRepository = listFormRepository;
|
2026-02-24 20:44:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public override async Task<AuditLogDto> GetAsync(Guid id)
|
|
|
|
|
|
{
|
|
|
|
|
|
var entity = await Repository.GetAsync(id, includeDetails: true);
|
|
|
|
|
|
|
|
|
|
|
|
return await MapToGetOutputDtoAsync(entity);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
[UnitOfWork]
|
2026-06-05 19:05:57 +00:00
|
|
|
|
public override async Task<PagedResultDto<AuditLogDto>> GetListAsync(AuditLogListRequestDto input)
|
2026-02-24 20:44:16 +00:00
|
|
|
|
{
|
2026-06-05 19:05:57 +00:00
|
|
|
|
var query = await Repository.WithDetailsAsync();
|
|
|
|
|
|
|
|
|
|
|
|
if (!input.ListFormCode.IsNullOrWhiteSpace())
|
|
|
|
|
|
{
|
|
|
|
|
|
var filterRules = await GetListFormFilterRulesAsync(input.ListFormCode);
|
|
|
|
|
|
query = ApplyAuditLogActionParametersFilter(query, filterRules, input.EntityId);
|
|
|
|
|
|
}
|
|
|
|
|
|
else if (!input.EntityId.IsNullOrWhiteSpace())
|
|
|
|
|
|
{
|
|
|
|
|
|
query = query.Where(a => a.Actions.Any(action => action.Parameters.Contains(input.EntityId)));
|
|
|
|
|
|
}
|
2026-02-24 20:44:16 +00:00
|
|
|
|
|
|
|
|
|
|
var totalCount = await AsyncExecuter.CountAsync(query);
|
|
|
|
|
|
|
|
|
|
|
|
query = ApplySorting(query, input);
|
|
|
|
|
|
query = ApplyPaging(query, input);
|
|
|
|
|
|
|
2026-06-05 19:05:57 +00:00
|
|
|
|
var auditLogsWithDetails = await AsyncExecuter.ToListAsync(query);
|
2026-02-24 20:44:16 +00:00
|
|
|
|
|
|
|
|
|
|
// Mapping tek seferde yap
|
|
|
|
|
|
var entityDtos = ObjectMapper.Map<List<AuditLog>, List<AuditLogDto>>(auditLogsWithDetails);
|
2026-06-05 19:05:57 +00:00
|
|
|
|
|
2026-02-24 20:44:16 +00:00
|
|
|
|
// EntityChangeCount'u doldur (artık EntityChanges yüklü)
|
|
|
|
|
|
foreach (var dto in entityDtos)
|
|
|
|
|
|
{
|
|
|
|
|
|
var auditLog = auditLogsWithDetails.FirstOrDefault(a => a.Id == dto.Id);
|
|
|
|
|
|
dto.EntityChangeCount = auditLog?.EntityChanges?.Count ?? 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return new PagedResultDto<AuditLogDto>(
|
|
|
|
|
|
totalCount,
|
|
|
|
|
|
entityDtos
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-06-05 19:05:57 +00:00
|
|
|
|
private async Task<List<AuditLogListFormFilterRule>> GetListFormFilterRulesAsync(string listFormCode)
|
|
|
|
|
|
{
|
|
|
|
|
|
var rules = new List<AuditLogListFormFilterRule>
|
|
|
|
|
|
{
|
|
|
|
|
|
new(listFormCode, null)
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
var listForm = await _listFormRepository.FirstOrDefaultAsync(a => a.ListFormCode == listFormCode);
|
|
|
|
|
|
if (listForm?.SubFormsJson.IsNullOrWhiteSpace() != false)
|
|
|
|
|
|
{
|
|
|
|
|
|
return rules;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
var subForms = JsonSerializer.Deserialize<List<SubFormDto>>(listForm.SubFormsJson) ?? [];
|
|
|
|
|
|
foreach (var subForm in subForms.Where(a => !a.Code.IsNullOrWhiteSpace()))
|
|
|
|
|
|
{
|
|
|
|
|
|
var childFieldNames = subForm.Relation?
|
|
|
|
|
|
.Select(a => a.ChildFieldName)
|
|
|
|
|
|
.Where(a => !a.IsNullOrWhiteSpace())
|
|
|
|
|
|
.Distinct(StringComparer.OrdinalIgnoreCase)
|
|
|
|
|
|
.ToList() ?? [];
|
|
|
|
|
|
|
|
|
|
|
|
rules.AddRange(childFieldNames.Select(childFieldName => new AuditLogListFormFilterRule(subForm.Code, childFieldName)));
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (JsonException)
|
|
|
|
|
|
{
|
|
|
|
|
|
// Invalid subform JSON should not block audit log listing for the main form.
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return rules
|
|
|
|
|
|
.DistinctBy(a => $"{a.ListFormCode}|{a.ChildFieldName}".ToLowerInvariant())
|
|
|
|
|
|
.ToList();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private static IQueryable<AuditLog> ApplyAuditLogActionParametersFilter(
|
|
|
|
|
|
IQueryable<AuditLog> query,
|
|
|
|
|
|
List<AuditLogListFormFilterRule> rules,
|
|
|
|
|
|
string entityId)
|
|
|
|
|
|
{
|
|
|
|
|
|
var validRules = rules
|
|
|
|
|
|
.Where(rule => !rule.ListFormCode.IsNullOrWhiteSpace())
|
|
|
|
|
|
.ToList();
|
|
|
|
|
|
|
|
|
|
|
|
if (validRules.Count == 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
return query;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var auditLog = Expression.Parameter(typeof(AuditLog), "auditLog");
|
|
|
|
|
|
var action = Expression.Parameter(typeof(AuditLogAction), "action");
|
|
|
|
|
|
var parameters = Expression.Property(action, nameof(AuditLogAction.Parameters));
|
|
|
|
|
|
|
|
|
|
|
|
Expression actionBody = Expression.Constant(false);
|
|
|
|
|
|
foreach (var rule in validRules)
|
|
|
|
|
|
{
|
|
|
|
|
|
Expression ruleBody = Contains(parameters, rule.ListFormCode);
|
|
|
|
|
|
if (!entityId.IsNullOrWhiteSpace())
|
|
|
|
|
|
{
|
|
|
|
|
|
ruleBody = Expression.AndAlso(ruleBody, Contains(parameters, entityId));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!rule.ChildFieldName.IsNullOrWhiteSpace())
|
|
|
|
|
|
{
|
|
|
|
|
|
ruleBody = Expression.AndAlso(ruleBody, Contains(parameters, rule.ChildFieldName));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
actionBody = Expression.OrElse(actionBody, ruleBody);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var actions = Expression.Property(auditLog, nameof(AuditLog.Actions));
|
|
|
|
|
|
var actionPredicate = Expression.Lambda<Func<AuditLogAction, bool>>(actionBody, action);
|
|
|
|
|
|
var anyCall = Expression.Call(
|
|
|
|
|
|
typeof(Enumerable),
|
|
|
|
|
|
nameof(Enumerable.Any),
|
|
|
|
|
|
[typeof(AuditLogAction)],
|
|
|
|
|
|
actions,
|
|
|
|
|
|
actionPredicate);
|
|
|
|
|
|
|
|
|
|
|
|
var auditLogPredicate = Expression.Lambda<Func<AuditLog, bool>>(anyCall, auditLog);
|
|
|
|
|
|
return query.Where(auditLogPredicate);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private static MethodCallExpression Contains(MemberExpression source, string value)
|
|
|
|
|
|
{
|
|
|
|
|
|
return Expression.Call(
|
|
|
|
|
|
source,
|
|
|
|
|
|
nameof(string.Contains),
|
|
|
|
|
|
Type.EmptyTypes,
|
|
|
|
|
|
Expression.Constant(value));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private sealed record AuditLogListFormFilterRule(string ListFormCode, string? ChildFieldName);
|
|
|
|
|
|
|
2026-02-24 20:44:16 +00:00
|
|
|
|
// Audit Log kayitlarini gormek istiyoruz fakat degistirmek istemiyoruz
|
|
|
|
|
|
[RemoteService(IsEnabled = false)]
|
|
|
|
|
|
public override Task<AuditLogDto> CreateAsync(AuditLogDto input)
|
|
|
|
|
|
{
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
[RemoteService(IsEnabled = false)]
|
|
|
|
|
|
public override Task<AuditLogDto> UpdateAsync(Guid id, AuditLogDto input)
|
|
|
|
|
|
{
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
[RemoteService(IsEnabled = false)]
|
|
|
|
|
|
public override Task DeleteAsync(Guid id)
|
|
|
|
|
|
{
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|