DD.WellWorkover.Cloud/AsbCloudInfrastructure/Repository/CrudRepositoryBase.cs

190 lines
6.5 KiB
C#
Raw Normal View History

using AsbCloudApp.Services;
2021-08-10 14:36:35 +05:00
using AsbCloudDb.Model;
using Mapster;
using Microsoft.EntityFrameworkCore;
2021-09-10 11:28:57 +05:00
using System;
2021-08-10 14:36:35 +05:00
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using AsbCloudApp.Exceptions;
2024-03-26 13:35:36 +05:00
using Microsoft.EntityFrameworkCore.ChangeTracking;
2021-08-10 14:36:35 +05:00
namespace AsbCloudInfrastructure.Repository
2021-08-10 14:36:35 +05:00
{
/// <summary>
/// CRUD сервис для работы с БД
/// </summary>
/// <typeparam name="TDto"></typeparam>
/// <typeparam name="TEntity"></typeparam>
public class CrudRepositoryBase<TDto, TEntity> : QueryContainer<TEntity>, ICrudRepository<TDto>
where TDto : AsbCloudApp.Data.IId
where TEntity : class, IId
2021-08-10 14:36:35 +05:00
{
public CrudRepositoryBase(IAsbCloudDbContext context)
: base(context)
{ }
2021-09-10 11:28:57 +05:00
public CrudRepositoryBase(IAsbCloudDbContext context, Func<DbSet<TEntity>, IQueryable<TEntity>> makeQuery)
: base(context, makeQuery)
{ }
/// <inheritdoc/>
2022-12-07 15:23:10 +05:00
public virtual async Task<IEnumerable<TDto>> GetAllAsync(CancellationToken token)
{
var entities = await GetQuery()
.AsNoTracking()
2021-09-10 11:28:57 +05:00
.ToListAsync(token)
.ConfigureAwait(false);
var dtos = entities.Select(Convert).ToList();
return dtos;
2021-09-10 11:28:57 +05:00
}
/// <inheritdoc/>
2022-12-07 15:23:10 +05:00
public virtual async Task<TDto?> GetOrDefaultAsync(int id, CancellationToken token)
2021-09-10 11:28:57 +05:00
{
var entity = await GetQuery()
.AsNoTracking()
.FirstOrDefaultAsync(e => e.Id == id, token)
.ConfigureAwait(false);
if (entity == default)
return default;
var dto = Convert(entity);
2021-09-10 11:28:57 +05:00
return dto;
}
2021-09-10 11:28:57 +05:00
/// <inheritdoc/>
public virtual TDto? GetOrDefault(int id)
2021-08-10 14:36:35 +05:00
{
var entity = GetQuery()
.AsNoTracking()
.FirstOrDefault(e => e.Id == id);
if (entity == default)
return default;
2021-09-10 11:28:57 +05:00
var dto = Convert(entity);
2021-08-10 14:36:35 +05:00
return dto;
}
/// <inheritdoc/>
2022-12-07 15:23:10 +05:00
public virtual async Task<int> InsertAsync(TDto item, CancellationToken token)
2021-08-10 14:36:35 +05:00
{
2021-09-10 11:28:57 +05:00
var entity = Convert(item);
entity.Id = 0;
2022-06-15 14:57:37 +05:00
var entry = dbSet.Add(entity);
await dbContext.SaveChangesAsync(token);
2022-06-09 17:25:26 +05:00
entry.State = EntityState.Detached;
return entity.Id;
2021-08-10 14:36:35 +05:00
}
/// <inheritdoc/>
2022-12-07 15:23:10 +05:00
public virtual async Task<int> InsertRangeAsync(IEnumerable<TDto> items, CancellationToken token)
2021-08-10 14:36:35 +05:00
{
if (!items.Any())
2022-06-09 17:25:26 +05:00
return 0;
2022-04-11 18:00:34 +05:00
var entities = items.Select(i =>
{
var entity = Convert(i);
entity.Id = 0;
return entity;
});
2024-03-26 13:35:36 +05:00
var entries = new List<EntityEntry>(items.Count());
2022-06-09 17:25:26 +05:00
foreach (var entity in entities)
{
var entry = dbSet.Add(entity);
entries.Add(entry);
}
var affected = await dbContext.SaveChangesAsync(token);
entries.ForEach(e => e.State = EntityState.Detached);
return affected;
2021-08-10 14:36:35 +05:00
}
/// <inheritdoc/>
2022-12-07 15:23:10 +05:00
public virtual async Task<int> UpdateAsync(TDto item, CancellationToken token)
2021-08-10 14:36:35 +05:00
{
var existingEntity = await dbSet
.AsNoTracking()
.FirstOrDefaultAsync(e => e.Id == item.Id, token)
.ConfigureAwait(false);
2022-06-15 14:57:37 +05:00
if (existingEntity is null)
return ICrudRepository<TDto>.ErrorIdNotFound;
2022-06-15 14:57:37 +05:00
2021-09-10 11:28:57 +05:00
var entity = Convert(item);
var entry = dbSet.Update(entity);
await dbContext.SaveChangesAsync(token);
2022-06-09 17:25:26 +05:00
entry.State = EntityState.Detached;
return entry.Entity.Id;
2021-08-10 14:36:35 +05:00
}
public virtual async Task<int> UpdateRangeAsync(IEnumerable<TDto> dtos, CancellationToken token)
{
if (!dtos.Any())
return 0;
var ids = dtos
.Select(o => o.Id)
.Distinct()
.ToArray();
if (ids.Any(id => id == default))
throw new ArgumentInvalidException(nameof(dtos), "Все записи должны иметь Id");
if (ids.Length != dtos.Count())
throw new ArgumentInvalidException(nameof(dtos), "Все записи должны иметь уникальные Id");
var existingEntitiesCount = await dbContext.Set<TEntity>()
.Where(o => ids.Contains(o.Id))
.CountAsync(token);
if (ids.Length != existingEntitiesCount)
throw new ArgumentInvalidException(nameof(dtos), "Все записи должны существовать в БД");
var entities = dtos.Select(Convert);
var entries = entities.Select(entity => dbContext.Set<TEntity>().Update(entity)).ToList();
var affected = await dbContext.SaveChangesAsync(token);
entries.ForEach(entry => entry.State = EntityState.Detached);
return affected;
}
2021-08-10 14:36:35 +05:00
/// <inheritdoc/>
2022-12-07 15:23:10 +05:00
public virtual Task<int> DeleteAsync(int id, CancellationToken token)
2021-08-10 14:36:35 +05:00
{
var entity = dbSet
.AsNoTracking()
.FirstOrDefault(e => e.Id == id);
2021-08-10 14:36:35 +05:00
if (entity == default)
return Task.FromResult(ICrudRepository<TDto>.ErrorIdNotFound);
dbSet.Remove(entity);
2022-06-09 17:25:26 +05:00
var affected = dbContext.SaveChangesAsync(token);
return affected;
2021-08-10 14:36:35 +05:00
}
public virtual async Task<int> DeleteRangeAsync(IEnumerable<int> ids, CancellationToken token)
{
2024-03-26 13:35:36 +05:00
if (!ids.Any())
return 0;
var countExistingEntities = await dbSet
.Where(d => ids.Contains(d.Id))
.CountAsync(token);
2024-03-26 13:35:36 +05:00
if (ids.Count() > countExistingEntities)
return ICrudRepository<TDto>.ErrorIdNotFound;
var entities = dbContext.Set<TEntity>().Where(e => ids.Contains(e.Id));
dbContext.Set<TEntity>().RemoveRange(entities);
return await dbContext.SaveChangesAsync(token);
}
protected virtual TDto Convert(TEntity src) => src.Adapt<TDto>();
2021-09-10 11:28:57 +05:00
protected virtual TEntity Convert(TDto src) => src.Adapt<TEntity>();
2021-08-10 14:36:35 +05:00
}
2021-08-10 14:36:35 +05:00
}