From fd276f5a43985e066bf36956e3449f7880c17cac Mon Sep 17 00:00:00 2001 From: Roman Efremov Date: Fri, 17 Jan 2025 17:21:54 +0500 Subject: [PATCH] =?UTF-8?q?=D0=94=D0=BE=D1=80=D0=B0=D0=B1=D0=BE=D1=82?= =?UTF-8?q?=D0=BA=D0=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../TimestampedValuesController.cs | 125 +++--- .../Clients/Base/BaseClient.cs | 4 +- .../Interfaces/ITimestampedValuesClient.cs | 102 ++--- .../Clients/Interfaces/Refit/IRefitClient.cs | 4 + .../Refit/IRefitTimestampedValuesClient.cs | 45 ++- .../Clients/TimestampedValuesClient.cs | 73 ++-- .../Controllers/TechMessagesControllerTest.cs | 2 +- .../TimestampedSetControllerTest.cs | 224 ----------- .../TimestampedValuesControllerTest.cs | 371 ++++++++++++++++++ .../Extensions/IEnumerableExtensions.cs | 15 + .../TimestampedValuesRepository.cs | 224 +++++------ .../RelatedDataCachedRepository.cs | 8 +- .../TimestampedValuesCachedRepository.cs | 55 ++- DD.Persistence/Models/TimestampedValuesDto.cs | 2 +- .../Repositories/IRelatedDataRepository.cs | 12 +- .../ITimestampedValuesRepository.cs | 31 +- 16 files changed, 771 insertions(+), 526 deletions(-) delete mode 100644 DD.Persistence.IntegrationTests/Controllers/TimestampedSetControllerTest.cs create mode 100644 DD.Persistence.IntegrationTests/Controllers/TimestampedValuesControllerTest.cs diff --git a/DD.Persistence.API/Controllers/TimestampedValuesController.cs b/DD.Persistence.API/Controllers/TimestampedValuesController.cs index 2b0b245..c4d498b 100644 --- a/DD.Persistence.API/Controllers/TimestampedValuesController.cs +++ b/DD.Persistence.API/Controllers/TimestampedValuesController.cs @@ -15,26 +15,26 @@ namespace DD.Persistence.API.Controllers; [Route("api/[controller]/{discriminatorId}")] public class TimestampedValuesController : ControllerBase { - private readonly ITimestampedValuesRepository repository; + private readonly ITimestampedValuesRepository timestampedValuesRepository; public TimestampedValuesController(ITimestampedValuesRepository repository) { - this.repository = repository; + this.timestampedValuesRepository = repository; } /// - /// Записать новые данные + /// Записать новые данные. /// Предполагается что данные с одним дискриминатором имеют одинаковую структуру /// /// Дискриминатор (идентификатор) набора - /// + /// /// - /// кол-во затронутых записей [HttpPost] - [ProducesResponseType(typeof(int), (int)HttpStatusCode.OK)] - public async Task AddRange([FromRoute] Guid discriminatorId, [FromBody] IEnumerable sets, CancellationToken token) + [ProducesResponseType(typeof(int), (int)HttpStatusCode.Created)] + public async Task AddRange([FromRoute] Guid discriminatorId, [FromBody] IEnumerable dtos, CancellationToken token) { - var result = await repository.AddRange(discriminatorId, sets, token); + var result = await timestampedValuesRepository.AddRange(discriminatorId, dtos, token); + return Ok(result); } @@ -42,81 +42,100 @@ public class TimestampedValuesController : ControllerBase /// Получение данных с фильтрацией. Значение фильтра null - отключен /// /// Дискриминатор (идентификатор) набора - /// Фильтр позднее даты - /// Фильтр свойств набора. Можно запросить только некоторые свойства из набора + /// Фильтр позднее даты + /// Фильтр свойств набора /// /// /// - /// Фильтрованный набор данных с сортировкой по отметке времени [HttpGet] - [ProducesResponseType(typeof(IEnumerable), (int)HttpStatusCode.OK)] - public async Task Get([FromRoute] Guid discriminatorId, DateTimeOffset? geTimestamp, [FromQuery] IEnumerable? columnNames, int skip, int take, CancellationToken token) + public async Task>> Get([FromRoute] Guid discriminatorId, DateTimeOffset? timestampBegin, [FromQuery] string[]? columnNames, int skip, int take, CancellationToken token) { - var result = await repository.Get(discriminatorId, geTimestamp, columnNames, skip, take, token); - return Ok(result); + var result = await timestampedValuesRepository.Get(discriminatorId, timestampBegin, columnNames, skip, take, token); + + return result.Any() ? Ok(result) : NoContent(); } /// - /// Получить последние данные + /// Получить данные, начиная с заданной отметки времени + /// + /// Дискриминатор (идентификатор) набора + /// Фильтр позднее даты + /// + [HttpGet("gtdate")] + public async Task>> GetGtDate([FromRoute] Guid discriminatorId, DateTimeOffset timestampBegin, CancellationToken token) + { + var result = await timestampedValuesRepository.GetGtDate(discriminatorId, timestampBegin, token); + + return result.Any() ? Ok(result) : NoContent(); + } + + /// + /// Получить данные c начала /// /// Дискриминатор (идентификатор) набора - /// Фильтр свойств набора. Можно запросить только некоторые свойства из набора /// /// - /// Фильтрованный набор данных с сортировкой по отметке времени - [HttpGet("last")] - [ProducesResponseType(typeof(IEnumerable), (int)HttpStatusCode.OK)] - public async Task GetLast([FromRoute] Guid discriminatorId, [FromQuery] IEnumerable? columnNames, int take, CancellationToken token) + [HttpGet("first")] + public async Task>> GetFirst([FromRoute] Guid discriminatorId, int take, CancellationToken token) { - var result = await repository.GetLast(discriminatorId, columnNames, take, token); - return Ok(result); + var result = await timestampedValuesRepository.GetFirst(discriminatorId, take, token); + + return result.Any() ? Ok(result) : NoContent(); } /// - /// Диапазон дат за которые есть данные - /// - /// - /// - /// Дата первой и последней записи - [HttpGet("datesRange")] - [ProducesResponseType(typeof(DatesRangeDto), (int)HttpStatusCode.OK)] - [ProducesResponseType((int)HttpStatusCode.NoContent)] - public async Task GetDatesRange([FromRoute] Guid discriminatorId, CancellationToken token) - { - var result = await repository.GetDatesRange(discriminatorId, token); - return Ok(result); - } - - /// - /// Количество записей по указанному набору в БД. Для пагинации. + /// Получить данные c конца /// /// Дискриминатор (идентификатор) набора + /// /// - /// - [HttpGet("count")] - [ProducesResponseType(typeof(int), (int)HttpStatusCode.OK)] - [ProducesResponseType((int)HttpStatusCode.NoContent)] - public async Task Count([FromRoute] Guid discriminatorId, CancellationToken token) + [HttpGet("last")] + public async Task>> GetLast([FromRoute] Guid discriminatorId, int take, CancellationToken token) { - var result = await repository.Count(discriminatorId, token); - return Ok(result); + var result = await timestampedValuesRepository.GetLast(discriminatorId, take, token); + + return result.Any() ? Ok(result) : NoContent(); } /// /// Получить список объектов с прореживанием, удовлетворяющий диапазону дат /// - /// - /// + /// Дискриминатор (идентификатор) набора + /// Фильтр позднее даты /// /// /// - /// [HttpGet("resampled")] - [ProducesResponseType(typeof(IEnumerable), (int)HttpStatusCode.OK)] - [ProducesResponseType((int)HttpStatusCode.NoContent)] - public async Task GetResampledData([FromRoute] Guid discriminatorId, DateTimeOffset dateBegin, double intervalSec = 600d, int approxPointsCount = 1024, CancellationToken token = default) + public async Task>> GetResampledData([FromRoute] Guid discriminatorId, DateTimeOffset timestampBegin, double intervalSec = 600d, int approxPointsCount = 1024, CancellationToken token = default) { - var result = await repository.GetResampledData(discriminatorId, dateBegin, intervalSec, approxPointsCount, token); + var result = await timestampedValuesRepository.GetResampledData(discriminatorId, timestampBegin, intervalSec, approxPointsCount, token); + + return result.Any() ? Ok(result) : NoContent(); + } + + /// + /// Получить количество записей по указанному набору в БД. Для пагинации + /// + /// Дискриминатор (идентификатор) набора + /// + [HttpGet("count")] + public async Task> Count([FromRoute] Guid discriminatorId, CancellationToken token) + { + var result = await timestampedValuesRepository.Count(discriminatorId, token); + + return Ok(result); + } + + /// + /// Получить диапазон дат, в пределах которых хранятся даные + /// + /// + /// + [HttpGet("datesRange")] + public async Task> GetDatesRange([FromRoute] Guid discriminatorId, CancellationToken token) + { + var result = await timestampedValuesRepository.GetDatesRange(discriminatorId, token); + return Ok(result); } } diff --git a/DD.Persistence.Client/Clients/Base/BaseClient.cs b/DD.Persistence.Client/Clients/Base/BaseClient.cs index 3b13cdb..f9bcdd7 100644 --- a/DD.Persistence.Client/Clients/Base/BaseClient.cs +++ b/DD.Persistence.Client/Clients/Base/BaseClient.cs @@ -12,13 +12,13 @@ public abstract class BaseClient this.logger = logger; } - public async Task ExecuteGetResponse(Func>> getMethod, CancellationToken token) + public async Task ExecuteGetResponse(Func>> getMethod, CancellationToken token) { var response = await getMethod.Invoke().WaitAsync(token); if (response.IsSuccessStatusCode) { - return response.Content; + return response.Content!; } var exception = response.GetPersistenceException(); diff --git a/DD.Persistence.Client/Clients/Interfaces/ITimestampedValuesClient.cs b/DD.Persistence.Client/Clients/Interfaces/ITimestampedValuesClient.cs index ea1f576..23f1ddf 100644 --- a/DD.Persistence.Client/Clients/Interfaces/ITimestampedValuesClient.cs +++ b/DD.Persistence.Client/Clients/Interfaces/ITimestampedValuesClient.cs @@ -10,61 +10,71 @@ namespace DD.Persistence.Client.Clients.Interfaces; /// public interface ITimestampedValuesClient : IDisposable { - /// - /// Записать новые данные - /// - /// - /// - /// - /// - Task AddRange(Guid discriminatorId, IEnumerable sets, CancellationToken token); - - /// - /// Количество записей по указанному набору в БД. Для пагинации - /// - /// - /// - /// - Task Count(Guid discriminatorId, CancellationToken token); - - /// - /// Получение данных с фильтрацией. Значение фильтра null - отключен - /// - /// - /// - /// - /// - /// - /// - /// - Task> Get(Guid discriminatorId, DateTimeOffset? geTimestamp, IEnumerable? columnNames, int skip, int take, CancellationToken token); - - /// - /// Диапазон дат за которые есть данные - /// - /// - /// - /// - Task GetDatesRange(Guid discriminatorId, CancellationToken token); + /// + /// Записать новые данные + /// Предполагается что данные с одним дискриминатором имеют одинаковую структуру + /// + /// Дискриминатор (идентификатор) набора + /// + /// + Task AddRange(Guid discriminatorId, IEnumerable dtos, CancellationToken token); /// - /// Получить последние данные + /// Получить данные с фильтрацией. Значение фильтра null - отключен /// - /// - /// + /// Дискриминатор (идентификатор) набора + /// Фильтр позднее даты + /// Фильтр свойств набора + /// /// /// - /// - Task> GetLast(Guid discriminatorId, IEnumerable? columnNames, int take, CancellationToken token); + Task> Get(Guid discriminatorId, DateTimeOffset? timestampBegin, IEnumerable? columnNames, int skip, int take, CancellationToken token); /// - /// Получить список объектов с прореживанием, удовлетворяющий диапазону дат + /// Получить данные, начиная с заданной отметки времени /// - /// - /// + /// Дискриминатор (идентификатор) набора + /// Фильтр позднее даты + /// + Task> GetGtDate(Guid discriminatorId, DateTimeOffset timestampBegin, CancellationToken token); + + /// + /// Получить данные с начала + /// + /// Дискриминатор (идентификатор) набора + /// + /// + Task> GetFirst(Guid discriminatorId, int take, CancellationToken token); + + /// + /// Получить данные с конца + /// + /// Дискриминатор (идентификатор) набора + /// + /// + Task> GetLast(Guid discriminatorId, int take, CancellationToken token); + + /// + /// Получить данные с прореживанием, удовлетворяющем диапазону дат + /// + /// Дискриминатор (идентификатор) набора + /// /// /// /// - /// - Task> GetResampledData(Guid discriminatorId, DateTimeOffset dateBegin, double intervalSec = 600d, int approxPointsCount = 1024, CancellationToken token = default); + Task> GetResampledData(Guid discriminatorId, DateTimeOffset timestampBegin, double intervalSec = 600d, int approxPointsCount = 1024, CancellationToken token = default); + + /// + /// Количество записей по указанному набору в БД. Для пагинации + /// + /// Дискриминатор (идентификатор) набора + /// + Task Count(Guid discriminatorId, CancellationToken token); + + /// + /// Диапазон дат, в пределах которых осуществляется хранение данных + /// + /// Дискриминатор (идентификатор) набора + /// + Task GetDatesRange(Guid discriminatorId, CancellationToken token); } \ No newline at end of file diff --git a/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitClient.cs b/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitClient.cs index 9568a51..4a140e0 100644 --- a/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitClient.cs +++ b/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitClient.cs @@ -5,6 +5,10 @@ using System.Text; using System.Threading.Tasks; namespace DD.Persistence.Client.Clients.Interfaces.Refit; + +/// +/// Базовый Refit интерфейс +/// public interface IRefitClient { } diff --git a/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitTimestampedValuesClient.cs b/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitTimestampedValuesClient.cs index 2515a57..bd94136 100644 --- a/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitTimestampedValuesClient.cs +++ b/DD.Persistence.Client/Clients/Interfaces/Refit/IRefitTimestampedValuesClient.cs @@ -4,25 +4,58 @@ using Refit; namespace DD.Persistence.Client.Clients.Interfaces.Refit; +/// +/// Refit интерфейс для TimestampedValuesController +/// public interface IRefitTimestampedValuesClient : IRefitClient, IDisposable { private const string baseUrl = "/api/TimestampedValues/{discriminatorId}"; + /// + /// Записать новые данные + /// [Post(baseUrl)] - Task> AddRange(Guid discriminatorId, IEnumerable sets, CancellationToken token); + Task> AddRange(Guid discriminatorId, IEnumerable dtos, CancellationToken token); + /// + /// Получение данных с фильтрацией + /// [Get(baseUrl)] - Task>> Get(Guid discriminatorId, [Query] DateTimeOffset? geTimestamp, [Query] IEnumerable? columnNames, int skip, int take, CancellationToken token); + Task>> Get(Guid discriminatorId, DateTimeOffset? timestampBegin, [Query(CollectionFormat.Multi)] IEnumerable? columnNames, int skip, int take, CancellationToken token); + /// + /// Получить данные, начиная с заданной отметки времени + /// + [Get($"{baseUrl}/gtdate")] + Task>> GetGtDate(Guid discriminatorId, DateTimeOffset timestampBegin, CancellationToken token); + + /// + /// Получить данные c начала + /// + [Get($"{baseUrl}/first")] + Task>> GetFirst(Guid discriminatorId, int take, CancellationToken token); + + /// + /// Получить данные c конца + /// [Get($"{baseUrl}/last")] - Task>> GetLast(Guid discriminatorId, [Query] IEnumerable? columnNames, int take, CancellationToken token); + Task>> GetLast(Guid discriminatorId, int take, CancellationToken token); + /// + /// Получить список объектов с прореживанием, удовлетворяющий диапазону временных отметок + /// + [Get($"{baseUrl}/resampled")] + Task>> GetResampledData(Guid discriminatorId, DateTimeOffset timestampBegin, double intervalSec = 600d, int approxPointsCount = 1024, CancellationToken token = default); + + /// + /// Получить количество записей по указанному набору в БД. Для пагинации + /// [Get($"{baseUrl}/count")] Task> Count(Guid discriminatorId, CancellationToken token); + /// + /// Получить диапазон дат, в пределах которых хранятся даные + /// [Get($"{baseUrl}/datesRange")] Task> GetDatesRange(Guid discriminatorId, CancellationToken token); - - [Get($"{baseUrl}/resampled")] - Task>> GetResampledData(Guid discriminatorId, DateTimeOffset dateBegin, double intervalSec = 600d, int approxPointsCount = 1024, CancellationToken token = default); } diff --git a/DD.Persistence.Client/Clients/TimestampedValuesClient.cs b/DD.Persistence.Client/Clients/TimestampedValuesClient.cs index d32e9b9..19fce3f 100644 --- a/DD.Persistence.Client/Clients/TimestampedValuesClient.cs +++ b/DD.Persistence.Client/Clients/TimestampedValuesClient.cs @@ -1,20 +1,24 @@ -using Microsoft.Extensions.Logging; -using DD.Persistence.Client.Clients.Base; +using DD.Persistence.Client.Clients.Base; using DD.Persistence.Client.Clients.Interfaces; using DD.Persistence.Client.Clients.Interfaces.Refit; using DD.Persistence.Models; using DD.Persistence.Models.Common; +using Microsoft.Extensions.Logging; namespace DD.Persistence.Client.Clients; + +/// public class TimestampedValuesClient : BaseClient, ITimestampedValuesClient { private readonly IRefitTimestampedValuesClient refitTimestampedSetClient; + /// public TimestampedValuesClient(IRefitClientFactory refitTimestampedSetClientFactory, ILogger logger) : base(logger) { this.refitTimestampedSetClient = refitTimestampedSetClientFactory.Create(); } + /// public async Task AddRange(Guid discriminatorId, IEnumerable sets, CancellationToken token) { var result = await ExecutePostResponse( @@ -23,38 +27,42 @@ public class TimestampedValuesClient : BaseClient, ITimestampedValuesClient return result; } + /// public async Task> Get(Guid discriminatorId, DateTimeOffset? geTimestamp, IEnumerable? columnNames, int skip, int take, CancellationToken token) { var result = await ExecuteGetResponse( async () => await refitTimestampedSetClient.Get(discriminatorId, geTimestamp, columnNames, skip, take, token), token); + return result; + } - return result!; - } + /// + public async Task> GetGtDate(Guid discriminatorId, DateTimeOffset timestampBegin, CancellationToken token) + { + var result = await ExecuteGetResponse( + async () => await refitTimestampedSetClient.GetGtDate(discriminatorId, timestampBegin, token), token); - public async Task> GetLast(Guid discriminatorId, IEnumerable? columnNames, int take, CancellationToken token) + return result; + } + + /// + public async Task> GetFirst(Guid discriminatorId, int take, CancellationToken token) + { + var result = await ExecuteGetResponse( + async () => await refitTimestampedSetClient.GetFirst(discriminatorId, take, token), token); + + return result; + } + + /// + public async Task> GetLast(Guid discriminatorId, int take, CancellationToken token) { var result = await ExecuteGetResponse( - async () => await refitTimestampedSetClient.GetLast(discriminatorId, columnNames, take, token), token); + async () => await refitTimestampedSetClient.GetLast(discriminatorId, take, token), token); - return result!; - } - - public async Task Count(Guid discriminatorId, CancellationToken token) - { - var result = await ExecuteGetResponse( - async () => await refitTimestampedSetClient.Count(discriminatorId, token), token); - - return result; - } - - public async Task GetDatesRange(Guid discriminatorId, CancellationToken token) - { - var result = await ExecuteGetResponse( - async () => await refitTimestampedSetClient.GetDatesRange(discriminatorId, token), token); - - return result; + return result; } + /// public async Task> GetResampledData(Guid discriminatorId, DateTimeOffset dateBegin, double intervalSec = 600, int approxPointsCount = 1024, CancellationToken token = default) { var result = await ExecuteGetResponse( @@ -63,6 +71,25 @@ public class TimestampedValuesClient : BaseClient, ITimestampedValuesClient return result; } + /// + public async Task Count(Guid discriminatorId, CancellationToken token) + { + var result = await ExecuteGetResponse( + async () => await refitTimestampedSetClient.Count(discriminatorId, token), token); + + return result; + } + + /// + public async Task GetDatesRange(Guid discriminatorId, CancellationToken token) + { + var result = await ExecuteGetResponse( + async () => await refitTimestampedSetClient.GetDatesRange(discriminatorId, token), token); + + return result; + } + + /// public void Dispose() { refitTimestampedSetClient.Dispose(); diff --git a/DD.Persistence.IntegrationTests/Controllers/TechMessagesControllerTest.cs b/DD.Persistence.IntegrationTests/Controllers/TechMessagesControllerTest.cs index 51afc02..7974470 100644 --- a/DD.Persistence.IntegrationTests/Controllers/TechMessagesControllerTest.cs +++ b/DD.Persistence.IntegrationTests/Controllers/TechMessagesControllerTest.cs @@ -15,7 +15,7 @@ namespace DD.Persistence.IntegrationTests.Controllers { public class TechMessagesControllerTest : BaseIntegrationTest { - private static readonly string SystemCacheKey = $"{typeof(Database.Entity.DataSourceSystem).FullName}CacheKey"; + private static readonly string SystemCacheKey = $"{typeof(DataSourceSystem).FullName}CacheKey"; private readonly ITechMessagesClient techMessagesClient; private readonly IMemoryCache memoryCache; public TechMessagesControllerTest(WebAppFactoryFixture factory) : base(factory) diff --git a/DD.Persistence.IntegrationTests/Controllers/TimestampedSetControllerTest.cs b/DD.Persistence.IntegrationTests/Controllers/TimestampedSetControllerTest.cs deleted file mode 100644 index 97882e5..0000000 --- a/DD.Persistence.IntegrationTests/Controllers/TimestampedSetControllerTest.cs +++ /dev/null @@ -1,224 +0,0 @@ -using Microsoft.Extensions.DependencyInjection; -using DD.Persistence.Client; -using DD.Persistence.Client.Clients.Interfaces; -using DD.Persistence.Models; -using Xunit; -using DD.Persistence.Client.Clients.Interfaces.Refit; -using DD.Persistence.Client.Clients; -using Microsoft.Extensions.Logging; -using System.Text.Json; -using System.Text; -using Microsoft.Extensions.Primitives; -using System.Dynamic; -using Newtonsoft.Json.Linq; - -namespace DD.Persistence.IntegrationTests.Controllers; -public class TimestampedSetControllerTest : BaseIntegrationTest -{ - private readonly ITimestampedValuesClient client; - - public TimestampedSetControllerTest(WebAppFactoryFixture factory) : base(factory) - { - var refitClientFactory = scope.ServiceProvider - .GetRequiredService>(); - var logger = scope.ServiceProvider.GetRequiredService>(); - - client = scope.ServiceProvider - .GetRequiredService(); - } - - [Fact] - public async Task InsertRange() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - IEnumerable testSets = Generate(10, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); - - // act - var response = await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - - // assert - Assert.Equal(testSets.Count(), response); - } - - [Fact] - public async Task Get_without_filter() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - int count = 10; - IEnumerable testSets = Generate(count, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); - await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - - // act - var response = await client.Get(idDiscriminator, null, null, 0, int.MaxValue, CancellationToken.None); - - // assert - Assert.NotNull(response); - Assert.Equal(count, response.Count()); - } - - [Fact] - public async Task Get_with_filter_props() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - int count = 10; - IEnumerable testSets = Generate(count, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); - await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - string[] props = ["A"]; - - // act - var response = await client.Get(idDiscriminator, null, props, 0, int.MaxValue, new CancellationToken()); - - // assert - Assert.NotNull(response); - Assert.Equal(count, response.Count()); - foreach (var item in response) - { - Assert.Single(item.Values!); - var kv = item.Values!.First(); - Assert.Equal("A", ((KeyValuePair) kv).Key); - } - } - - [Fact] - public async Task Get_geDate() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - int count = 10; - var dateMin = DateTimeOffset.Now; - var dateMax = DateTimeOffset.Now.AddSeconds(count); - IEnumerable testSets = Generate(count, dateMin.ToOffset(TimeSpan.FromHours(7))); - var insertResponse = await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - var tail = testSets.OrderBy(t => t.Timestamp).Skip(count / 2).Take(int.MaxValue); - var geDate = tail.First().Timestamp; - var tolerance = TimeSpan.FromSeconds(1); - var expectedCount = tail.Count(); - - // act - var response = await client.Get(idDiscriminator, geDate, null, 0, int.MaxValue, CancellationToken.None); - - // assert - Assert.NotNull(response); - Assert.Equal(expectedCount, response.Count()); - var minDate = response.Min(t => t.Timestamp); - Assert.Equal(geDate, geDate, tolerance); - } - - [Fact] - public async Task Get_with_skip_take() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - int count = 10; - IEnumerable testSets = Generate(count, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); - await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - var expectedCount = count / 2; - - // act - var response = await client.Get(idDiscriminator, null, null, 2, expectedCount, new CancellationToken()); - - // assert - Assert.NotNull(response); - Assert.Equal(expectedCount, response.Count()); - } - - - [Fact] - public async Task Get_with_big_skip_take() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - var expectedCount = 1; - int count = 10 + expectedCount; - IEnumerable testSets = Generate(count, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); - await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - - // act - var response = await client.Get(idDiscriminator, null, null, count - expectedCount, count, new CancellationToken()); - - // assert - Assert.NotNull(response); - Assert.Equal(expectedCount, response.Count()); - } - - [Fact] - public async Task GetLast() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - int count = 10; - IEnumerable testSets = Generate(count, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); - await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - var expectedCount = 8; - - // act - var response = await client.GetLast(idDiscriminator, null, expectedCount, new CancellationToken()); - - // assert - Assert.NotNull(response); - Assert.Equal(expectedCount, response.Count()); - } - - [Fact] - public async Task GetDatesRange() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - int count = 10; - var dateMin = DateTimeOffset.Now; - var dateMax = DateTimeOffset.Now.AddSeconds(count - 1); - IEnumerable testSets = Generate(count, dateMin.ToOffset(TimeSpan.FromHours(7))); - await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - var tolerance = TimeSpan.FromSeconds(1); - - // act - var response = await client.GetDatesRange(idDiscriminator, new CancellationToken()); - - // assert - Assert.NotNull(response); - Assert.Equal(dateMin, response.From, tolerance); - Assert.Equal(dateMax, response.To, tolerance); - } - - [Fact] - public async Task Count() - { - // arrange - Guid idDiscriminator = Guid.NewGuid(); - int count = 144; - IEnumerable testSets = Generate(count, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); - await client.AddRange(idDiscriminator, testSets, CancellationToken.None); - - // act - var response = await client.Count(idDiscriminator, new CancellationToken()); - - // assert - Assert.Equal(count, response); - } - - private static IEnumerable Generate(int n, DateTimeOffset from) - { - var result = new List(); - for (int i = 0; i < n; i++) - { - var t = new object[] { - new { A = i }, - new { B = i * 1.1 }, - new { C = $"Any{i}" }, - new { D = DateTimeOffset.Now } - }; - string jsonString = JsonSerializer.Serialize(t); - var values = JsonSerializer.Deserialize(jsonString); - - - yield return new TimestampedValuesDto() - { - Timestamp = from.AddSeconds(i), - Values = values - }; - } - } -} diff --git a/DD.Persistence.IntegrationTests/Controllers/TimestampedValuesControllerTest.cs b/DD.Persistence.IntegrationTests/Controllers/TimestampedValuesControllerTest.cs new file mode 100644 index 0000000..b166631 --- /dev/null +++ b/DD.Persistence.IntegrationTests/Controllers/TimestampedValuesControllerTest.cs @@ -0,0 +1,371 @@ +using DD.Persistence.Client; +using DD.Persistence.Client.Clients; +using DD.Persistence.Client.Clients.Interfaces; +using DD.Persistence.Client.Clients.Interfaces.Refit; +using DD.Persistence.Database.Entity; +using DD.Persistence.Models; +using Microsoft.Extensions.Caching.Memory; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Logging; +using System.Text.Json; +using Xunit; + +namespace DD.Persistence.IntegrationTests.Controllers; +public class TimestampedValuesControllerTest : BaseIntegrationTest +{ + private static readonly string SystemCacheKey = $"{typeof(ValuesIdentity).FullName}CacheKey"; + private readonly ITimestampedValuesClient timestampedValuesClient; + private readonly IMemoryCache memoryCache; + + public TimestampedValuesControllerTest(WebAppFactoryFixture factory) : base(factory) + { + var refitClientFactory = scope.ServiceProvider + .GetRequiredService>(); + var logger = scope.ServiceProvider.GetRequiredService>(); + + timestampedValuesClient = scope.ServiceProvider + .GetRequiredService(); + memoryCache = scope.ServiceProvider.GetRequiredService(); + } + + [Fact] + public async Task AddRange_returns_success() + { + var discriminatorId = Guid.NewGuid(); + + await AddRange(discriminatorId); + } + + [Fact] + public async Task Get_returns_success() + { + //arrange + Cleanup(); + + var discriminatorId = Guid.NewGuid(); + + //act + var response = await timestampedValuesClient.Get(discriminatorId, null, null, 0, 1, CancellationToken.None); + + //assert + Assert.Null(response); + } + + [Fact] + + public async Task Get_AfterSave_returns_success() + { + //arrange + Cleanup(); + + var discriminatorId = Guid.NewGuid(); + var timestampBegin = DateTimeOffset.UtcNow.AddDays(-1); + var columnNames = new List() { "A", "C" }; + var skip = 5; + var take = 5; + + var dtos = await AddRange(discriminatorId); + + //act + var response = await timestampedValuesClient.Get(discriminatorId, timestampBegin, columnNames, skip, take, CancellationToken.None); + + //assert + Assert.NotNull(response); + Assert.NotEmpty(response); + + var actualCount = response.Count(); + Assert.Equal(take, actualCount); + + var actualColumnNames = response.SelectMany(e => e.Values.Keys).Distinct().ToList(); + Assert.Equal(columnNames, actualColumnNames); + + var expectedValueKind = JsonValueKind.Number; + var actualValueKind = ((JsonElement) response.First().Values["A"]).ValueKind; + Assert.Equal(expectedValueKind, actualValueKind); + + expectedValueKind = JsonValueKind.String; + actualValueKind = ((JsonElement)response.First().Values["C"]).ValueKind; + Assert.Equal(expectedValueKind, actualValueKind); + } + + [Fact] + public async Task GetGtDate_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var timestampBegin = DateTimeOffset.UtcNow.AddDays(-1); + + //act + var response = await timestampedValuesClient.GetGtDate(discriminatorId, timestampBegin, CancellationToken.None); + + //assert + Assert.Null(response); + } + + [Fact] + public async Task GetGtDate_AfterSave_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var dtos = await AddRange(discriminatorId); + var timestampBegin = DateTimeOffset.UtcNow.AddSeconds(-5); + + //act + var response = await timestampedValuesClient.GetGtDate(discriminatorId, timestampBegin, CancellationToken.None); + + //assert + Assert.NotNull(response); + Assert.NotEmpty(response); + + var expectedCount = dtos.Count(dto => dto.Timestamp.ToUniversalTime() > timestampBegin); + var actualCount = response.Count(); + Assert.Equal(expectedCount, actualCount); + } + + [Fact] + public async Task GetFirst_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var take = 1; + + //act + var response = await timestampedValuesClient.GetFirst(discriminatorId, take, CancellationToken.None); + + //assert + Assert.Null(response); + } + + [Fact] + public async Task GetFirst_AfterSave_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var dtos = await AddRange(discriminatorId); + var take = 1; + + //act + var response = await timestampedValuesClient.GetFirst(discriminatorId, take, CancellationToken.None); + + //assert + Assert.NotNull(response); + Assert.NotEmpty(response); + + var expectedTimestampString = dtos + .OrderBy(dto => dto.Timestamp) + .First().Timestamp + .ToUniversalTime() + .ToString(); + var actualTimestampString = response + .First().Timestamp + .ToUniversalTime() + .ToString(); + Assert.Equal(expectedTimestampString, actualTimestampString); + } + + [Fact] + public async Task GetLast_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var take = 1; + + //act + var response = await timestampedValuesClient.GetLast(discriminatorId, take, CancellationToken.None); + + //assert + Assert.Null(response); + } + + [Fact] + public async Task GetLast_AfterSave_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var dtos = await AddRange(discriminatorId); + var take = 1; + + //act + var response = await timestampedValuesClient.GetLast(discriminatorId, take, CancellationToken.None); + + //assert + Assert.NotNull(response); + Assert.NotEmpty(response); + + var expectedTimestampString = dtos + .OrderByDescending(dto => dto.Timestamp) + .First().Timestamp + .ToUniversalTime() + .ToString(); + var actualTimestampString = response + .First().Timestamp + .ToUniversalTime() + .ToString(); + Assert.Equal(expectedTimestampString, actualTimestampString); + } + + [Fact] + public async Task GetResampledData_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var timestampBegin = DateTimeOffset.UtcNow; + + //act + var response = await timestampedValuesClient.GetResampledData(discriminatorId, timestampBegin); + + //assert + Assert.Null(response); + } + + [Fact] + public async Task GetResampledData_AfterSave_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var count = 2048; + var timestampBegin = DateTimeOffset.UtcNow; + var dtos = await AddRange(discriminatorId, count); + + + //act + var response = await timestampedValuesClient.GetResampledData(discriminatorId, timestampBegin, count); + + //assert + Assert.NotNull(response); + Assert.NotEmpty(response); + + var expectedCount = count / 2; + var actualCount = response.Count(); + Assert.Equal(expectedCount, actualCount); + } + + [Fact] + public async Task Count_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + + //act + var response = await timestampedValuesClient.Count(discriminatorId, CancellationToken.None); + + //assert + Assert.Equal(0, response); + } + + [Fact] + public async Task Count_AfterSave_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var dtos = await AddRange(discriminatorId); + + //act + var response = await timestampedValuesClient.Count(discriminatorId, CancellationToken.None); + + //assert + var expectedCount = dtos.Count(); + Assert.Equal(expectedCount, response); + } + + [Fact] + public async Task GetDatesRange_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + + //act + var response = await timestampedValuesClient.GetDatesRange(discriminatorId, CancellationToken.None); + + //assert + Assert.Null(response); + } + + [Fact] + public async Task GetDatesRange_AfterSave_returns_success() + { + //arrange + Cleanup(); + var discriminatorId = Guid.NewGuid(); + var dtos = await AddRange(discriminatorId); + + //act + var response = await timestampedValuesClient.GetDatesRange(discriminatorId, CancellationToken.None); + + //assert + Assert.NotNull(response); + + var expectedDateFromString = dtos + .OrderBy(dto => dto.Timestamp) + .First().Timestamp + .ToUniversalTime() + .ToString(); + var actualDateFromString = response.From + .ToUniversalTime() + .ToString(); + Assert.Equal(expectedDateFromString, actualDateFromString); + + var expectedDateToString = dtos + .OrderByDescending(dto => dto.Timestamp) + .First().Timestamp + .ToUniversalTime() + .ToString(); + var actualDateToString = response.To + .ToUniversalTime() + .ToString(); + Assert.Equal(expectedDateToString, actualDateToString); + } + + private async Task> AddRange(Guid discriminatorId, int countToCreate = 10) + { + // arrange + IEnumerable generatedDtos = Generate(countToCreate, DateTimeOffset.Now.ToOffset(TimeSpan.FromHours(7))); + + // act + var response = await timestampedValuesClient.AddRange(discriminatorId, generatedDtos, CancellationToken.None); + + // assert + Assert.Equal(generatedDtos.Count(), response); + + return generatedDtos; + } + + private static IEnumerable Generate(int countToCreate, DateTimeOffset from) + { + var result = new List(); + for (int i = 0; i < countToCreate; i++) + { + var values = new Dictionary() + { + { "A", i }, + { "B", i * 1.1 }, + { "C", $"Any{i}" }, + { "D", DateTimeOffset.Now }, + }; + + yield return new TimestampedValuesDto() + { + Timestamp = from.AddSeconds(i), + Values = values + }; + } + } + + private void Cleanup() + { + memoryCache.Remove(SystemCacheKey); + dbContext.CleanupDbSet(); + dbContext.CleanupDbSet(); + } +} diff --git a/DD.Persistence.Repository/Extensions/IEnumerableExtensions.cs b/DD.Persistence.Repository/Extensions/IEnumerableExtensions.cs index 270107f..e307513 100644 --- a/DD.Persistence.Repository/Extensions/IEnumerableExtensions.cs +++ b/DD.Persistence.Repository/Extensions/IEnumerableExtensions.cs @@ -14,4 +14,19 @@ public static class IEnumerableExtensions action(item); } } + + public static bool IsNullOrEmpty(this IEnumerable? enumerable) + { + if (enumerable == null) + { + return true; + } + + var collection = enumerable as ICollection; + if (collection != null) + { + return collection.Count < 1; + } + return !enumerable.Any(); + } } diff --git a/DD.Persistence.Repository/Repositories/TimestampedValuesRepository.cs b/DD.Persistence.Repository/Repositories/TimestampedValuesRepository.cs index 9be62db..9e1597c 100644 --- a/DD.Persistence.Repository/Repositories/TimestampedValuesRepository.cs +++ b/DD.Persistence.Repository/Repositories/TimestampedValuesRepository.cs @@ -3,12 +3,7 @@ using DD.Persistence.Models; using DD.Persistence.Models.Common; using DD.Persistence.Repositories; using DD.Persistence.Repository.Extensions; -using Mapster; using Microsoft.EntityFrameworkCore; -using Newtonsoft.Json.Linq; -using System.Linq; -using System.Text.Json; -using System.Text.Json.Nodes; namespace DD.Persistence.Repository.Repositories; public class TimestampedValuesRepository : ITimestampedValuesRepository @@ -25,54 +20,19 @@ public class TimestampedValuesRepository : ITimestampedValuesRepository protected virtual IQueryable GetQueryReadOnly() => this.db.Set() .Include(e => e.ValuesIdentity); - public virtual async Task GetDatesRange(Guid discriminatorId, CancellationToken token) - { - var query = GetQueryReadOnly() - .GroupBy(entity => entity.DiscriminatorId) - .Select(group => new - { - Min = group.Min(entity => entity.Timestamp), - Max = group.Max(entity => entity.Timestamp), - }); - - var item = await query.FirstOrDefaultAsync(token); - if (item is null) - return null; - - return new DatesRangeDto - { - From = item.Min, - To = item.Max, - }; - } - - public virtual async Task> GetGtDate(Guid discriminatorId, DateTimeOffset date, CancellationToken token) - { - var query = GetQueryReadOnly().Where(e => e.Timestamp > date); - var entities = await query.ToArrayAsync(token); - - var dtos = entities.Select(e => e.Adapt()); - - return dtos; - } - - public virtual async Task AddRange(Guid discriminatorId, IEnumerable dtos, CancellationToken token) + public async virtual Task AddRange(Guid discriminatorId, IEnumerable dtos, CancellationToken token) { var timestampedValuesEntities = new List(); foreach (var dto in dtos) { - var values = dto.Values - .SelectMany(v => JsonSerializer.Deserialize>(v.ToString()!)!) - .ToDictionary(); - - var keys = values.Keys.ToArray(); + var keys = dto.Values.Keys.ToArray(); await CreateValuesIdentityIfNotExist(discriminatorId, keys, token); var timestampedValuesEntity = new TimestampedValues() { DiscriminatorId = discriminatorId, Timestamp = dto.Timestamp.ToUniversalTime(), - Values = values.Values.ToArray() + Values = dto.Values.Values.ToArray() }; timestampedValuesEntities.Add(timestampedValuesEntity); } @@ -84,32 +44,54 @@ public class TimestampedValuesRepository : ITimestampedValuesRepository return result; } - protected async Task> GetLastAsync(int takeCount, CancellationToken token) + public async virtual Task> Get(Guid discriminatorId, DateTimeOffset? timestampBegin, IEnumerable? columnNames, int skip, int take, CancellationToken token) + { + var query = GetQueryReadOnly() + .Where(entity => entity.DiscriminatorId == discriminatorId); + + // Фильтрация по дате + if (timestampBegin.HasValue) + { + query = ApplyGeTimestamp(query, timestampBegin.Value); + } + + query = query + .OrderBy(item => item.Timestamp) + .Skip(skip) + .Take(take); + var data = await Materialize(discriminatorId, query, token); + + // Фильтрация по запрашиваемым полям + if (!columnNames.IsNullOrEmpty()) + { + data = ReduceSetColumnsByNames(data, columnNames!); + } + + return data; + } + + public async virtual Task> GetFirst(Guid discriminatorId, int takeCount, CancellationToken token) + { + var query = GetQueryReadOnly() + .OrderBy(e => e.Timestamp) + .Take(takeCount); + + var dtos = await Materialize(discriminatorId, query, token); + + return dtos; + } + + public async virtual Task> GetLast(Guid discriminatorId, int takeCount, CancellationToken token) { var query = GetQueryReadOnly() .OrderByDescending(e => e.Timestamp) .Take(takeCount); - var entities = await query.ToArrayAsync(token); - var dtos = entities.Select(e => e.Adapt()); + var dtos = await Materialize(discriminatorId, query, token); return dtos; } - protected async Task GetFirstAsync(CancellationToken token) - { - var query = GetQueryReadOnly() - .OrderBy(e => e.Timestamp); - - var entity = await query.FirstOrDefaultAsync(token); - - if (entity == null) - return null; - - var dto = entity.Adapt(); - return dto; - } - public async virtual Task> GetResampledData( Guid discriminatorId, DateTimeOffset dateBegin, @@ -131,99 +113,103 @@ public class TimestampedValuesRepository : ITimestampedValuesRepository return dtos; } - public async Task> Get(Guid discriminatorId, DateTimeOffset? geTimestamp, IEnumerable? columnNames, int skip, int take, CancellationToken token) + public async virtual Task> GetGtDate(Guid discriminatorId, DateTimeOffset timestampBegin, CancellationToken token) { - var dbSet = db.Set(); - var query = dbSet.Where(entity => entity.DiscriminatorId == discriminatorId); + var query = GetQueryReadOnly() + .Where(e => e.Timestamp > timestampBegin); - if (geTimestamp.HasValue) - query = ApplyGeTimestamp(query, geTimestamp.Value); + var dtos = await Materialize(discriminatorId, query, token); - query = query - .OrderBy(item => item.Timestamp) - .Skip(skip) - .Take(take); - - var data = await Materialize(discriminatorId, query, token); - - if (columnNames is not null && columnNames.Any()) - data = ReduceSetColumnsByNames(data, columnNames); - - return data; + return dtos; } - public async Task> GetLast(Guid discriminatorId, IEnumerable? columnNames, int take, CancellationToken token) + public async virtual Task GetDatesRange(Guid discriminatorId, CancellationToken token) { - var dbSet = db.Set(); - var query = dbSet.Where(entity => entity.DiscriminatorId == discriminatorId); + var query = GetQueryReadOnly() + .GroupBy(entity => entity.DiscriminatorId) + .Select(group => new + { + Min = group.Min(entity => entity.Timestamp), + Max = group.Max(entity => entity.Timestamp), + }); - query = query.OrderByDescending(entity => entity.Timestamp) - .Take(take) - .OrderBy(entity => entity.Timestamp); + var item = await query.FirstOrDefaultAsync(token); + if (item is null) + { + return null; + } - var data = await Materialize(discriminatorId, query, token); + var dto = new DatesRangeDto + { + From = item.Min, + To = item.Max, + }; - if (columnNames is not null && columnNames.Any()) - data = ReduceSetColumnsByNames(data, columnNames); - - return data; + return dto; } - public Task Count(Guid discriminatorId, CancellationToken token) + public virtual Task Count(Guid discriminatorId, CancellationToken token) { var dbSet = db.Set(); var query = dbSet.Where(entity => entity.DiscriminatorId == discriminatorId); + return query.CountAsync(token); } private async Task> Materialize(Guid discriminatorId, IQueryable query, CancellationToken token) { - var dtoQuery = query.Select(entity => new TimestampedValuesDto() - { - Timestamp = entity.Timestamp, - Values = entity.Values - }); + var valuesIdentities = await relatedDataRepository.Get(token); + var valuesIdentity = valuesIdentities? + .FirstOrDefault(e => e.DiscriminatorId == discriminatorId); + if (valuesIdentity == null) + return []; - var dtos = await dtoQuery.ToArrayAsync(token); - foreach(var dto in dtos) + var entities = await query.ToArrayAsync(token); + + var dtos = entities.Select(entity => { - var valuesIdentities = await relatedDataRepository.Get(token); - var valuesIdentity = valuesIdentities? - .FirstOrDefault(e => e.DiscriminatorId == discriminatorId); - if (valuesIdentity == null) - return []; // ToDo: какая логика должна быть? + var dto = new TimestampedValuesDto() + { + Timestamp = entity.Timestamp.ToUniversalTime() + }; for (var i = 0; i < valuesIdentity.Identity.Count(); i++) { var key = valuesIdentity.Identity[i]; - var value = dto.Values[i]; + var value = entity.Values[i]; - dto.Values[i] = new { key = value }; // ToDo: вывод? + dto.Values.Add(key, value); } - } + + return dto; + }); return dtos; } - private IQueryable ApplyGeTimestamp(IQueryable query, DateTimeOffset geTimestamp) + private IQueryable ApplyGeTimestamp(IQueryable query, DateTimeOffset timestampBegin) { - var geTimestampUtc = geTimestamp.ToUniversalTime(); - return query.Where(entity => entity.Timestamp >= geTimestampUtc); + var geTimestampUtc = timestampBegin.ToUniversalTime(); + + var result = query + .Where(entity => entity.Timestamp >= geTimestampUtc); + + return result; } - private static IEnumerable ReduceSetColumnsByNames(IEnumerable query, IEnumerable columnNames) + private IEnumerable ReduceSetColumnsByNames(IEnumerable dtos, IEnumerable columnNames) { - var newQuery = query; - //.Select(entity => new TimestampedValuesDto() - //{ - // Timestamp = entity.Timestamp, - // Values = entity.Values? - // .Where(prop => columnNames.Contains( - // JsonSerializer.Deserialize>(prop.ToString()!)? - // .FirstOrDefault().Key - // )).ToArray() - //}); - return newQuery; + var result = dtos.Select(dto => + { + var reducedValues = dto.Values + .Where(v => columnNames.Contains(v.Key)) + .ToDictionary(); + dto.Values = reducedValues; + + return dto; + }); + + return result; } private async Task CreateValuesIdentityIfNotExist(Guid discriminatorId, string[] keys, CancellationToken token) @@ -232,7 +218,7 @@ public class TimestampedValuesRepository : ITimestampedValuesRepository var valuesIdentity = valuesIdentities? .FirstOrDefault(e => e.DiscriminatorId == discriminatorId); - if (valuesIdentity == null) + if (valuesIdentity is null) { valuesIdentity = new ValuesIdentityDto() { diff --git a/DD.Persistence.Repository/RepositoriesCached/RelatedDataCachedRepository.cs b/DD.Persistence.Repository/RepositoriesCached/RelatedDataCachedRepository.cs index bc680f0..6f596ba 100644 --- a/DD.Persistence.Repository/RepositoriesCached/RelatedDataCachedRepository.cs +++ b/DD.Persistence.Repository/RepositoriesCached/RelatedDataCachedRepository.cs @@ -1,16 +1,14 @@ -using Microsoft.EntityFrameworkCore; +using DD.Persistence.Repository.Repositories; +using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.Caching.Memory; -using DD.Persistence.Models; -using DD.Persistence.Repository.Repositories; namespace DD.Persistence.Repository.RepositoriesCached; public class RelatedDataCachedRepository : RelatedDataRepository where TEntity : class, new() where TDto : class, new() { - private static readonly string SystemCacheKey = $"{typeof(Database.Entity.DataSourceSystem).FullName}CacheKey"; + private static readonly string SystemCacheKey = $"{typeof(TEntity).FullName}CacheKey"; private readonly IMemoryCache memoryCache; - private const int CacheExpirationInMinutes = 60; private readonly TimeSpan? AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(60); public RelatedDataCachedRepository(DbContext db, IMemoryCache memoryCache) : base(db) diff --git a/DD.Persistence.Repository/RepositoriesCached/TimestampedValuesCachedRepository.cs b/DD.Persistence.Repository/RepositoriesCached/TimestampedValuesCachedRepository.cs index d1eba07..7bdaa98 100644 --- a/DD.Persistence.Repository/RepositoriesCached/TimestampedValuesCachedRepository.cs +++ b/DD.Persistence.Repository/RepositoriesCached/TimestampedValuesCachedRepository.cs @@ -1,43 +1,41 @@ -//using Microsoft.EntityFrameworkCore; +//using DD.Persistence.Models; //using DD.Persistence.Models.Common; -//using DD.Persistence.ModelsAbstractions; -//using DD.Persistence.Database.EntityAbstractions; +//using DD.Persistence.Repositories; +//using Microsoft.EntityFrameworkCore; //namespace DD.Persistence.Repository.Repositories; -//public class TimestampedValuesCachedRepository : TimeSeriesDataRepository -// where TEntity : class, ITimestampedItem, new() -// where TDto : class, ITimestampAbstractDto, new() +//public class TimestampedValuesCachedRepository : TimestampedValuesRepository //{ -// public static TDto? FirstByDate { get; private set; } -// public static CyclicArray LastData { get; } = new CyclicArray(CacheItemsCount); +// public static TimestampedValuesDto? FirstByDate { get; private set; } +// public static CyclicArray LastData { get; } = new CyclicArray(CacheItemsCount); // private const int CacheItemsCount = 3600; -// public TimestampedValuesCachedRepository(DbContext db) : base(db) +// public TimestampedValuesCachedRepository(DbContext db, IRelatedDataRepository relatedDataRepository) : base(db, relatedDataRepository) // { -// Task.Run(async () => -// { -// var firstDateItem = await base.GetFirstAsync(CancellationToken.None); -// if (firstDateItem == null) -// { -// return; -// } +// //Task.Run(async () => +// //{ +// // var firstDateItem = await base.GetFirst(CancellationToken.None); +// // if (firstDateItem == null) +// // { +// // return; +// // } -// FirstByDate = firstDateItem; +// // FirstByDate = firstDateItem; -// var dtos = await base.GetLastAsync(CacheItemsCount, CancellationToken.None); -// dtos = dtos.OrderBy(d => d.Timestamp); -// LastData.AddRange(dtos); -// }).Wait(); +// // var dtos = await base.GetLast(CacheItemsCount, CancellationToken.None); +// // dtos = dtos.OrderBy(d => d.Timestamp); +// // LastData.AddRange(dtos); +// //}).Wait(); // } -// public override async Task> GetGtDate(DateTimeOffset dateBegin, CancellationToken token) +// public override async Task> GetGtDate(Guid discriminatorId, DateTimeOffset dateBegin, CancellationToken token) // { // if (LastData.Count == 0 || LastData[0].Timestamp > dateBegin) // { -// var dtos = await base.GetGtDate(dateBegin, token); +// var dtos = await base.GetGtDate(discriminatorId, dateBegin, token); // return dtos; // } @@ -47,9 +45,9 @@ // return items; // } -// public override async Task AddRange(IEnumerable dtos, CancellationToken token) +// public override async Task AddRange(Guid discriminatorId, IEnumerable dtos, CancellationToken token) // { -// var result = await base.AddRange(dtos, token); +// var result = await base.AddRange(discriminatorId, dtos, token); // if (result > 0) // { @@ -62,7 +60,7 @@ // return result; // } -// public override async Task GetDatesRange(CancellationToken token) +// public override async Task GetDatesRange(Guid discriminatorId, CancellationToken token) // { // if (FirstByDate == null) // return null; @@ -77,7 +75,8 @@ // }); // } -// public override async Task> GetResampledData( +// public override async Task> GetResampledData( +// Guid discriminatorId, // DateTimeOffset dateBegin, // double intervalSec = 600d, // int approxPointsCount = 1024, @@ -86,7 +85,7 @@ // var dtos = LastData.Where(i => i.Timestamp >= dateBegin); // if (LastData.Count == 0 || LastData[0].Timestamp > dateBegin) // { -// dtos = await base.GetGtDate(dateBegin, token); +// dtos = await base.GetGtDate(discriminatorId, dateBegin, token); // } // var dateEnd = dateBegin.AddSeconds(intervalSec); diff --git a/DD.Persistence/Models/TimestampedValuesDto.cs b/DD.Persistence/Models/TimestampedValuesDto.cs index d30b549..13b592e 100644 --- a/DD.Persistence/Models/TimestampedValuesDto.cs +++ b/DD.Persistence/Models/TimestampedValuesDto.cs @@ -15,5 +15,5 @@ public class TimestampedValuesDto : ITimestampAbstractDto /// /// Набор данных /// - public object[] Values { get; set; } = []; + public Dictionary Values { get; set; } = []; } diff --git a/DD.Persistence/Repositories/IRelatedDataRepository.cs b/DD.Persistence/Repositories/IRelatedDataRepository.cs index a5f5fbc..e7eae87 100644 --- a/DD.Persistence/Repositories/IRelatedDataRepository.cs +++ b/DD.Persistence/Repositories/IRelatedDataRepository.cs @@ -1,14 +1,14 @@ -using DD.Persistence.Models; - -namespace DD.Persistence.Repositories; +namespace DD.Persistence.Repositories; /// -/// Интерфейс по работе с системами - источниками данных +/// Интерфейс по работе с простой структурой данных, подразумевающей наличие связи с более сложной +/// В контексте TechMessagesRepository это системы - источники данных +/// В контексте TimestampedValuesRepository это идентификационные наборы (ключи для значений в соответствии с индексами в хранимых массивах) /// public interface IRelatedDataRepository { /// - /// Добавить систему + /// Добавить данные /// /// /// @@ -16,7 +16,7 @@ public interface IRelatedDataRepository public Task Add(TDto dataSourceSystemDto, CancellationToken token); /// - /// Получить список систем + /// Получить список данных /// /// public Task> Get(CancellationToken token); diff --git a/DD.Persistence/Repositories/ITimestampedValuesRepository.cs b/DD.Persistence/Repositories/ITimestampedValuesRepository.cs index 72cef69..abb860c 100644 --- a/DD.Persistence/Repositories/ITimestampedValuesRepository.cs +++ b/DD.Persistence/Repositories/ITimestampedValuesRepository.cs @@ -1,5 +1,4 @@ using DD.Persistence.Models; -using DD.Persistence.ModelsAbstractions; using DD.Persistence.RepositoriesAbstractions; namespace DD.Persistence.Repositories; @@ -19,23 +18,13 @@ public interface ITimestampedValuesRepository : ISyncRepository, ITimeSeriesBase Task AddRange(Guid idDiscriminator, IEnumerable dtos, CancellationToken token); /// - /// Количество записей по указанному набору в БД. Для пагинации. + /// Количество записей по указанному набору в БД. Для пагинации /// /// Дискриминатор (идентификатор) набора /// /// Task Count(Guid idDiscriminator, CancellationToken token); - /// - /// Получить последние данные - /// - /// Дискриминатор (идентификатор) набора - /// Фильтр свойств набора. Можно запросить только некоторые свойства из набора - /// - /// - /// - Task> GetLast(Guid idDiscriminator, IEnumerable? columnNames, int take, CancellationToken token); - /// /// Получение данных с фильтрацией. Значение фильтра null - отключен /// @@ -47,4 +36,22 @@ public interface ITimestampedValuesRepository : ISyncRepository, ITimeSeriesBase /// /// Task> Get(Guid idDiscriminator, DateTimeOffset? geTimestamp, IEnumerable? columnNames, int skip, int take, CancellationToken token); + + /// + /// Получение данных с начала + /// + /// Дискриминатор (идентификатор) набора + /// Количество + /// + /// + Task> GetFirst(Guid discriminatorId, int takeCount, CancellationToken token); + + /// + /// Получение данных с конца + /// + /// Дискриминатор (идентификатор) набора + /// Количество + /// + /// + Task> GetLast(Guid discriminatorId, int takeCount, CancellationToken token); }