2024-11-14 15:17:43 +05:00
|
|
|
|
using Mapster;
|
|
|
|
|
using Microsoft.EntityFrameworkCore;
|
2024-12-16 15:38:46 +05:00
|
|
|
|
using DD.Persistence.Database.Model;
|
|
|
|
|
using DD.Persistence.Models;
|
|
|
|
|
using DD.Persistence.Repositories;
|
2024-11-14 15:17:43 +05:00
|
|
|
|
|
2024-12-16 15:38:46 +05:00
|
|
|
|
namespace DD.Persistence.Repository.Repositories;
|
2024-11-15 16:29:15 +05:00
|
|
|
|
public class TimeSeriesDataRepository<TEntity, TDto> : ITimeSeriesDataRepository<TDto>
|
2024-11-18 14:22:09 +05:00
|
|
|
|
where TEntity : class, ITimestampedData, new()
|
2024-11-14 15:17:43 +05:00
|
|
|
|
where TDto : class, ITimeSeriesAbstractDto, new()
|
|
|
|
|
{
|
2024-12-10 10:43:12 +05:00
|
|
|
|
private readonly DbContext db;
|
2024-11-14 15:17:43 +05:00
|
|
|
|
|
|
|
|
|
public TimeSeriesDataRepository(DbContext db)
|
|
|
|
|
{
|
|
|
|
|
this.db = db;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-15 16:29:15 +05:00
|
|
|
|
protected virtual IQueryable<TEntity> GetQueryReadOnly() => this.db.Set<TEntity>();
|
2024-11-14 15:17:43 +05:00
|
|
|
|
|
2024-11-21 17:02:36 +05:00
|
|
|
|
public virtual async Task<DatesRangeDto?> GetDatesRange(CancellationToken token)
|
2024-11-14 15:17:43 +05:00
|
|
|
|
{
|
2024-11-18 14:22:09 +05:00
|
|
|
|
var query = GetQueryReadOnly();
|
|
|
|
|
var minDate = await query.MinAsync(o => o.Date, token);
|
|
|
|
|
var maxDate = await query.MaxAsync(o => o.Date, token);
|
|
|
|
|
|
|
|
|
|
return new DatesRangeDto
|
|
|
|
|
{
|
|
|
|
|
From = minDate,
|
|
|
|
|
To = maxDate
|
|
|
|
|
};
|
2024-11-14 15:17:43 +05:00
|
|
|
|
}
|
|
|
|
|
|
2024-11-19 11:32:56 +05:00
|
|
|
|
public virtual async Task<IEnumerable<TDto>> GetGtDate(DateTimeOffset date, CancellationToken token)
|
2024-11-14 15:17:43 +05:00
|
|
|
|
{
|
2024-11-18 14:22:09 +05:00
|
|
|
|
var query = this.db.Set<TEntity>().Where(e => e.Date > date);
|
|
|
|
|
var entities = await query.ToArrayAsync(token);
|
|
|
|
|
|
|
|
|
|
var dtos = entities.Select(e => e.Adapt<TDto>());
|
|
|
|
|
|
|
|
|
|
return dtos;
|
2024-11-14 15:17:43 +05:00
|
|
|
|
}
|
|
|
|
|
|
2024-12-04 14:18:57 +05:00
|
|
|
|
public virtual async Task<int> AddRange(IEnumerable<TDto> dtos, CancellationToken token)
|
2024-11-14 15:17:43 +05:00
|
|
|
|
{
|
|
|
|
|
var entities = dtos.Select(d => d.Adapt<TEntity>());
|
|
|
|
|
|
|
|
|
|
await db.Set<TEntity>().AddRangeAsync(entities, token);
|
|
|
|
|
var result = await db.SaveChangesAsync(token);
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
2024-11-19 17:51:51 +05:00
|
|
|
|
|
2024-11-21 17:02:36 +05:00
|
|
|
|
protected async Task<IEnumerable<TDto>> GetLastAsync(int takeCount, CancellationToken token)
|
2024-11-19 17:51:51 +05:00
|
|
|
|
{
|
|
|
|
|
var query = GetQueryReadOnly()
|
|
|
|
|
.OrderByDescending(e => e.Date)
|
|
|
|
|
.Take(takeCount);
|
|
|
|
|
|
|
|
|
|
var entities = await query.ToArrayAsync(token);
|
|
|
|
|
var dtos = entities.Select(e => e.Adapt<TDto>());
|
|
|
|
|
|
|
|
|
|
return dtos;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-21 17:02:36 +05:00
|
|
|
|
protected async Task<TDto?> GetFirstAsync(CancellationToken token)
|
2024-11-19 17:51:51 +05:00
|
|
|
|
{
|
|
|
|
|
var query = GetQueryReadOnly()
|
|
|
|
|
.OrderBy(e => e.Date);
|
|
|
|
|
|
|
|
|
|
var entity = await query.FirstOrDefaultAsync(token);
|
|
|
|
|
|
2024-11-21 17:02:36 +05:00
|
|
|
|
if (entity == null)
|
2024-11-19 17:51:51 +05:00
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
var dto = entity.Adapt<TDto>();
|
|
|
|
|
return dto;
|
|
|
|
|
}
|
2024-11-21 17:02:36 +05:00
|
|
|
|
|
2024-11-22 16:48:55 +05:00
|
|
|
|
public async virtual Task<IEnumerable<TDto>> GetResampledData(
|
2024-12-09 13:19:55 +05:00
|
|
|
|
DateTimeOffset dateBegin,
|
|
|
|
|
double intervalSec = 600d,
|
2024-11-22 16:48:55 +05:00
|
|
|
|
int approxPointsCount = 1024,
|
|
|
|
|
CancellationToken token = default)
|
2024-11-21 17:02:36 +05:00
|
|
|
|
{
|
2024-11-22 16:48:55 +05:00
|
|
|
|
var dtos = await GetGtDate(dateBegin, token);
|
2024-11-22 15:47:00 +05:00
|
|
|
|
|
2024-11-22 16:48:55 +05:00
|
|
|
|
var dateEnd = dateBegin.AddSeconds(intervalSec);
|
|
|
|
|
dtos = dtos
|
|
|
|
|
.Where(i => i.Date <= dateEnd);
|
2024-11-22 15:47:00 +05:00
|
|
|
|
|
2024-11-22 16:48:55 +05:00
|
|
|
|
var ratio = dtos.Count() / approxPointsCount;
|
|
|
|
|
if (ratio > 1)
|
|
|
|
|
dtos = dtos
|
|
|
|
|
.Where((_, index) => index % ratio == 0);
|
2024-11-22 15:47:00 +05:00
|
|
|
|
|
2024-11-22 16:48:55 +05:00
|
|
|
|
return dtos;
|
2024-11-21 17:02:36 +05:00
|
|
|
|
}
|
2024-11-14 15:17:43 +05:00
|
|
|
|
}
|