using Microsoft.EntityFrameworkCore; using System; using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; namespace AsbCloudInfrastructure.Services.Cache { public class CacheTable where TEntity : class { private readonly DbContext context; private (DateTime refreshDate, IEnumerable entities) data; private readonly List cached; private readonly DbSet dbSet; private readonly SemaphoreSlim semaphore = new SemaphoreSlim(1, 1); internal CacheTable(DbContext context, (DateTime refreshDate, IEnumerable entities) data) { this.context = context; this.data = data; this.cached = (List)data.entities; dbSet = context.Set(); } public TEntity this[int index] { get => cached.ElementAt(index); } public int Refresh() { if (cached.Any()) { semaphore.Wait(); cached.Clear(); } var dbEntities = context.Set().AsNoTracking().ToList(); cached.AddRange(dbEntities); data.refreshDate = DateTime.Now; return cached.Count; } public async Task RefreshAsync(CancellationToken token = default) { if (cached.Any()) { await semaphore.WaitAsync(token).ConfigureAwait(true); cached.Clear(); } var dbEntities = await context.Set().AsNoTracking().ToListAsync(token).ConfigureAwait(false); cached.AddRange(dbEntities); data.refreshDate = DateTime.Now; return cached.Count; } private bool CheckRefresh(RefreshMode refreshMode) { if (refreshMode == RefreshMode.Force) { Refresh(); return true; } if ((!cached.Any()) && (refreshMode == RefreshMode.IfResultEmpty)) { Refresh(); return true; } return false; } private async Task CheckRefreshAsync(RefreshMode refreshMode, CancellationToken token = default) { if (refreshMode == RefreshMode.Force) { await RefreshAsync(token); return true; } if ((!cached.Any()) && (refreshMode == RefreshMode.IfResultEmpty)) { await RefreshAsync(token); return true; } return false; } public bool Contains(Func predicate, RefreshMode refreshMode = RefreshMode.IfResultEmpty) => FirstOrDefault(predicate, refreshMode) != default; public Task ContainsAsync(Func predicate, CancellationToken token = default) => ContainsAsync(predicate, RefreshMode.IfResultEmpty, token); public async Task ContainsAsync(Func predicate, RefreshMode refreshMode = RefreshMode.IfResultEmpty, CancellationToken token = default) => await FirstOrDefaultAsync(predicate, refreshMode, token) != default; public Task FirstOrDefaultAsync(CancellationToken token = default) => FirstOrDefaultAsync(RefreshMode.IfResultEmpty, token); public TEntity FirstOrDefault(RefreshMode refreshMode = RefreshMode.IfResultEmpty) { bool isUpdated = CheckRefresh(refreshMode); var result = cached.FirstOrDefault(); if (result == default && refreshMode == RefreshMode.IfResultEmpty && !isUpdated) { Refresh(); return cached.FirstOrDefault(); } return result; } public async Task FirstOrDefaultAsync(RefreshMode refreshMode = RefreshMode.IfResultEmpty, CancellationToken token = default) { bool isUpdated = await CheckRefreshAsync(refreshMode, token); var result = cached.FirstOrDefault(); if (result == default && refreshMode == RefreshMode.IfResultEmpty && !isUpdated) { await RefreshAsync(token); return cached.FirstOrDefault(); } return result; } public Task FirstOrDefaultAsync(Func predicate, CancellationToken token = default) => FirstOrDefaultAsync(predicate, RefreshMode.IfResultEmpty, token); public TEntity FirstOrDefault(Func predicate, RefreshMode refreshMode = RefreshMode.IfResultEmpty) { bool isUpdated = CheckRefresh(refreshMode); var result = cached.FirstOrDefault(predicate); if (result == default && refreshMode == RefreshMode.IfResultEmpty && !isUpdated) { Refresh(); return cached.FirstOrDefault(predicate); } return result; } public async Task FirstOrDefaultAsync(Func predicate, RefreshMode refreshMode = RefreshMode.IfResultEmpty, CancellationToken token = default) { bool isUpdated = await CheckRefreshAsync(refreshMode, token); var result = cached.FirstOrDefault(predicate); if (result == default && refreshMode == RefreshMode.IfResultEmpty && !isUpdated) { await RefreshAsync(token); return cached.FirstOrDefault(predicate); } return result; } public Task> WhereAsync(CancellationToken token = default) => WhereAsync(default, RefreshMode.IfResultEmpty, token); public Task> WhereAsync(Func predicate, CancellationToken token = default) => WhereAsync(predicate, RefreshMode.IfResultEmpty, token); public IEnumerable Where(Func predicate = default, RefreshMode refreshMode = RefreshMode.IfResultEmpty) { bool isUpdated = CheckRefresh(refreshMode); var result = (predicate != default) ? cached.Where(predicate) : cached; if (!result.Any() && refreshMode == RefreshMode.IfResultEmpty && !isUpdated) { Refresh(); result = (predicate != default) ? cached.Where(predicate) : cached; } return result; } public async Task> WhereAsync(Func predicate = default, RefreshMode refreshMode = RefreshMode.IfResultEmpty, CancellationToken token = default) { bool isUpdated = await CheckRefreshAsync(refreshMode, token); var result = (predicate != default) ? cached.Where(predicate) : cached; if (!result.Any() && refreshMode == RefreshMode.IfResultEmpty && !isUpdated) { await RefreshAsync(token); result = (predicate != default) ? cached.Where(predicate) : cached; } return result; } public IEnumerable Mutate(Func predicate, Action mutation) { var dbEntities = dbSet.Where(predicate); if (dbEntities.Any()) { foreach (var dbEntity in dbEntities) mutation(dbEntity); context.SaveChanges(); } cached.RemoveAll(e => predicate(e)); cached.AddRange(dbEntities); return dbEntities; } public async Task> MutateAsync(Func predicate, Action mutation, CancellationToken token = default) { var dbEntities = dbSet.Where(predicate); if (dbEntities.Any()) { foreach (var dbEntity in dbEntities) mutation(dbEntity); await context.SaveChangesAsync(token).ConfigureAwait(false); } cached.RemoveAll(e => predicate(e)); cached.AddRange(dbEntities); return dbEntities; } public TEntity Upsert(TEntity entity) { var updated = dbSet.Update(entity); context.SaveChanges(); Refresh(); return updated.Entity; } public async Task UpsertAsync(TEntity entity, CancellationToken token = default) { Microsoft.EntityFrameworkCore.ChangeTracking.EntityEntry updated; if (dbSet.Contains(entity)) updated = dbSet.Update(entity); else updated = dbSet.Add(entity); await context.SaveChangesAsync(token).ConfigureAwait(false); await RefreshAsync(token).ConfigureAwait(false); return updated.Entity; } public IEnumerable Upsert(IEnumerable entities) { var upsertedEntries = new List(entities.Count()); foreach (var entity in entities) { Microsoft.EntityFrameworkCore.ChangeTracking.EntityEntry updated; if (dbSet.Contains(entity)) // TODO: это очень ммедленно updated = dbSet.Update(entity); else updated = dbSet.Add(entity); upsertedEntries.Add(updated.Entity); } context.SaveChanges(); Refresh(); return upsertedEntries; } public async Task> UpsertAsync(IEnumerable entities, CancellationToken token = default) { var upsertedEntries = new List(entities.Count()); foreach (var entity in entities) { Microsoft.EntityFrameworkCore.ChangeTracking.EntityEntry updated; if (dbSet.Contains(entity)) updated = dbSet.Update(entity); else updated = dbSet.Add(entity); upsertedEntries.Add(updated.Entity); } await context.SaveChangesAsync(token).ConfigureAwait(false); await RefreshAsync(token).ConfigureAwait(false); return upsertedEntries; } public void Remove(Func predicate) { cached.RemoveAll(e => predicate(e)); dbSet.RemoveRange(dbSet.Where(predicate)); context.SaveChanges(); return; } public async Task RemoveAsync(Func predicate, CancellationToken token = default) { cached.RemoveAll(e => predicate(e)); dbSet.RemoveRange(dbSet.Where(predicate)); await context.SaveChangesAsync(token).ConfigureAwait(false); return; } public TEntity Insert(TEntity entity) { var dbEntity = dbSet.Add(entity).Entity; context.SaveChanges(); cached.Add(dbEntity); return dbEntity; } public async Task InsertAsync(TEntity entity, CancellationToken token = default) { var dbEntity = dbSet.Add(entity).Entity; await context.SaveChangesAsync(token).ConfigureAwait(false); cached.Add(dbEntity); return dbEntity; } public IEnumerable Insert(IEnumerable newEntities) { var dbEntities = new List(newEntities.Count()); foreach (var item in newEntities) dbEntities.Add(dbSet.Add(item).Entity); context.SaveChanges(); cached.AddRange(dbEntities); return dbEntities; } public async Task> InsertAsync(IEnumerable newEntities, CancellationToken token = default) { var dbEntities = new List(newEntities.Count()); foreach (var item in newEntities) dbEntities.Add(dbSet.Add(item).Entity); await context.SaveChangesAsync(token).ConfigureAwait(false); cached.AddRange(dbEntities); return dbEntities; } } }