diff --git a/AsbCloudApp/Data/DrillerDto.cs b/AsbCloudApp/Data/DrillerDto.cs
index 1a11bd2a..01a4473d 100644
--- a/AsbCloudApp/Data/DrillerDto.cs
+++ b/AsbCloudApp/Data/DrillerDto.cs
@@ -29,4 +29,9 @@ public class DrillerDto : IId
/// Отчество
///
public string? Patronymic { get; set; }
+
+ ///
+ /// Полное имя
+ ///
+ public string FullName => $"{Surname} {Name} {Patronymic}";
}
diff --git a/AsbCloudApp/Data/WellReport/DrillerReportDto.cs b/AsbCloudApp/Data/WellReport/DrillerReportDto.cs
new file mode 100644
index 00000000..fed9a917
--- /dev/null
+++ b/AsbCloudApp/Data/WellReport/DrillerReportDto.cs
@@ -0,0 +1,21 @@
+using AsbCloudApp.Data.Subsystems;
+using System.Collections.Generic;
+using System.Linq;
+
+namespace AsbCloudApp.Data.WellReport;
+
+///
+/// Показатели бурильщиков
+///
+public class DrillerReportDto
+{
+ ///
+ /// Расписание
+ ///
+ public ScheduleDto Shedule { get; set; } = null!;
+
+ ///
+ /// Наработка подсистем
+ ///
+ public IEnumerable SubsystemsStat { get; set; } = Enumerable.Empty();
+}
\ No newline at end of file
diff --git a/AsbCloudApp/Data/WellReport/DrillingBySetpointsDto.cs b/AsbCloudApp/Data/WellReport/DrillingBySetpointsDto.cs
new file mode 100644
index 00000000..a1c43036
--- /dev/null
+++ b/AsbCloudApp/Data/WellReport/DrillingBySetpointsDto.cs
@@ -0,0 +1,27 @@
+namespace AsbCloudApp.Data.WellReport;
+
+///
+/// Бурение по уставкам
+///
+public class DrillingBySetpointsDto
+{
+ ///
+ /// Метры пробуренные по уставке давления
+ ///
+ public double? MetersByPressure { get; set; }
+
+ ///
+ /// Метры пробуренные по уставке нагрузки
+ ///
+ public double? MetersByLoad { get; set; }
+
+ ///
+ /// Метры пробуренные по уставке момента
+ ///
+ public double? MetersByTorque { get; set; }
+
+ ///
+ /// Метры пробуренные по уставке скорости
+ ///
+ public double? MetersBySpeed { get; set; }
+}
\ No newline at end of file
diff --git a/AsbCloudApp/Data/WellReport/OperatingModeDto.cs b/AsbCloudApp/Data/WellReport/OperatingModeDto.cs
new file mode 100644
index 00000000..c355c623
--- /dev/null
+++ b/AsbCloudApp/Data/WellReport/OperatingModeDto.cs
@@ -0,0 +1,87 @@
+namespace AsbCloudApp.Data.WellReport;
+
+///
+/// Режим работы
+///
+public class OperatingModeDto
+{
+ ///
+ /// Интервал от
+ ///
+ public double DepthStart { get; set; }
+
+ ///
+ /// Интервал до
+ ///
+ public double DepthEnd { get; set; }
+
+ ///
+ /// Скорость проходки мин, м/ч
+ ///
+ public double? RopMin { get; set; }
+
+ ///
+ /// Скорость проходки максимум, м/ч
+ ///
+ public double? RopMax { get; set; }
+
+ ///
+ /// Скорость проходки среднее, м/ч
+ ///
+ public double? RopAvg { get; set; }
+
+ ///
+ /// Нагрузка на долото минимум, т
+ ///
+ public double? WeightOnBitMin { get; set; }
+
+ ///
+ /// Нагрузка на долото максимум, т
+ ///
+ public double? WeightOnBitMax { get; set; }
+
+ ///
+ /// Нагрузка на долото среднее, т
+ ///
+ public double? WeightOnBitAvg { get; set; }
+
+ ///
+ /// Момент минимум, кН*м
+ ///
+ public double? DriveTorqueMin { get; set; }
+
+ ///
+ /// Момент максимум, кН*м
+ ///
+ public double? DriveTorqueMax { get; set; }
+
+ ///
+ /// Момент среднее, кН*м
+ ///
+ public double? DriveTorqueAvg { get; set; }
+
+ ///
+ /// Перепад давления минимум, атм
+ ///
+ public double? DifferentialPressureMin { get; set; }
+
+ ///
+ /// Перепад давления максимум, атм
+ ///
+ public double? DifferentialPressureMax { get; set; }
+
+ ///
+ /// Перепад давления среднее, атм
+ ///
+ public double? DifferentialPressureAvg { get; set; }
+
+ ///
+ /// Q насосов минимум л/с
+ ///
+ public double? FrowRateMin { get; set; }
+
+ ///
+ /// Q насосов максимум л/с
+ ///
+ public double? FrowRateMax { get; set; }
+}
\ No newline at end of file
diff --git a/AsbCloudApp/Data/WellReport/SectionReportDto.cs b/AsbCloudApp/Data/WellReport/SectionReportDto.cs
new file mode 100644
index 00000000..0acf094d
--- /dev/null
+++ b/AsbCloudApp/Data/WellReport/SectionReportDto.cs
@@ -0,0 +1,30 @@
+using System.Collections.Generic;
+using AsbCloudApp.Data.Subsystems;
+
+namespace AsbCloudApp.Data.WellReport;
+
+///
+/// Показатели по секции
+///
+public class SectionReportDto
+{
+ ///
+ /// Идентификатор секции
+ ///
+ public int IdSection { get; set; }
+
+ ///
+ /// Наработка подсистем
+ ///
+ public IEnumerable SubsystemsStat { get; set; } = [];
+
+ ///
+ /// Режимы бурения
+ ///
+ public PlanFactDto OperatingMode { get; set; } = null!;
+
+ ///
+ /// Бурение по уставкам
+ ///
+ public DrillingBySetpointsDto? DrillingBySetpoints { get; set; }
+}
\ No newline at end of file
diff --git a/AsbCloudApp/Data/WellReport/WellReportDto.cs b/AsbCloudApp/Data/WellReport/WellReportDto.cs
new file mode 100644
index 00000000..ae8e1b9c
--- /dev/null
+++ b/AsbCloudApp/Data/WellReport/WellReportDto.cs
@@ -0,0 +1,61 @@
+using System;
+using System.Collections.Generic;
+using AsbCloudApp.Data.User;
+
+namespace AsbCloudApp.Data.WellReport;
+
+///
+/// Отчёт по скважине
+///
+public class WellReportDto
+{
+ ///
+ /// Информация о скважине
+ ///
+ public WellDto Well { get; set; }
+
+ ///
+ /// Дата начала бурения
+ ///
+ public DateTimeOffset? DateFrom { get; set; }
+
+ ///
+ /// Дата окончания бурения
+ ///
+ public DateTimeOffset? DateTo { get; set; }
+
+ ///
+ /// Дни бурения
+ ///
+ public PlanFactDto Days { get; set; } = null!;
+
+ ///
+ /// Проектная глубина
+ ///
+ public PlanFactDto WellBoreDepth { get; set; } = null!;
+
+ ///
+ /// Вертикальная глубина
+ ///
+ public PlanFactDto VerticalDepth { get; set; } = null!;
+
+ ///
+ /// Дни бурения без НПВ
+ ///
+ public double WithoutNtpDays { get; set; }
+
+ ///
+ /// Контакты
+ ///
+ public IEnumerable Contacts { get; set; } = [];
+
+ ///
+ /// Показатели по секциям
+ ///
+ public IEnumerable SectionReports { get; set; } = [];
+
+ ///
+ /// Показатели по бурильщикам
+ ///
+ public IEnumerable DrillerReports { get; set; } = [];
+}
\ No newline at end of file
diff --git a/AsbCloudApp/Repositories/IWellOperationRepository.cs b/AsbCloudApp/Repositories/IWellOperationRepository.cs
index f80ad02d..c18056e2 100644
--- a/AsbCloudApp/Repositories/IWellOperationRepository.cs
+++ b/AsbCloudApp/Repositories/IWellOperationRepository.cs
@@ -68,13 +68,13 @@ public interface IWellOperationRepository
///
(WellOperationBaseDto First, WellOperationBaseDto Last)? GetFirstAndLastFact(int idWell);
- ///
- /// Получить список операций по запросу
- ///
- ///
- ///
- ///
- Task> GetAll(WellOperationRequest request, CancellationToken token);
+ ///
+ /// Получить список операций по запросу
+ ///
+ ///
+ ///
+ ///
+ Task> GetAll(WellOperationRequest request, CancellationToken token);
///
/// Получить список операций по запросу
diff --git a/AsbCloudApp/Services/WellReport/IWellReportExportService.cs b/AsbCloudApp/Services/WellReport/IWellReportExportService.cs
new file mode 100644
index 00000000..a697e260
--- /dev/null
+++ b/AsbCloudApp/Services/WellReport/IWellReportExportService.cs
@@ -0,0 +1,19 @@
+using System.IO;
+using System.Threading.Tasks;
+using System.Threading;
+
+namespace AsbCloudApp.Services.WellReport;
+
+///
+/// Сервис экспорта отчёта
+///
+public interface IWellReportExportService
+{
+ ///
+ /// Экспортировать
+ ///
+ ///
+ ///
+ ///
+ Task<(string Name, Stream File)?> ExportAsync(int idWell, CancellationToken token);
+}
\ No newline at end of file
diff --git a/AsbCloudApp/Services/WellReport/IWellReportService.cs b/AsbCloudApp/Services/WellReport/IWellReportService.cs
new file mode 100644
index 00000000..9bf61049
--- /dev/null
+++ b/AsbCloudApp/Services/WellReport/IWellReportService.cs
@@ -0,0 +1,19 @@
+using System.Threading;
+using System.Threading.Tasks;
+using AsbCloudApp.Data.WellReport;
+
+namespace AsbCloudApp.Services.WellReport;
+
+///
+/// Сервис формирования отчёта
+///
+public interface IWellReportService
+{
+ ///
+ /// Сформировать
+ ///
+ ///
+ ///
+ ///
+ Task GetAsync(int idWell, CancellationToken token);
+}
\ No newline at end of file
diff --git a/AsbCloudInfrastructure.Tests/Services/WellReportServiceTest.cs b/AsbCloudInfrastructure.Tests/Services/WellReportServiceTest.cs
new file mode 100644
index 00000000..2c3fe4c2
--- /dev/null
+++ b/AsbCloudInfrastructure.Tests/Services/WellReportServiceTest.cs
@@ -0,0 +1,335 @@
+using AsbCloudApp.Data;
+using AsbCloudApp.Data.ProcessMaps.Operations;
+using AsbCloudApp.Data.Trajectory;
+using AsbCloudApp.Data.User;
+using AsbCloudApp.Data.WellOperation;
+using AsbCloudApp.Repositories;
+using AsbCloudApp.Requests;
+using AsbCloudApp.Services;
+using AsbCloudApp.Services.ProcessMaps.WellDrilling;
+using AsbCloudInfrastructure.Services.WellReport;
+using NSubstitute;
+using System.Collections.Generic;
+using System;
+using System.Threading;
+using Xunit;
+using AsbCloudDb.Model;
+using System.Linq;
+using System.Threading.Tasks;
+
+namespace AsbCloudInfrastructure.Tests.Services;
+
+public class WellReportServiceTest
+{
+ private static readonly WellDto Well = new()
+ {
+ Caption = "Скважина №1",
+ Cluster = "Кластер A",
+ Deposit = "Месторождение Б",
+ Latitude = 55.7558,
+ Longitude = 37.6176,
+ Timezone = new SimpleTimezoneDto { Hours = 3 },
+ WellType = "Разведочная",
+ IdWellType = 1,
+ IdCluster = 1001,
+ IdState = 1,
+ StartDate = DateTimeOffset.Now.AddMonths(-2),
+ LastTelemetryDate = DateTimeOffset.Now,
+ IdTelemetry = 12345,
+ };
+
+ private static readonly IEnumerable WellOperations = new[]
+ {
+ new WellOperationDto
+ {
+ Id = 1,
+ IdWell = 101,
+ IdWellSectionType = 1001,
+ IdType = 1,
+ IdCategory = 2001,
+ DepthStart = 1500,
+ DepthEnd = 1550,
+ DateStart = new DateTimeOffset(new DateTime(2024, 1, 13, 2, 0, 0)),
+ DurationHours = 48,
+ IdPlan = null,
+ IdParentCategory = 2001,
+ Day = 5
+ },
+ new WellOperationDto
+ {
+ Id = 4,
+ IdWell = 101,
+ IdWellSectionType = 1002,
+ IdType = 1,
+ IdCategory = 2001,
+ DepthStart = 1500,
+ DepthEnd = 1550,
+ DateStart = new DateTimeOffset(new DateTime(2024, 1, 10, 0, 0, 0)),
+ DurationHours = 48,
+ IdPlan = null,
+ IdParentCategory = 2001,
+ Day = 3
+ },
+ new WellOperationDto
+ {
+ Id = 2,
+ IdWell = 102,
+ IdWellSectionType = 1002,
+ IdType = 0,
+ IdCategory = 2002,
+ DepthStart = 2500,
+ DepthEnd = 2600,
+ DateStart = new DateTimeOffset(new DateTime(2024, 1, 10, 0, 0, 0)),
+ DurationHours = 72,
+ IdPlan = 1,
+ IdParentCategory = 3002,
+ Day = 3
+ },
+ new WellOperationDto
+ {
+ Id = 3,
+ IdWell = 103,
+ IdWellSectionType = 1003,
+ IdType = 0,
+ IdCategory = 2003,
+ DepthStart = 3500,
+ DepthEnd = 3600,
+ DateStart = new DateTimeOffset(new DateTime(2024, 1, 10, 1, 0, 0)),
+ DurationHours = 24,
+ IdPlan = 2,
+ IdParentCategory = 3003,
+ Day = 4
+ }
+ };
+
+ private readonly IWellService wellService;
+ private readonly IWellOperationService wellOperationService;
+ private readonly IWellContactService wellContactService;
+ private readonly IProcessMapReportDrillingService processMapReportDrillingService;
+ private readonly ISubsystemService subsystemService;
+
+ private readonly ITrajectoryRepository trajectoryPlanRepository;
+ private readonly ITrajectoryRepository trajectoryFactRepository;
+
+ private readonly IChangeLogRepository
+ processMapPlanRotorRepository;
+
+ private readonly IScheduleRepository scheduleRepository;
+
+ private readonly WellReportService wellReportService;
+
+ public WellReportServiceTest()
+ {
+ wellService = Substitute.For();
+ wellOperationService = Substitute.For();
+ wellContactService = Substitute.For();
+ processMapReportDrillingService = Substitute.For();
+ subsystemService = Substitute.For();
+ trajectoryPlanRepository = Substitute.For>();
+ trajectoryFactRepository = Substitute.For>();
+ processMapPlanRotorRepository =
+ Substitute.For>();
+
+ scheduleRepository = Substitute.For();
+
+ wellService.GetOrDefaultAsync(Arg.Any(), Arg.Any())
+ .ReturnsForAnyArgs(Well);
+
+ wellOperationService
+ .GetAsync(Arg.Is(x => x.OperationType == WellOperation.IdOperationTypeFact),
+ Arg.Any())
+ .Returns(WellOperations.Where(x => x.IdType == WellOperation.IdOperationTypeFact));
+
+ wellOperationService
+ .GetAsync(Arg.Is(x => x.OperationType == WellOperation.IdOperationTypePlan),
+ Arg.Any())
+ .Returns(WellOperations.Where(x => x.IdType == WellOperation.IdOperationTypePlan));
+
+ wellReportService = new WellReportService(wellService,
+ wellOperationService,
+ wellContactService,
+ processMapReportDrillingService,
+ subsystemService,
+ trajectoryPlanRepository,
+ trajectoryFactRepository,
+ processMapPlanRotorRepository,
+ scheduleRepository);
+ }
+
+ [Fact]
+ public async Task Returns_well_info_not_null()
+ {
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+ Assert.NotNull(result.Well);
+ }
+
+ [Fact]
+ public async Task Returns_contacts_not_empty()
+ {
+ //arrange
+ var contacts = new[]
+ {
+ new ContactDto()
+ {
+ Id = 1,
+ IdCompanyType = 2,
+ IdWell = 101,
+ FullName = "Ivan Petrov",
+ Email = "ivan.petrov@example.com",
+ Phone = "+7 (123) 456-78-90",
+ Position = "Chief Engineer",
+ Company = "test"
+ }
+ };
+
+ wellContactService.GetAllAsync(Arg.Any(), Arg.Any())
+ .ReturnsForAnyArgs(contacts);
+
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+ Assert.Single(result.Contacts);
+ }
+
+ [Fact]
+ public async Task Returns_valid_from_and_to_dates()
+ {
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+
+ var expectedDateFrom = new DateTimeOffset(new DateTime(2024, 1, 10, 0, 0, 0));
+ var expectedDateTo = new DateTimeOffset(new DateTime(2024, 1, 11, 1, 0, 0));
+
+ Assert.Equal(expectedDateFrom, result.DateFrom);
+ Assert.Equal(expectedDateTo, result.DateTo);
+ }
+
+ [Fact]
+ public async Task Returns_valid_days()
+ {
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+
+ Assert.Equal(4, result.Days.Plan);
+ Assert.Equal(5, result.Days.Fact);
+ }
+
+ [Fact]
+ public async Task Returns_valid_vertical_depth()
+ {
+ //arrange
+ var planTrajectory = new TrajectoryGeoPlanDto
+ {
+ Id = 1,
+ IdWell = 123,
+ WellboreDepth = 1500.75,
+ ZenithAngle = 45.5,
+ AzimuthGeo = 120.0,
+ AzimuthMagnetic = 115.5,
+ VerticalDepth = 1480.3,
+ UpdateDate = DateTimeOffset.UtcNow,
+ IdUser = 42
+ };
+
+ var factTrajectory = new TrajectoryGeoFactDto
+ {
+ Id = 1,
+ IdWell = 123,
+ WellboreDepth = 1500.75,
+ ZenithAngle = 45.5,
+ AzimuthGeo = 120.0,
+ AzimuthMagnetic = 115.5,
+ VerticalDepth = 1600,
+ UpdateDate = DateTimeOffset.UtcNow,
+ IdUser = 42
+ };
+
+ trajectoryPlanRepository.GetAsync(Arg.Any(), Arg.Any())
+ .ReturnsForAnyArgs(new[] { planTrajectory });
+
+ trajectoryFactRepository.GetAsync(Arg.Any(), Arg.Any())
+ .ReturnsForAnyArgs(new[] { factTrajectory });
+
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+
+ Assert.Equal(result.VerticalDepth.Plan, 1480.3);
+ Assert.Equal(result.VerticalDepth.Fact, 1600);
+ }
+
+ [Fact]
+ public async Task Returns_valid_without_ntp_days()
+ {
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+
+ Assert.Equal(4, result.WithoutNtpDays);
+ }
+
+
+ [Fact]
+ public async Task Returns_section_reports_not_empty()
+ {
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+ Assert.Equal(2, result.SectionReports.Count());
+ }
+
+ [Fact]
+ public async Task Returns_driller_reports_not_empty()
+ {
+ //arrange
+ var schedules = new[]
+ {
+ new ScheduleDto
+ {
+ Id = 1,
+ IdDriller = 2001,
+ ShiftStart = new TimeDto(7),
+ ShiftEnd = new TimeDto(20),
+ DrillStart = new DateTimeOffset(2024, 9, 1, 7, 0, 0, TimeSpan.Zero),
+ DrillEnd = new DateTimeOffset(2024, 9, 1, 19, 0, 0, TimeSpan.Zero),
+ },
+ new ScheduleDto
+ {
+ Id = 2,
+ IdDriller = 2002,
+ ShiftStart = new TimeDto(20),
+ ShiftEnd = new TimeDto(16),
+ DrillStart = new DateTimeOffset(2024, 9, 1, 7, 0, 0, TimeSpan.Zero),
+ DrillEnd = new DateTimeOffset(2024, 9, 1, 19, 0, 0, TimeSpan.Zero),
+ }
+ };
+
+ scheduleRepository.GetByIdWellAsync(Arg.Any(), Arg.Any())
+ .ReturnsForAnyArgs(schedules);
+
+ //act
+ var result = await wellReportService.GetAsync(1, CancellationToken.None);
+
+ //assert
+ Assert.NotNull(result);
+ Assert.Equal(2, result.DrillerReports.Count());
+ }
+}
\ No newline at end of file
diff --git a/AsbCloudInfrastructure/AsbCloudInfrastructure.csproj b/AsbCloudInfrastructure/AsbCloudInfrastructure.csproj
index 2942a43a..327b70b7 100644
--- a/AsbCloudInfrastructure/AsbCloudInfrastructure.csproj
+++ b/AsbCloudInfrastructure/AsbCloudInfrastructure.csproj
@@ -53,6 +53,7 @@
+
diff --git a/AsbCloudInfrastructure/DependencyInjection.cs b/AsbCloudInfrastructure/DependencyInjection.cs
index b3d06d6d..37368596 100644
--- a/AsbCloudInfrastructure/DependencyInjection.cs
+++ b/AsbCloudInfrastructure/DependencyInjection.cs
@@ -47,6 +47,8 @@ using Microsoft.Extensions.Caching.Memory;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using System;
+using AsbCloudApp.Services.WellReport;
+using AsbCloudInfrastructure.Services.WellReport;
namespace AsbCloudInfrastructure;
@@ -470,7 +472,9 @@ public static class DependencyInjection
services.AddTransient, TrajectoryEditableRepository>();
services.AddTransient, TrajectoryEditableRepository>();
services.AddTransient();
- services.AddTransient();
+ services.AddTransient, TrajectoryEditableRepository>();
+ services.AddTransient, TrajectoryEditableRepository>();
+ services.AddTransient();
services.AddTransient();
services.AddTransient();
services.AddTransient, CrudCacheRepositoryBase();
services.AddTransient();
+ services.AddTransient();
+ services.AddTransient();
+
return services;
}
}
diff --git a/AsbCloudInfrastructure/Repository/WellOperationRepository.cs b/AsbCloudInfrastructure/Repository/WellOperationRepository.cs
index f557d106..92f5844c 100644
--- a/AsbCloudInfrastructure/Repository/WellOperationRepository.cs
+++ b/AsbCloudInfrastructure/Repository/WellOperationRepository.cs
@@ -20,6 +20,7 @@ namespace AsbCloudInfrastructure.Repository;
public class WellOperationRepository : CrudRepositoryBase,
IWellOperationRepository
{
+ private const string keyCacheTemplate = "OperationsBySectionSummaries_{0}";
private const string cacheKeyWellOperations = "FirstAndLastFactWellsOperations";
private readonly IMemoryCache memoryCache;
private readonly IWellOperationCategoryRepository wellOperationCategoryRepository;
@@ -109,13 +110,26 @@ public class WellOperationRepository : CrudRepositoryBase> GetSectionsAsync(IEnumerable idsWells, CancellationToken token)
{
- const string keyCacheSections = "OperationsBySectionSummarties";
+ var result = new List();
+ var notFoundIds = new List();
- var cache = await memoryCache.GetOrCreateAsync(keyCacheSections, async (entry) =>
+ foreach (var idWell in idsWells)
{
- entry.AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(30);
+ var cacheKey = string.Format(keyCacheTemplate, idWell);
+ if (memoryCache.TryGetValue>(cacheKey, out var section))
+ {
+ result.AddRange(section!);
+ }
+ else
+ {
+ notFoundIds.Add(idWell);
+ }
+ }
+ if (notFoundIds.Count != 0)
+ {
var query = dbContext.Set()
+ .Where(operation => notFoundIds.Contains( operation.IdWell))
.GroupBy(operation => new
{
operation.IdWell,
@@ -129,51 +143,49 @@ public class WellOperationRepository : CrudRepositoryBase operation.DateStart)
- .Select(operation => new
- {
- operation.DateStart,
- operation.DepthStart,
- })
- .First(),
-
+ .OrderBy(operation => operation.DateStart)
+ .Select(operation => new { operation.DateStart, operation.DepthStart, })
+ .First(),
Last = group
- .OrderByDescending(operation => operation.DateStart)
- .Select(operation => new
- {
- operation.DateStart,
- operation.DurationHours,
- operation.DepthEnd,
- })
- .First(),
- })
- .Where(s => idsWells.Contains(s.IdWell));
- var dbData = await query.ToArrayAsync(token);
- var sections = dbData.Select(
- item => new SectionByOperationsDto
+ .OrderByDescending(operation => operation.DateStart)
+ .Select(operation => new
+ {
+ operation.DateStart,
+ operation.DurationHours,
+ operation.DepthEnd,
+ })
+ .First(),
+ });
+
+ var entities = await query.ToArrayAsync(token);
+ var dtos = entities.Select(
+ entity => new SectionByOperationsDto
{
- IdWell = item.IdWell,
- IdType = item.IdType,
- IdWellSectionType = item.IdWellSectionType,
-
- Caption = item.Caption,
-
- DateStart = item.First.DateStart,
- DepthStart = item.First.DepthStart,
-
- DateEnd = item.Last.DateStart.AddHours(item.Last.DurationHours),
- DepthEnd = item.Last.DepthEnd,
+ IdWell = entity.IdWell,
+ IdType = entity.IdType,
+ IdWellSectionType = entity.IdWellSectionType,
+ Caption = entity.Caption,
+ DateStart = entity.First.DateStart,
+ DepthStart = entity.First.DepthStart,
+ DateEnd = entity.Last.DateStart.AddHours(entity.Last.DurationHours),
+ DepthEnd = entity.Last.DepthEnd,
})
- .ToArray()
- .AsEnumerable();
+ .ToList();
- entry.Value = sections;
- return sections;
- });
+ result.AddRange(dtos);
- return cache!;
+ var groupedByWellDtos = dtos
+ .GroupBy(dto => dto.IdWell);
+
+ foreach (var group in groupedByWellDtos)
+ {
+ var cacheKey = string.Format(keyCacheTemplate, group.Key);
+ memoryCache.Set(cacheKey, group.AsEnumerable(), TimeSpan.FromMinutes(30));
+ }
+ }
+
+ return result;
}
public async Task GetDatesRangeAsync(int idWell, int idType, CancellationToken cancellationToken)
@@ -197,34 +209,34 @@ public class WellOperationRepository : CrudRepositoryBase
- {
- entry.AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(5);
- var query = dbContext.Set()
- .Where(o => o.IdType == WellOperation.IdOperationTypeFact)
- .GroupBy(o => o.IdWell)
- .Select(group => new
- {
- IdWell = group.Key,
- FirstFact = group.OrderBy(o => o.DateStart).First(),
- LastFact = group.OrderBy(o => o.DateStart).Last(),
- });
+ var cachedDictionary = memoryCache.GetOrCreate(cacheKeyWellOperations, (entry) =>
+ {
+ entry.AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(5);
+ var query = dbContext.Set()
+ .Where(o => o.IdType == WellOperation.IdOperationTypeFact)
+ .GroupBy(o => o.IdWell)
+ .Select(group => new
+ {
+ IdWell = group.Key,
+ FirstFact = group.OrderBy(o => o.DateStart).First(),
+ LastFact = group.OrderBy(o => o.DateStart).Last(),
+ });
- var entities = query.ToArray();
+ var entities = query.ToArray();
- var dictionary = entities.ToDictionary(s => s.IdWell, s => (Convert(s.FirstFact), Convert(s.LastFact)));
- entry.Value = dictionary;
+ var dictionary = entities.ToDictionary(s => s.IdWell, s => (Convert(s.FirstFact), Convert(s.LastFact)));
+ entry.Value = dictionary;
- return dictionary;
+ return dictionary;
- })!;
+ })!;
- var firstAndLast = cachedDictionary.GetValueOrDefault(idWell);
- return firstAndLast;
+ var firstAndLast = cachedDictionary.GetValueOrDefault(idWell);
+ return firstAndLast;
}
- public override async Task DeleteAsync(int id, CancellationToken token)
+ public override async Task DeleteAsync(int id, CancellationToken token)
{
var result = await base.DeleteAsync(id, token);
if (result > 0)
diff --git a/AsbCloudInfrastructure/Services/WellReport/WellReport.xlsx b/AsbCloudInfrastructure/Services/WellReport/WellReport.xlsx
new file mode 100644
index 00000000..c8283d28
Binary files /dev/null and b/AsbCloudInfrastructure/Services/WellReport/WellReport.xlsx differ
diff --git a/AsbCloudInfrastructure/Services/WellReport/WellReportExportService.cs b/AsbCloudInfrastructure/Services/WellReport/WellReportExportService.cs
new file mode 100644
index 00000000..bfb4d3fb
--- /dev/null
+++ b/AsbCloudInfrastructure/Services/WellReport/WellReportExportService.cs
@@ -0,0 +1,386 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using System.Reflection;
+using System.Threading;
+using System.Threading.Tasks;
+using AsbCloudApp.Data;
+using AsbCloudApp.Data.Subsystems;
+using AsbCloudApp.Data.User;
+using AsbCloudApp.Data.WellReport;
+using AsbCloudApp.Services.WellReport;
+using ClosedXML.Excel;
+
+namespace AsbCloudInfrastructure.Services.WellReport;
+
+public class WellReportExportService : IWellReportExportService
+{
+ private static readonly IDictionary PlanOperatingModeRows = new Dictionary()
+ {
+ { 2, 79 }, // Направление план
+ { 3, 81 }, // Кондуктор план
+ { 9, 83 }, // Кондуктор 2 план
+ { 1, 85 }, // Пилотный ствол план
+ { 7, 87 }, // Пилотный ствол 2 план
+ { 13, 89 }, // Пилотный ствол 3 план
+ { 4, 91 }, // Эксплуатационная колонна план
+ { 10, 93 }, // Эксплуатационная колонна 2 план
+ { 6, 95 }, // Хвостовик план
+ { 12, 97 }, // Хвостовик 2 план
+ { 18, 99 }, // Хвостовик 3 план
+ { 24, 101 }, // Хвостовик 4 план
+ { 30, 103 }, // Хвостовик 5 план
+ { 34, 105 }, // Хвостовик 6 план
+ { 35, 107 }, // Хвостовик 7 план
+ { 36, 109 }, // Хвостовик 8 план
+ { 37, 111 }, // Хвостовик 9 план
+ { 38, 113 } // Хвостовик 10 план
+ };
+
+ private static readonly IDictionary FactOperatingModeRows = new Dictionary
+ {
+ { 2, 80 }, // Направление факт
+ { 3, 82 }, // Кондуктор факт
+ { 9, 84 }, // Кондуктор 2 факт
+ { 1, 86 }, // Пилотный ствол факт
+ { 7, 88 }, // Пилотный ствол 2 факт
+ { 13, 90 }, // Пилотный ствол 3 факт
+ { 4, 92 }, // Эксплуатационная колонна факт
+ { 10, 94 }, // Эксплуатационная колонна 2 факт
+ { 6, 96 }, // Хвостовик факт
+ { 12, 98 }, // Хвостовик 2 факт
+ { 18, 100 }, // Хвостовик 3 факт
+ { 24, 102 }, // Хвостовик 4 факт
+ { 30, 104 }, // Хвостовик 5 факт
+ { 34, 106 }, // Хвостовик 6 факт
+ { 35, 108 }, // Хвостовик 7 факт
+ { 36, 110 }, // Хвостовик 8 факт
+ { 37, 112 }, // Хвостовик 9 факт
+ { 38, 114 } // Хвостовик 10 факт
+ };
+
+ private static readonly IDictionary SubsystemRows = new Dictionary()
+ {
+ { 2, 140 }, // Направление
+ { 3, 141 }, // Кондуктор
+ { 9, 142 }, // Кондуктор 2
+ { 1, 143 }, // Пилотный ствол
+ { 7, 144 }, // Пилотный ствол 2
+ { 13, 145 }, // Пилотный ствол 3
+ { 4, 146 }, // Эксплуатационная колонна
+ { 10, 147 }, // Эксплуатационная колонна 2
+ { 6, 148 }, // Хвостовик
+ { 12, 149 }, // Хвостовик 2
+ { 18, 150 }, // Хвостовик 3
+ { 24, 151 }, // Хвостовик 4
+ { 30, 152 }, // Хвостовик 5
+ { 34, 153 }, // Хвостовик 6
+ { 35, 154 }, // Хвостовик 7
+ { 36, 155 }, // Хвостовик 8
+ { 37, 156 }, // Хвостовик 9
+ { 38, 157 } // Хвостовик 10
+ };
+
+ private static readonly IDictionary SetpointsRows = new Dictionary()
+ {
+ { 2, 161 }, // Направление
+ { 3, 162 }, // Кондуктор
+ { 9, 163 }, // Кондуктор 2
+ { 1, 164 }, // Пилотный ствол
+ { 7, 165 }, // Пилотный ствол 2
+ { 13, 166 }, // Пилотный ствол 3
+ { 4, 167 }, // Эксплуатационная колонна
+ { 10, 168 }, // Эксплуатационная колонна 2
+ { 6, 169 }, // Хвостовик
+ { 12, 170 }, // Хвостовик 2
+ { 18, 171 }, // Хвостовик 3
+ { 24, 172 }, // Хвостовик 4
+ { 30, 173 }, // Хвостовик 5
+ { 34, 174 }, // Хвостовик 6
+ { 35, 175 }, // Хвостовик 7
+ { 36, 176 }, // Хвостовик 8
+ { 37, 177 }, // Хвостовик 9
+ { 38, 178 } // Хвостовик 10
+ };
+
+ private const string TemplateName = "WellReport.xlsx";
+ private const string SheetName = "Отчёт";
+
+ private readonly IWellReportService wellReportService;
+
+ private const string DateFromCell = "D5";
+ private const string DateToCell = "E5";
+ private const string DaysPlanCell = "D6";
+ private const string DaysFactCell = "E6";
+ private const string WithoutNtpDaysCell = "D8";
+ private const string WellBoreDepthPlanCell = "D12";
+ private const string VerticalDepthPlanCell = "E12";
+ private const string WellBoreDepthFactCell = "D13";
+ private const string VerticalDepthFactCell = "E13";
+ private const string WellCell = "I5";
+ private const string ClusterCell = "I6";
+ private const string DepositCell = "I7";
+ private const string CustomerCell = "N5";
+
+ public WellReportExportService(IWellReportService wellReportService)
+ {
+ this.wellReportService = wellReportService;
+ }
+
+ public async Task<(string Name, Stream File)?> ExportAsync(int idWell, CancellationToken token)
+ {
+ var report = await wellReportService.GetAsync(idWell, token);
+
+ if (report == null)
+ return null;
+
+ var stream = Assembly.GetExecutingAssembly().GetTemplateCopyStream(TemplateName);
+ using var workbook = new XLWorkbook(stream);
+
+ var sheet = workbook.GetWorksheet(SheetName);
+
+ FillSheet(sheet, report);
+
+ MemoryStream memoryStream = new();
+ workbook.SaveAs(memoryStream, new SaveOptions { });
+ memoryStream.Seek(0, SeekOrigin.Begin);
+
+ var name = $"Отчёт по скважине {report.Well.Caption} куст {report.Well.Cluster}.xlsx";
+
+ return (name, memoryStream);
+ }
+
+ private static void FillSheet(IXLWorksheet sheet, WellReportDto report)
+ {
+ sheet.Cell(DateFromCell).SetCellValue(report.DateFrom);
+ sheet.Cell(DateToCell).SetCellValue(report.DateTo);
+ sheet.Cell(DaysPlanCell).SetCellValue(report.Days.Plan);
+ sheet.Cell(DaysFactCell).SetCellValue(report.Days.Fact);
+ sheet.Cell(WithoutNtpDaysCell).SetCellValue(report.WithoutNtpDays);
+ sheet.Cell(WellBoreDepthPlanCell).SetCellValue(report.WellBoreDepth.Plan);
+ sheet.Cell(WellBoreDepthFactCell).SetCellValue(report.WellBoreDepth.Fact);
+ sheet.Cell(VerticalDepthPlanCell).SetCellValue(report.VerticalDepth.Plan);
+ sheet.Cell(VerticalDepthFactCell).SetCellValue(report.VerticalDepth.Fact);
+ sheet.Cell(WellCell).SetCellValue(report.Well.Caption);
+ sheet.Cell(ClusterCell).SetCellValue(report.Well.Cluster);
+ sheet.Cell(DepositCell).SetCellValue(report.Well.Deposit);
+
+ var customer = report.Well.Companies.FirstOrDefault(x => x.IdCompanyType == 1);
+ sheet.Cell(CustomerCell).SetCellValue(customer?.Caption);
+
+ FillContacts(sheet, report.Contacts);
+ FillSectionReports(sheet, report.SectionReports);
+ FillDrillerReports(sheet, report.DrillerReports);
+ }
+
+ private static void FillContacts(IXLWorksheet sheet, IEnumerable contacts)
+ {
+ var positionsByCompanyType = new Dictionary()
+ {
+ { 7, "Супервайзер" },
+ { 2, "Мастер" },
+ { 3, "Инженер по автоматизации" },
+ { 5, "Инженер по р-рам " },
+ { 6, "Инженер ННБ" },
+ { 14, "Инженер по долотам" },
+ { 4, "Инженер ГТИ" },
+ { 9, "Инженер по цементированию" }
+ };
+
+ const int positionColumn = 11;
+ const int fullNameColumn = 14;
+ const int companyColumn = 16;
+ const int phoneColumn = 18;
+
+ contacts = contacts.OrderByDescending(x => x.Id)
+ .GroupBy(x => x.IdCompanyType)
+ .Select(x => x.First());
+
+ var row = 6;
+
+ foreach (var contact in contacts)
+ {
+ if (!positionsByCompanyType.TryGetValue(contact.IdCompanyType, out var position))
+ continue;
+
+ sheet.Cell(row, positionColumn).SetCellValue(position);
+ sheet.Cell(row, fullNameColumn).SetCellValue(contact.FullName);
+ sheet.Cell(row, companyColumn).SetCellValue(contact.Company);
+ sheet.Cell(row, phoneColumn).SetCellValue(contact.Phone);
+
+ row++;
+ }
+ }
+
+ private static void FillDrillerReports(IXLWorksheet sheet, IEnumerable drillerReports)
+ {
+ drillerReports = drillerReports.OrderBy(x => x.Shedule.DrillStart);
+
+ const int IdSubsystemAPDRotor = 11;
+ const int IdSubsystemAPDSlide = 12;
+ const int IdSubsystemOscillation = 65536;
+
+ const int fullNameColumn = 1;
+ const int drillStartColumn = 5;
+ const int drillEndColumn = 6;
+ const int shiftStart = 7;
+ const int shiftEnd = 8;
+ const int kUsageApdRotorColumn = 9;
+ const int kUsageApdSlideColumn = 10;
+ const int kUsageOscillationColumn = 11;
+
+ var row = 182;
+
+ foreach (var drillingReport in drillerReports)
+ {
+ sheet.Cell(row, fullNameColumn).SetCellValue(drillingReport.Shedule.Driller?.FullName);
+ sheet.Cell(row, drillStartColumn).SetCellValue(drillingReport.Shedule.DrillStart);
+ sheet.Cell(row, drillEndColumn).SetCellValue(drillingReport.Shedule.DrillEnd);
+ sheet.Cell(row, shiftStart).SetCellValue(drillingReport.Shedule.ShiftStart.ToString());
+ sheet.Cell(row, shiftEnd).SetCellValue(drillingReport.Shedule.ShiftEnd.ToString());
+
+ foreach (var subsystemStat in drillingReport.SubsystemsStat)
+ {
+ switch (subsystemStat.IdSubsystem)
+ {
+ case IdSubsystemAPDRotor:
+ sheet.Cell(row, kUsageApdRotorColumn).SetCellValue(subsystemStat.KUsage);
+ break;
+ case IdSubsystemAPDSlide:
+ sheet.Cell(row, kUsageApdSlideColumn).SetCellValue(subsystemStat.KUsage);
+ break;
+ case IdSubsystemOscillation:
+ sheet.Cell(row, kUsageOscillationColumn).SetCellValue(subsystemStat.KUsage);
+ break;
+ }
+ }
+
+ row++;
+ }
+ }
+
+ private static void FillSectionReports(IXLWorksheet sheet, IEnumerable sectionReports)
+ {
+ foreach (var sectionReport in sectionReports)
+ {
+ FillOperatingMode(sheet, sectionReport.IdSection, sectionReport.OperatingMode);
+
+ var drillingBySetpoints = sectionReport.DrillingBySetpoints;
+
+ if (drillingBySetpoints != null)
+ FillDrillingBySetpoints(sheet, sectionReport.IdSection, drillingBySetpoints);
+
+ FillSubsystemsStat(sheet, sectionReport.IdSection, sectionReport.SubsystemsStat);
+ }
+ }
+
+ private static void FillDrillingBySetpoints(IXLWorksheet sheet, int idSection,
+ DrillingBySetpointsDto drillingBySetpoints)
+ {
+ const int pressureColumn = 8;
+ const int axialLoadColumn = 9;
+ const int topDriveTorqueColumn = 10;
+ const int speedLimitColumn = 11;
+
+ if (!SetpointsRows.TryGetValue(idSection, out var row))
+ return;
+
+ sheet.Cell(row, pressureColumn).SetCellValue(drillingBySetpoints.MetersByPressure);
+ sheet.Cell(row, axialLoadColumn).SetCellValue(drillingBySetpoints.MetersByLoad);
+ sheet.Cell(row, topDriveTorqueColumn).SetCellValue(drillingBySetpoints.MetersByTorque);
+ sheet.Cell(row, speedLimitColumn).SetCellValue(drillingBySetpoints.MetersBySpeed);
+ }
+
+ private static void FillSubsystemsStat(IXLWorksheet sheet, int idSection,
+ IEnumerable subsystemsStat)
+ {
+ const int idSubsystemAPDRotor = 11;
+ const int idSubsystemAPDSlide = 12;
+ const int idSubsystemOscillation = 65536;
+
+ const int kUsageApdRotorColumn = 3;
+ const int kUsageApdSlideColumn = 4;
+ const int kUsageOscillationColumn = 5;
+
+ const int sumDepthIntervalApdRotorColumn = 14;
+ const int sumDepthIntervalApdSlideColumn = 15;
+ const int sumDepthIntervalApdOscillation = 17;
+
+ if (!SubsystemRows.TryGetValue(idSection, out var row))
+ return;
+
+ foreach (var subsystemStat in subsystemsStat)
+ {
+ switch (subsystemStat.IdSubsystem)
+ {
+ case idSubsystemAPDRotor:
+ sheet.Cell(row, kUsageApdRotorColumn).SetCellValue(subsystemStat.KUsage);
+ sheet.Cell(row, sumDepthIntervalApdRotorColumn).SetCellValue(subsystemStat.SumDepthInterval);
+ break;
+ case idSubsystemAPDSlide:
+ sheet.Cell(row, kUsageApdSlideColumn).SetCellValue(subsystemStat.KUsage);
+ sheet.Cell(row, sumDepthIntervalApdSlideColumn).SetCellValue(subsystemStat.SumDepthInterval);
+ break;
+ case idSubsystemOscillation:
+ sheet.Cell(row, kUsageOscillationColumn).SetCellValue(subsystemStat.KUsage);
+ sheet.Cell(row, sumDepthIntervalApdOscillation).SetCellValue(subsystemStat.SumDepthInterval);
+ break;
+ }
+ }
+ }
+
+ private static void FillOperatingMode(IXLWorksheet sheet, int idSection,
+ PlanFactDto operatingMode)
+ {
+ const int depthStartColumn = 3;
+ const int depthEndColumn = 4;
+ const int ropMinColumn = 6;
+ const int ropMaxColumn = 7;
+ const int ropAvgColumn = 8;
+ const int weightOnBitMinColumn = 9;
+ const int weightOnBitMaxColumn = 10;
+ const int weightOnBitAvgColumn = 11;
+ const int driveTorqueMinColumn = 12;
+ const int driveTorqueMaxColumn = 13;
+ const int driveTorqueAvgColumn = 14;
+ const int differentialPressureMinColumn = 15;
+ const int differentialPressureMaxColumn = 16;
+ const int differentialPressureAvgColumn = 17;
+ const int frowRateMinColumn = 18;
+ const int frowRateMaxColumn = 19;
+
+ if (PlanOperatingModeRows.TryGetValue(idSection, out var planRow))
+ {
+ sheet.Cell(planRow, depthStartColumn).SetCellValue(operatingMode.Plan?.DepthStart);
+ sheet.Cell(planRow, depthEndColumn).SetCellValue(operatingMode.Plan?.DepthEnd);
+
+ sheet.Cell(planRow, ropMinColumn).SetCellValue(operatingMode.Plan?.RopMin);
+ sheet.Cell(planRow, ropMaxColumn).SetCellValue(operatingMode.Plan?.RopMax);
+ sheet.Cell(planRow, ropAvgColumn).SetCellValue(operatingMode.Plan?.RopAvg);
+
+ sheet.Cell(planRow, weightOnBitMinColumn).SetCellValue(operatingMode.Plan?.WeightOnBitMin);
+ sheet.Cell(planRow, weightOnBitMaxColumn).SetCellValue(operatingMode.Plan?.WeightOnBitMax);
+ sheet.Cell(planRow, weightOnBitAvgColumn).SetCellValue(operatingMode.Plan?.WeightOnBitAvg);
+
+ sheet.Cell(planRow, driveTorqueMinColumn).SetCellValue(operatingMode.Plan?.DriveTorqueMin);
+ sheet.Cell(planRow, driveTorqueMaxColumn).SetCellValue(operatingMode.Plan?.DriveTorqueMax);
+ sheet.Cell(planRow, driveTorqueAvgColumn).SetCellValue(operatingMode.Plan?.DriveTorqueAvg);
+
+ sheet.Cell(planRow, differentialPressureMinColumn)
+ .SetCellValue(operatingMode.Plan?.DifferentialPressureMin);
+ sheet.Cell(planRow, differentialPressureMaxColumn)
+ .SetCellValue(operatingMode.Plan?.DifferentialPressureMax);
+ sheet.Cell(planRow, differentialPressureAvgColumn)
+ .SetCellValue(operatingMode.Plan?.DifferentialPressureAvg);
+
+ sheet.Cell(planRow, frowRateMinColumn).SetCellValue(operatingMode.Plan?.FrowRateMin);
+ sheet.Cell(planRow, frowRateMaxColumn).SetCellValue(operatingMode.Plan?.FrowRateMax);
+ }
+
+ if (FactOperatingModeRows.TryGetValue(idSection, out var factRow))
+ {
+ sheet.Cell(factRow, depthStartColumn).SetCellValue(operatingMode.Fact?.DepthStart);
+ sheet.Cell(factRow, depthEndColumn).SetCellValue(operatingMode.Fact?.DepthEnd);
+ }
+ }
+}
\ No newline at end of file
diff --git a/AsbCloudInfrastructure/Services/WellReport/WellReportService.cs b/AsbCloudInfrastructure/Services/WellReport/WellReportService.cs
new file mode 100644
index 00000000..5d34fa14
--- /dev/null
+++ b/AsbCloudInfrastructure/Services/WellReport/WellReportService.cs
@@ -0,0 +1,239 @@
+using System.Collections.Generic;
+using System.Linq;
+using System.Threading;
+using System.Threading.Tasks;
+using AsbCloudApp.Data;
+using AsbCloudApp.Data.ProcessMaps.Operations;
+using AsbCloudApp.Data.Trajectory;
+using AsbCloudApp.Data.WellOperation;
+using AsbCloudApp.Data.WellReport;
+using AsbCloudApp.Repositories;
+using AsbCloudApp.Requests;
+using AsbCloudApp.Services;
+using AsbCloudApp.Services.ProcessMaps.WellDrilling;
+using AsbCloudApp.Services.WellReport;
+using AsbCloudDb.Model;
+
+namespace AsbCloudInfrastructure.Services.WellReport;
+
+public class WellReportService : IWellReportService
+{
+ private readonly IWellService wellService;
+ private readonly IWellOperationService wellOperationService;
+ private readonly IWellContactService wellContactService;
+ private readonly IProcessMapReportDrillingService processMapReportDrillingService;
+ private readonly ISubsystemService subsystemService;
+
+ private readonly ITrajectoryRepository trajectoryPlanRepository;
+ private readonly ITrajectoryRepository trajectoryFactRepository;
+
+ private readonly IChangeLogRepository
+ processMapPlanRotorRepository;
+
+ private readonly IScheduleRepository scheduleRepository;
+
+ private IEnumerable factWellOperations;
+ private IEnumerable planWellOperations;
+
+ public WellReportService(IWellService wellService,
+ IWellOperationService wellOperationService,
+ IWellContactService wellContactService,
+ IProcessMapReportDrillingService processMapReportDrillingService,
+ ISubsystemService subsystemService,
+ ITrajectoryRepository trajectoryPlanRepository,
+ ITrajectoryRepository trajectoryFactRepository,
+ IChangeLogRepository processMapPlanRotorRepository,
+ IScheduleRepository scheduleRepository)
+ {
+ this.wellService = wellService;
+ this.wellOperationService = wellOperationService;
+ this.wellContactService = wellContactService;
+ this.processMapReportDrillingService = processMapReportDrillingService;
+ this.subsystemService = subsystemService;
+ this.trajectoryPlanRepository = trajectoryPlanRepository;
+ this.trajectoryFactRepository = trajectoryFactRepository;
+ this.processMapPlanRotorRepository = processMapPlanRotorRepository;
+ this.scheduleRepository = scheduleRepository;
+ }
+
+ public async Task GetAsync(int idWell, CancellationToken token)
+ {
+ var well = await wellService.GetOrDefaultAsync(idWell, token);
+
+ if (well == null)
+ return null;
+
+ await InitWellOperations(idWell, token);
+
+ var wellContactRequest = new WellContactRequest
+ {
+ IdsWells = new[] { idWell },
+ };
+
+ var contacts = await wellContactService.GetAllAsync(wellContactRequest, token);
+
+ var sectionReports = await GetSectionReportsAsync(idWell, token);
+ var drillerReports = await GetDrillerReportsAsync(idWell, token);
+
+ var firstFactOperation = factWellOperations.MinByOrDefault(x => x.DateStart);
+ var lastPlanOperation = planWellOperations.MaxByOrDefault(x => x.DateStart);
+
+ var planTrajectories = await trajectoryPlanRepository.GetAsync(idWell, token);
+ var factTrajectories = await trajectoryFactRepository.GetAsync(idWell, token);
+
+ var factOperationsWithoutNpt = factWellOperations.Where(o => !WellOperationCategory.NonProductiveTimeSubIds.Contains(o.IdCategory));
+
+ return new WellReportDto
+ {
+ Well = well,
+ DateFrom = firstFactOperation?.DateStart,
+ DateTo = lastPlanOperation?.DateStart.AddHours(lastPlanOperation.DurationHours),
+ Days = new PlanFactDto
+ {
+ Plan = planWellOperations.Sum(x => x.DurationHours) / 24,
+ Fact = factWellOperations.Sum(x => x.DurationHours) / 24
+ },
+ WellBoreDepth = new PlanFactDto
+ {
+ Plan = planWellOperations.MaxOrDefault(x => x.DepthEnd),
+ Fact = factWellOperations.MaxOrDefault(x => x.DepthEnd)
+ },
+ VerticalDepth = new PlanFactDto
+ {
+ Plan = planTrajectories.Max(x => x.VerticalDepth),
+ Fact = factTrajectories.Max(x => x.VerticalDepth)
+ },
+ WithoutNtpDays = factOperationsWithoutNpt.Sum(x => x.DurationHours) / 24,
+ Contacts = contacts,
+ SectionReports = sectionReports,
+ DrillerReports = drillerReports,
+ };
+ }
+
+ private async Task InitWellOperations(int idWell, CancellationToken token)
+ {
+ var request = new WellOperationRequest(new[] { idWell })
+ {
+ OperationType = WellOperation.IdOperationTypeFact
+ };
+
+ factWellOperations = await wellOperationService.GetAsync(request, token);
+
+ request.OperationType = WellOperation.IdOperationTypePlan;
+
+ planWellOperations = await wellOperationService.GetAsync(request, token);
+ }
+
+ private async Task> GetSectionReportsAsync(int idWell, CancellationToken token)
+ {
+ var factWellOperationsBySection = factWellOperations.GroupBy(x => x.IdWellSectionType)
+ .ToDictionary(x => x.Key, x => x.AsEnumerable());
+
+ var processMapPlanRequest = new ProcessMapPlanBaseRequestWithWell(idWell);
+
+ var processMapPlanRotorBySection =
+ (await processMapPlanRotorRepository.GetCurrent(processMapPlanRequest, token))
+ .GroupBy(x => x.IdWellSectionType)
+ .ToDictionary(x => x.Key, x => x.AsEnumerable());
+
+ var dataSaubStatRequest = new DataSaubStatRequest();
+
+ var processMapReportBySection =
+ (await processMapReportDrillingService.GetAsync(idWell, dataSaubStatRequest, token))
+ .GroupBy(x => x.IdWellSectionType)
+ .ToDictionary(x => x.Key, x => x.AsEnumerable());
+
+ var idsSection = factWellOperationsBySection.Keys
+ .Concat(processMapPlanRotorBySection.Keys)
+ .Concat(processMapReportBySection.Keys)
+ .Distinct();
+
+ var sectionReports = new List();
+
+ foreach (var idSection in idsSection)
+ {
+ var sectionReport = new SectionReportDto
+ {
+ IdSection = idSection,
+ OperatingMode = new PlanFactDto()
+ };
+
+ if (factWellOperationsBySection.TryGetValue(idSection, out var factOperations))
+ {
+ var subsystemRequest = new SubsystemRequest
+ {
+ IdWell = idWell,
+ GeDepth = factOperations.Min(y => y.DepthStart),
+ LeDepth = factOperations.Max(y => y.DepthEnd)
+ };
+
+ sectionReport.SubsystemsStat = await subsystemService.GetStatAsync(subsystemRequest, token);
+ sectionReport.OperatingMode.Fact = new OperatingModeDto
+ {
+ DepthStart = factOperations.Min(w => w.DepthStart),
+ DepthEnd = factOperations.Max(w => w.DepthEnd)
+ };
+ }
+
+ if (processMapPlanRotorBySection.TryGetValue(idSection, out var processMapPlanRotor))
+ sectionReport.OperatingMode.Plan = new OperatingModeDto
+ {
+ DepthStart = processMapPlanRotor.Min(p => p.DepthStart),
+ DepthEnd = processMapPlanRotor.Max(p => p.DepthEnd),
+ RopMin = processMapPlanRotor.Min(p => p.RopMax),
+ RopMax = processMapPlanRotor.Max(p => p.RopMax),
+ RopAvg = processMapPlanRotor.Average(p => p.RopMax),
+ WeightOnBitMin = processMapPlanRotor.Min(p => p.WeightOnBit),
+ WeightOnBitMax = processMapPlanRotor.Max(p => p.WeightOnBitMax),
+ WeightOnBitAvg = processMapPlanRotor.Average(p => p.WeightOnBit),
+ DriveTorqueMin = processMapPlanRotor.Min(p => p.TopDriveTorque),
+ DriveTorqueMax = processMapPlanRotor.Max(p => p.TopDriveTorqueMax),
+ DriveTorqueAvg = processMapPlanRotor.Average(p => p.TopDriveTorque),
+ DifferentialPressureMin = processMapPlanRotor.Min(p => p.DifferentialPressure),
+ DifferentialPressureMax = processMapPlanRotor.Max(p => p.DifferentialPressureMax),
+ DifferentialPressureAvg = processMapPlanRotor.Average(p => p.DifferentialPressure),
+ FrowRateMin = processMapPlanRotor.Min(p => p.FlowRate),
+ FrowRateMax = processMapPlanRotor.Max(p => p.FlowRateMax)
+ };
+
+ if (processMapReportBySection.TryGetValue(idSection, out var processMapReport))
+ sectionReport.DrillingBySetpoints = new DrillingBySetpointsDto
+ {
+ MetersByPressure = processMapReport.Sum(x => x.DeltaDepth * x.PressureDiff.SetpointUsage / 100),
+ MetersByLoad = processMapReport.Sum(x => x.DeltaDepth * x.AxialLoad.SetpointUsage / 100),
+ MetersByTorque = processMapReport.Sum(x => x.DeltaDepth * x.TopDriveTorque.SetpointUsage / 100),
+ MetersBySpeed = processMapReport.Sum(x => x.DeltaDepth * x.SpeedLimit.SetpointUsage / 100)
+ };
+
+ sectionReports.Add(sectionReport);
+ }
+
+ return sectionReports;
+ }
+
+ private async Task> GetDrillerReportsAsync(int idWell, CancellationToken token)
+ {
+ var schedules = await scheduleRepository.GetByIdWellAsync(idWell, token);
+
+ var result = new List();
+
+ foreach (var schedule in schedules)
+ {
+ var subsystemRequest = new SubsystemRequest
+ {
+ IdWell = idWell,
+ IdDriller = schedule.IdDriller
+ };
+
+ var drillerReport = new DrillerReportDto
+ {
+ Shedule = schedule,
+ SubsystemsStat = await subsystemService.GetStatAsync(subsystemRequest, token)
+ };
+
+ result.Add(drillerReport);
+ }
+
+ return result;
+ }
+}
\ No newline at end of file
diff --git a/AsbCloudInfrastructure/XLExtentions.cs b/AsbCloudInfrastructure/XLExtentions.cs
index 7c85c0a6..62992ec2 100644
--- a/AsbCloudInfrastructure/XLExtentions.cs
+++ b/AsbCloudInfrastructure/XLExtentions.cs
@@ -15,22 +15,25 @@ public static class XLExtentions
workbook.Worksheets.FirstOrDefault(ws => string.Equals(ws.Name.Trim(), sheetName.Trim(), StringComparison.CurrentCultureIgnoreCase))
?? throw new FileFormatException(string.Format(NotFoundSheetTemplate, sheetName));
- public static IXLCell SetCellValue(this IXLCell cell, T value, string? format = null)
+ public static IXLCell SetCellValue(this IXLCell cell, T? value, string? format = null)
{
- if (value is DateTime || value is DateTimeOffset)
- {
- cell.Style.DateFormat.Format = format ?? "DD.MM.YYYY HH:MM:SS";
+ if (value == null)
+ return cell;
- if (value is DateTimeOffset dateTimeOffset)
- {
- cell.Value = XLCellValue.FromObject(dateTimeOffset.DateTime);
- return cell;
- }
- }
+ if (value is DateTime || value is DateTimeOffset)
+ {
+ cell.Style.DateFormat.Format = format ?? "DD.MM.YYYY HH:MM:SS";
- cell.Value = XLCellValue.FromObject(value);
+ if (value is DateTimeOffset dateTimeOffset)
+ {
+ cell.Value = XLCellValue.FromObject(dateTimeOffset.DateTime);
+ return cell;
+ }
+ }
- return cell;
+ cell.Value = XLCellValue.FromObject(value);
+
+ return cell;
}
public static IXLCell SetHyperlink(this IXLCell cell, string link)
diff --git a/AsbCloudWebApi/Controllers/WellController.cs b/AsbCloudWebApi/Controllers/WellController.cs
index 98e44773..d6d97547 100644
--- a/AsbCloudWebApi/Controllers/WellController.cs
+++ b/AsbCloudWebApi/Controllers/WellController.cs
@@ -1,13 +1,13 @@
using AsbCloudApp.Data;
-using AsbCloudApp.Exceptions;
using AsbCloudApp.Services;
-using AsbCloudDb.Model;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Threading;
using System.Threading.Tasks;
+using AsbCloudApp.Services.WellReport;
+using Microsoft.AspNetCore.Http;
namespace AsbCloudWebApi.Controllers;
@@ -20,11 +20,13 @@ namespace AsbCloudWebApi.Controllers;
public class WellController : ControllerBase
{
private readonly IWellService wellService;
+ private readonly IWellReportExportService wellReportExportService;
- public WellController(IWellService wellService)
- {
- this.wellService = wellService;
- }
+ public WellController(IWellService wellService, IWellReportExportService wellReportExportService)
+ {
+ this.wellService = wellService;
+ this.wellReportExportService = wellReportExportService;
+ }
///
/// Возвращает список доступных скважин
@@ -149,4 +151,25 @@ public class WellController : ControllerBase
return Ok(result);
}
+
+ //TODO: навзание пока такое. У нас в API уже есть метод с такой сигнатурой.
+
+ ///
+ /// Получить отчёт по скважине
+ ///
+ ///
+ ///
+ ///
+ [HttpGet("{idWell}/report/export")]
+ [ProducesResponseType(typeof(PhysicalFileResult), StatusCodes.Status200OK, "application/octet-stream")]
+ [ProducesResponseType(StatusCodes.Status204NoContent)]
+ public async Task ExportAsync(int idWell, CancellationToken token)
+ {
+ var report = await wellReportExportService.ExportAsync(idWell, token);
+
+ if (report is null)
+ return NoContent();
+
+ return File(report.Value.File, "application/octet-stream", report.Value.Name);
+ }
}