DD.WellWorkover.Cloud/AsbCloudInfrastructure/Services/WellOperationService/WellOperationsStatService.cs

483 lines
19 KiB
C#
Raw Normal View History

using AsbCloudApp.Data;
using AsbCloudApp.Services;
using AsbCloudDb.Model;
using AsbCloudInfrastructure.Services.Cache;
2021-09-10 11:28:57 +05:00
using Mapster;
using Microsoft.EntityFrameworkCore;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
namespace AsbCloudInfrastructure.Services.WellOperationService
{
public class WellOperationsStatService : IWellOperationsStatService
{
private readonly IAsbCloudDbContext db;
2021-08-25 11:13:56 +05:00
private readonly IWellService wellService;
2021-08-25 17:58:35 +05:00
private readonly CacheTable<WellSectionType> cacheSectionsTypes;
private readonly CacheTable<Well> cacheWell;
private readonly CacheTable<WellType> cacheWellType;
2021-08-25 11:13:56 +05:00
private readonly CacheTable<Cluster> cacheCluster;
private const int idOperationBhaAssembly = 1025;
private const int idOperationBhaDisassembly = 1026;
private const int idOperationNonProductiveTime = 1043;
private const int idOperationDrilling = 1001;
private const int idOperationBhaDown = 1046;
private const int idOperationBhaUp = 1047;
private const int idOperationCasingDown = 1048;
private const int idOperationTypePlan = 0;
private const int idOperationTypeFact = 1;
2021-08-25 11:13:56 +05:00
public WellOperationsStatService(IAsbCloudDbContext db, CacheDb cache, IWellService wellService)
{
this.db = db;
2021-08-25 11:13:56 +05:00
this.wellService = wellService;
2021-08-25 17:58:35 +05:00
cacheSectionsTypes = cache.GetCachedTable<WellSectionType>((DbContext)db);
cacheWell = cache.GetCachedTable<Well>((DbContext)db);
cacheWellType = cache.GetCachedTable<WellType>((DbContext)db);
2021-08-25 11:13:56 +05:00
cacheCluster = cache.GetCachedTable<Cluster>((DbContext)db);
}
2021-08-25 11:30:50 +05:00
public async Task<StatClusterDto> GetStatClusterAsync(int idCluster, CancellationToken token = default)
{
var wells = await db.Wells
.Include(w => w.WellOperations)
.Where(o => o.IdCluster == idCluster)
.AsNoTracking()
.ToListAsync(token);
var statsWells = new List<StatWellDto>(wells.Count());
2021-08-25 11:13:56 +05:00
foreach (var well in wells)
{
var statWellDto = await CalcStatWellAsync(well, token);
2021-08-25 11:13:56 +05:00
statsWells.Add(statWellDto);
}
var cluster = await cacheCluster.FirstOrDefaultAsync(c => c.Id == idCluster, token);
var statClusterDto = new StatClusterDto
{
2021-08-25 11:13:56 +05:00
Id = idCluster,
Caption = cluster.Caption,
StatsWells = statsWells,
};
return statClusterDto;
}
public async Task<IEnumerable<StatWellDto>> GetWellsStatAsync(IEnumerable<int> idWells, CancellationToken token)
{
var wells = await db.Wells
.Include(w => w.WellOperations)
.Where(w => idWells.Contains(w.Id))
.Where(w => w.WellOperations.Any())
.AsNoTracking()
.ToListAsync(token);
var statsWells = new List<StatWellDto>(wells.Count);
foreach (var well in wells)
{
var statWellDto = await CalcStatWellAsync(well, token);
statsWells.Add(statWellDto);
}
return statsWells;
2021-08-25 11:13:56 +05:00
}
public async Task<StatWellDto> GetStatWellAsync(int idWell,
2021-08-25 11:13:56 +05:00
CancellationToken token = default)
{
var well = await db.Wells
.Include(w => w.WellOperations)
.FirstOrDefaultAsync(w => w.Id == idWell, token)
.ConfigureAwait(false);
2021-08-25 11:13:56 +05:00
var statWellDto = await CalcStatWellAsync(well, token);
return statWellDto;
}
2021-08-25 11:13:56 +05:00
private async Task<StatWellDto> CalcStatWellAsync(Well well, CancellationToken token = default)
{
var wellOperations = well.WellOperations
.OrderBy(o => o.DateStart)
.ThenBy(o => o.DepthEnd);
var wellType = await cacheWellType.FirstOrDefaultAsync(t => t.Id == well.IdWellType, token);
var statWellDto = new StatWellDto()
{
Id = well.Id,
2021-08-25 11:13:56 +05:00
Caption = well.Caption,
WellType = wellType.Caption
};
if (!wellOperations.Any())
return statWellDto;
statWellDto.Companies = await wellService.GetCompaniesAsync(well.Id, token);
statWellDto.Sections = CalcSectionsStats(wellOperations);
statWellDto.Total = GetStat(wellOperations);
return statWellDto;
}
2021-08-25 11:13:56 +05:00
private IEnumerable<StatSectionDto> CalcSectionsStats(IEnumerable<WellOperation> operations)
{
2021-08-25 11:13:56 +05:00
var sectionTypeIds = operations
.Select(o => o.IdWellSectionType)
.Distinct();
2021-08-25 17:58:35 +05:00
var sectionTypes = cacheSectionsTypes
2021-08-25 11:13:56 +05:00
.Where(s => sectionTypeIds.Contains(s.Id))
.ToDictionary(s => s.Id);
var sections = new List<StatSectionDto>(sectionTypes.Count);
var operationsPlan = operations.Where(o => o.IdType == idOperationTypePlan);
var operationsFact = operations.Where(o => o.IdType == idOperationTypeFact);
2021-08-25 11:13:56 +05:00
foreach ((var id, var sectionType) in sectionTypes)
{
var section = new StatSectionDto
2021-08-25 11:13:56 +05:00
{
Id = id,
2021-09-10 11:28:57 +05:00
Caption = sectionType.Caption,
2021-08-25 11:13:56 +05:00
Plan = CalcSectionStat(operationsPlan, id),
Fact = CalcSectionStat(operationsFact, id),
};
sections.Add(section);
}
return sections;
}
2021-08-25 11:13:56 +05:00
private static PlanFactBase<StatOperationsDto> GetStat(IEnumerable<WellOperation> operations)
{
var operationsPlan = operations.Where(o => o.IdType == idOperationTypePlan);
var operationsFact = operations.Where(o => o.IdType == idOperationTypeFact);
2021-08-25 11:13:56 +05:00
var section = new PlanFactBase<StatOperationsDto>
{
Plan = CalcStat(operationsPlan),
Fact = CalcStat(operationsFact),
};
return section;
}
private static StatOperationsDto CalcSectionStat(IEnumerable<WellOperation> operations, int idSectionType)
{
var sectionOperations = operations
.Where(o => o.IdWellSectionType == idSectionType)
.OrderBy(o => o.DateStart)
.ThenBy(o => o.DepthStart);
2021-09-10 11:28:57 +05:00
return CalcStat(sectionOperations);
}
private static StatOperationsDto CalcStat(IEnumerable<WellOperation> operations)
{
2021-08-29 12:05:43 +05:00
if (!operations.Any())
return null;
2021-08-26 10:18:59 +05:00
var races = GetCompleteRaces(operations);
2021-08-25 11:13:56 +05:00
var section = new StatOperationsDto
{
Start = operations.First().DateStart,
End = operations.Max(o => (o.DateStart.AddHours(o.DurationHours))),
WellDepthStart = operations.Min(o => o.DepthStart),
WellDepthEnd = operations.Max(o => o.DepthStart),
Rop = CalcROP(operations),
2021-08-26 10:18:59 +05:00
RouteSpeed = CalcAvgRaceSpeed(races),
BhaDownSpeed = CalcBhaDownSpeed(races),
BhaUpSpeed = CalcBhaUpSpeed(races),
CasingDownSpeed = CalcCasingDownSpeed(operations),
NonProductiveHours = operations
.Where(o => o.IdCategory == idOperationNonProductiveTime)
.Sum(o => o.DurationHours),
};
return section;
}
private static double CalcROP(IEnumerable<WellOperation> operationsProps)
{
var drillingOperations = operationsProps.Where(o => o.IdCategory == idOperationDrilling);
var dDepth = 0d;
var dHours = 0d;
foreach (var operation in drillingOperations)
{
var deltaDepth = operation.DepthEnd - operation.DepthStart;
dDepth += deltaDepth;
dHours += operation.DurationHours;
}
return dDepth / (dHours + double.Epsilon);
}
private static double CalcCasingDownSpeed(IEnumerable<WellOperation> operationsProps)
{
var ops = operationsProps.Where(o => o.IdCategory == idOperationCasingDown);
2021-08-26 10:18:59 +05:00
var depth = 0d;
var dHours = 0d;
2021-08-26 10:18:59 +05:00
foreach (var operation in ops)
{
depth += operation.DepthStart;
dHours += operation.DurationHours;
}
2021-08-26 10:18:59 +05:00
return depth / (dHours + double.Epsilon);
}
private static IEnumerable<Race> GetCompleteRaces(IEnumerable<WellOperation> operations)
{
var races = new List<Race>();
2021-08-25 17:58:35 +05:00
var iterator = operations
.OrderBy(o => o.DateStart)
2021-08-25 17:58:35 +05:00
.GetEnumerator();
while (iterator.MoveNext())
{
if (iterator.Current.IdCategory == idOperationBhaAssembly)
{
var race = new Race
{
StartDate = iterator.Current.DateStart.AddHours(iterator.Current.DurationHours),
StartWellDepth = iterator.Current.DepthStart,
Operations = new List<WellOperation>(10),
};
while (iterator.MoveNext())
{
if (iterator.Current.IdCategory == idOperationNonProductiveTime)
{
race.NonProductiveHours += iterator.Current.DurationHours;
}
if (iterator.Current.IdCategory == idOperationBhaDisassembly)
{
race.EndDate = iterator.Current.DateStart;
race.EndWellDepth = iterator.Current.DepthStart;
races.Add(race);
2021-08-25 17:58:35 +05:00
break;
}
2021-08-27 15:53:38 +05:00
race.Operations.Add(iterator.Current);
}
}
}
return races;
}
2021-08-26 10:18:59 +05:00
private static double CalcAvgRaceSpeed(IEnumerable<Race> races)
{
var dDepth = 0d;
var dHours = 0d;
foreach (var race in races)
{
dHours += race.DeltaHoursTimeNoNpt;
dDepth += race.DeltaDepth;
}
return dDepth / (dHours + double.Epsilon);
}
private static double CalcBhaDownSpeed(IEnumerable<Race> races)
{
var dDepth = 0d;
var dHours = 0d;
foreach (var race in races)
{
dDepth += race.StartWellDepth;
2021-08-27 15:53:38 +05:00
for (var i = 0; i < race.Operations.Count; i++)
2021-08-26 10:18:59 +05:00
{
2021-08-27 15:53:38 +05:00
if (race.Operations[i].IdCategory == idOperationBhaDown)
dHours += race.Operations[i].DurationHours;
2021-08-27 15:53:38 +05:00
if (race.Operations[i].IdCategory == idOperationDrilling)
2021-08-26 10:18:59 +05:00
break;
}
}
return dDepth / (dHours + double.Epsilon);
}
private static double CalcBhaUpSpeed(IEnumerable<Race> races)
{
var dDepth = 0d;
var dHours = 0d;
foreach (var race in races)
{
dDepth += race.EndWellDepth;
2021-09-10 11:28:57 +05:00
for (var i = race.Operations.Count - 1; i > 0; i--)
2021-08-26 10:18:59 +05:00
{
2021-08-27 15:53:38 +05:00
if (race.Operations[i].IdCategory == idOperationBhaUp)
dHours += race.Operations[i].DurationHours;
2021-08-27 15:53:38 +05:00
if (race.Operations[i].IdCategory == idOperationDrilling)
2021-08-26 10:18:59 +05:00
break;
}
}
return dDepth / (dHours + double.Epsilon);
}
public async Task<IEnumerable<PlanFactPredictBase<WellOperationDto>>> GetTvdAsync(int idWell, CancellationToken token)
{
var wellOperations = await db.WellOperations
.Include(o => o.OperationCategory)
.Include(o => o.WellSectionType)
.Where(o => o.IdWell == idWell)
.OrderBy(o => o.DateStart)
.ThenBy(o => o.DepthEnd)
.AsNoTracking()
.ToListAsync(token)
.ConfigureAwait(false);
var wellOperationsPlan = wellOperations
.Where(o => o.IdType == idOperationTypePlan)
.OrderBy(o => o.DateStart)
.ThenBy(o => o.DepthEnd);
var wellOperationsFact = wellOperations
.Where(o => o.IdType == idOperationTypeFact)
.OrderBy(o => o.DateStart)
.ThenBy(o => o.DepthEnd);
2021-10-06 16:30:46 +05:00
var sectionsIds = wellOperations
.Select(o => o.IdWellSectionType)
.Distinct();
if (!wellOperationsPlan.Any())
return null;
2021-10-06 16:30:46 +05:00
var merged = MergeArraysBySections(sectionsIds, wellOperationsPlan, wellOperationsFact);
var tvd = new List<PlanFactPredictBase<WellOperationDto>>(merged.Count);
int iLastMatch = 0;
int iLastFact = 0;
for (int i = 0; i < merged.Count; i++)
{
var item = merged[i];
var planFactPredict = new PlanFactPredictBase<WellOperationDto>
{
Plan = item.Item1?.Adapt(WellOperationDtoMutation),
Fact = item.Item2?.Adapt(WellOperationDtoMutation),
Predict = null,
};
tvd.Add(planFactPredict);
if ((item.Item1 is not null) && (item.Item2 is not null))
iLastMatch = i;
if (item.Item2 is not null)
iLastFact = i;
}
if (iLastMatch == 0 || iLastMatch == merged.Count - 1)
return tvd;
var lastMatchPlan = merged[iLastMatch].Item1;
var lastMatchPlanOperationEnd = lastMatchPlan.DateStart.AddHours(lastMatchPlan.DurationHours);
//var lastMatchFact = merged[iLastMatch].Item2;
//var lastMatchFactDateEnd = lastMatchFact.DateStart.AddHours(lastMatchFact.DurationHours);
var lastFact = merged[iLastFact].Item2;
var lastFactDateEnd = lastFact.DateStart.AddHours(lastFact.DurationHours);
var startOffset = lastFactDateEnd - lastMatchPlanOperationEnd;
for (int i = iLastMatch + 1; i < merged.Count; i++)
{
if (merged[i].Item1 is null)
continue;
tvd[i].Predict = merged[i].Item1.Adapt<WellOperationDto>();
tvd[i].Predict.IdType = 2;
tvd[i].Predict.DateStart = tvd[i].Predict.DateStart + startOffset;
}
return tvd;
}
2021-10-06 16:30:46 +05:00
private List<Tuple<WellOperation, WellOperation>> MergeArraysBySections(
IEnumerable<int> sectionsIds,
2021-10-06 16:30:46 +05:00
IOrderedEnumerable<WellOperation> wellOperationsPlan,
IOrderedEnumerable<WellOperation> wellOperationsFact)
{
var merged = new List<Tuple<WellOperation, WellOperation>>(wellOperationsPlan.Count());
foreach (var sectionId in sectionsIds)
{
var sectionOperationsPlan = wellOperationsPlan
.Where(o => o.IdWellSectionType == sectionId);
var sectionOperationsFact = wellOperationsFact
.Where(o => o.IdWellSectionType == sectionId);
var sectionMerged = MergeArrays(sectionOperationsPlan, sectionOperationsFact);
merged.AddRange(sectionMerged);
}
return merged;
}
private static List<Tuple<WellOperation, WellOperation>> MergeArrays(IEnumerable<WellOperation> array1, IEnumerable<WellOperation> array2)
{
var a1 = array1.ToArray();
var a2 = array2.ToArray();
var m = new List<Tuple<WellOperation, WellOperation>>(a1.Length);
void Add(WellOperation item1, WellOperation item2) =>
m.Add(new Tuple<WellOperation, WellOperation>(item1, item2));
2021-10-06 16:30:46 +05:00
static bool Compare(WellOperation item1, WellOperation item2) =>
item1.IdCategory == item2.IdCategory && Math.Abs(item1.DepthEnd - item2.DepthEnd) < (30d + 0.005d * (item1.DepthEnd + item2.DepthEnd));
int i1 = 0;
int i2 = 0;
while (true)
{
var is1 = a1.Length > i1;
var is2 = a2.Length > i2;
if (!(is1 || is2))
break;
if (is1 && is2)
{
if (Compare(a1[i1], a2[i2]))
Add(a1[i1++], a2[i2++]);
else
{
int nextI1 = Array.FindIndex(a1, i1, (item) => Compare(item, a2[i2]));
int nextI2 = Array.FindIndex(a2, i2, (item) => Compare(item, a1[i1]));
2021-10-06 16:30:46 +05:00
bool deltaI1_Lt_deltaI2 = (nextI1 - i1) < (nextI2 - i2);
if (nextI1 == -1 && nextI2 == -1)
{
if (a1[i1].DepthEnd < a2[i2].DepthEnd)
2021-10-06 16:30:46 +05:00
{
Add(a1[i1++], null);
}
else
{
Add(null, a2[i2++]);
}
}
2021-10-06 16:30:46 +05:00
else if (nextI1 > -1 && nextI2 == -1)
{
2021-10-06 16:30:46 +05:00
Add(a1[i1++], null);
}
2021-10-06 16:30:46 +05:00
else if (nextI1 == -1 && nextI2 > -1)
{
2021-10-06 16:30:46 +05:00
Add(null, a2[i2++]);
}
else if (deltaI1_Lt_deltaI2)
{
Add(a1[i1++], null);
}
else if (!deltaI1_Lt_deltaI2)
{
Add(null, a2[i2++]);
}
}
}
else if (is1)
{
Add(a1[i1++], null);
}
else if (is2)
{
Add(null, a2[i2++]);
}
}
return m;
}
2021-09-10 11:28:57 +05:00
private static readonly Action<WellOperationDto, WellOperation> WellOperationDtoMutation = (WellOperationDto dest, WellOperation source) =>
{
dest.CategoryName = source.OperationCategory?.Name;
dest.WellSectionTypeName = source.WellSectionType?.Caption;
};
}
}