using DotBased.AspNet.Authority.Models.Authority; using DotBased.AspNet.Authority.Repositories; using Microsoft.EntityFrameworkCore; namespace DotBased.AspNet.Authority.EFCore.Repositories; public class AttributeRepository(IDbContextFactory contextFactory) : RepositoryBase, IAttributeRepository { public async Task> GetAttributesAsync(int limit = 20, int offset = 0, string search = "", CancellationToken cancellationToken = default) { try { await using var context = await contextFactory.CreateDbContextAsync(cancellationToken); var query = context.Attributes.AsQueryable(); if (!string.IsNullOrEmpty(search)) { query = query.Where(a => $"{a.AttributeKey} {a.BoundId} {a.AttributeValue}".Contains(search, StringComparison.CurrentCultureIgnoreCase)); } var total = await query.CountAsync(cancellationToken); var select = await query.OrderBy(a => a.AttributeKey).Skip(offset).Take(limit).Select(a => new AuthorityAttributeItem() { BoundId = a.BoundId, AttributeKey = a.AttributeKey, AttributeValue = a.AttributeValue }).ToListAsync(cancellationToken); return ListResult.Ok(select, total, limit, offset); } catch (Exception e) { return HandleExceptionListResult("Failed to get attributes", e); } } public async Task> GetAttributeByKeyAsync(string key, CancellationToken cancellationToken = default) { try { await using var context = await contextFactory.CreateDbContextAsync(cancellationToken); var attribute = await context.Attributes.FirstOrDefaultAsync(a => a.AttributeKey == key, cancellationToken); return attribute == null ? Result.Failed("Attribute not found") : Result.Ok(attribute); } catch (Exception e) { return HandleExceptionResult("Failed to get attribute by id", e); } } public async Task> CreateAttributeAsync(AuthorityAttribute attribute, CancellationToken cancellationToken = default) { try { await using var context = await contextFactory.CreateDbContextAsync(cancellationToken); if (string.IsNullOrWhiteSpace(attribute.AttributeKey) || attribute.BoundId == Guid.Empty) { return Result.Failed("Attribute key and/or bound id is empty"); } var entry = context.Attributes.Add(attribute); var saveResult = await context.SaveChangesAsync(cancellationToken); return saveResult <= 0 ? Result.Failed("Failed to create attribute") : Result.Ok(entry.Entity); } catch (Exception e) { return HandleExceptionResult("Failed to create attribute", e); } } public async Task> UpdateAttributeAsync(AuthorityAttribute attribute, CancellationToken cancellationToken = default) { try { await using var context = await contextFactory.CreateDbContextAsync(cancellationToken); var currentAttribute = await context.Attributes.FirstOrDefaultAsync(a => a.AttributeKey == attribute.AttributeKey, cancellationToken); if (currentAttribute == null) { return Result.Failed("Attribute not found"); } if (currentAttribute.Version != attribute.Version) { return Result.Failed("Attribute version doesn't match"); } var entry = context.Attributes.Update(currentAttribute); var saveResult = await context.SaveChangesAsync(cancellationToken); return saveResult <= 0 ? Result.Failed("Failed to update attribute") : Result.Ok(entry.Entity); } catch (Exception e) { return HandleExceptionResult("Failed to update attribute", e); } } public async Task DeleteAttributeAsync(AuthorityAttribute attribute, CancellationToken cancellationToken = default) { try { await using var context = await contextFactory.CreateDbContextAsync(cancellationToken); var currentAttribute = await context.Attributes.FirstOrDefaultAsync(a => a.AttributeKey == attribute.AttributeKey, cancellationToken); if (currentAttribute == null) { return Result.Failed("Attribute not found"); } context.Attributes.Remove(currentAttribute); var saveResult = await context.SaveChangesAsync(cancellationToken); return saveResult <= 0 ? Result.Failed("Failed to delete attribute") : Result.Ok(); } catch (Exception e) { return HandleException("Failed to delete attribute", e); } } }