2023-02-16 16:27:14 +05:00
|
|
|
|
using AsbCloudApp.Data;
|
2024-02-20 15:54:18 +05:00
|
|
|
|
using AsbCloudApp.Data.ProcessMapPlan;
|
2023-05-22 10:20:45 +05:00
|
|
|
|
using AsbCloudApp.Data.SAUB;
|
2023-06-29 15:54:54 +05:00
|
|
|
|
using AsbCloudApp.Data.WITS;
|
2024-02-20 15:54:18 +05:00
|
|
|
|
using AsbCloudApp.IntegrationEvents;
|
|
|
|
|
using AsbCloudApp.IntegrationEvents.Interfaces;
|
2023-02-16 16:27:14 +05:00
|
|
|
|
using AsbCloudApp.Repositories;
|
|
|
|
|
using AsbCloudApp.Requests;
|
|
|
|
|
using AsbCloudApp.Services;
|
|
|
|
|
using AsbCloudInfrastructure.Background;
|
|
|
|
|
using Mapster;
|
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
namespace AsbCloudInfrastructure.Services;
|
|
|
|
|
|
|
|
|
|
public class WellInfoService
|
2023-02-16 16:27:14 +05:00
|
|
|
|
{
|
2023-10-09 13:12:45 +05:00
|
|
|
|
public class WorkWellInfoUpdate : Work
|
2023-02-16 16:27:14 +05:00
|
|
|
|
{
|
2023-10-09 13:12:45 +05:00
|
|
|
|
public WorkWellInfoUpdate()
|
|
|
|
|
: base("Well statistics update")
|
2023-02-16 16:27:14 +05:00
|
|
|
|
{
|
2023-10-10 13:45:30 +05:00
|
|
|
|
Timeout = TimeSpan.FromMinutes(30);
|
2023-02-16 16:27:14 +05:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
protected override async Task Action(string id, IServiceProvider services, Action<string, double?> onProgressCallback, CancellationToken token)
|
2023-02-16 16:27:14 +05:00
|
|
|
|
{
|
2023-10-09 13:12:45 +05:00
|
|
|
|
var wellService = services.GetRequiredService<IWellService>();
|
|
|
|
|
var operationsStatService = services.GetRequiredService<IOperationsStatService>();
|
2024-02-20 15:54:18 +05:00
|
|
|
|
var processMapPlanWellDrillingRepository = services.GetRequiredService<IChangeLogRepository<ProcessMapPlanDrillingDto, ProcessMapPlanBaseRequestWithWell>>();
|
2023-12-16 13:22:23 +05:00
|
|
|
|
var subsystemService = services.GetRequiredService<ISubsystemService>();
|
2023-10-24 09:23:07 +05:00
|
|
|
|
var telemetryDataSaubCache = services.GetRequiredService<ITelemetryDataCache<TelemetryDataSaubDto>>();
|
2023-10-09 13:12:45 +05:00
|
|
|
|
var messageHub = services.GetRequiredService<IIntegrationEventHandler<UpdateWellInfoEvent>>();
|
2023-05-22 10:20:45 +05:00
|
|
|
|
|
2023-08-24 10:50:34 +05:00
|
|
|
|
var wells = await wellService.GetAllAsync(token);
|
2023-10-10 13:45:30 +05:00
|
|
|
|
var activeWells = wells.Where(well => well.IdState == 1);
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2023-10-10 13:45:30 +05:00
|
|
|
|
var wellsIds = activeWells.Select(w => w.Id);
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2024-02-20 15:54:18 +05:00
|
|
|
|
var processMapPlanWellDrillingRequests = wellsIds.Select(id => new ProcessMapPlanBaseRequestWithWell(id));
|
|
|
|
|
var processMapPlanWellDrillings = new List<ProcessMapPlanDrillingDto>();
|
|
|
|
|
foreach (var processMapPlanWellDrillingRequest in processMapPlanWellDrillingRequests)
|
|
|
|
|
{
|
|
|
|
|
var processMaps = await processMapPlanWellDrillingRepository.Get(processMapPlanWellDrillingRequest, token);
|
|
|
|
|
processMapPlanWellDrillings.AddRange(processMaps);
|
|
|
|
|
}
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2023-10-12 15:47:26 +05:00
|
|
|
|
var wellDepthByProcessMap = processMapPlanWellDrillings
|
2023-02-16 16:27:14 +05:00
|
|
|
|
.GroupBy(p => p.IdWell)
|
|
|
|
|
.Select(g => new
|
|
|
|
|
{
|
|
|
|
|
Id = g.Key,
|
|
|
|
|
DepthEnd = g.Max(p => p.DepthEnd)
|
|
|
|
|
});
|
|
|
|
|
|
2023-08-24 10:50:34 +05:00
|
|
|
|
var operationsStat = await operationsStatService.GetWellsStatAsync(wellsIds, token);
|
2023-10-09 13:12:45 +05:00
|
|
|
|
|
2023-12-16 13:22:23 +05:00
|
|
|
|
var subsystemStat = await subsystemService
|
2023-08-24 10:50:34 +05:00
|
|
|
|
.GetStatByActiveWells(wellsIds, token);
|
2023-10-10 15:06:58 +05:00
|
|
|
|
subsystemStat = subsystemStat.ToArray();
|
|
|
|
|
|
2023-10-10 13:45:30 +05:00
|
|
|
|
var count = activeWells.Count();
|
2023-10-09 13:12:45 +05:00
|
|
|
|
var i = 0d;
|
2024-02-20 15:54:18 +05:00
|
|
|
|
WellMapInfo = activeWells.Select(well =>
|
|
|
|
|
{
|
2023-02-16 16:27:14 +05:00
|
|
|
|
var wellMapInfo = well.Adapt<WellMapInfoWithComanies>();
|
2023-06-29 15:54:54 +05:00
|
|
|
|
wellMapInfo.IdState = well.IdState;
|
2023-10-09 13:12:45 +05:00
|
|
|
|
onProgressCallback($"Start updating info by well({well.Id}): {well.Caption}", i++ / count);
|
2023-05-22 10:20:45 +05:00
|
|
|
|
double? currentDepth = null;
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2023-08-18 15:51:58 +05:00
|
|
|
|
TelemetryDataSaubDto? lastSaubTelemetry = null;
|
2023-10-09 13:12:45 +05:00
|
|
|
|
|
2023-05-22 10:20:45 +05:00
|
|
|
|
if (well.IdTelemetry.HasValue)
|
|
|
|
|
{
|
2023-06-29 15:54:54 +05:00
|
|
|
|
wellMapInfo.IdTelemetry = well.IdTelemetry.Value;
|
2023-08-18 15:51:58 +05:00
|
|
|
|
lastSaubTelemetry = telemetryDataSaubCache.GetLastOrDefault(well.IdTelemetry.Value);
|
2023-10-09 13:12:45 +05:00
|
|
|
|
if (lastSaubTelemetry is not null)
|
2023-05-22 10:20:45 +05:00
|
|
|
|
{
|
|
|
|
|
currentDepth = lastSaubTelemetry.WellDepth;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
var wellOperationsStat = operationsStat.FirstOrDefault(s => s.Id == well.Id);
|
2023-11-08 10:46:39 +05:00
|
|
|
|
var wellLastFactSection = wellOperationsStat?.Sections.OrderBy(s => s.Fact?.WellDepthStart).LastOrDefault(s => s.Fact is not null);
|
2023-05-22 10:20:45 +05:00
|
|
|
|
currentDepth ??= wellLastFactSection?.Fact?.WellDepthEnd;
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2023-10-12 15:47:26 +05:00
|
|
|
|
var wellProcessMaps = processMapPlanWellDrillings
|
2023-02-16 16:27:14 +05:00
|
|
|
|
.Where(p => p.IdWell == well.Id)
|
|
|
|
|
.OrderBy(p => p.DepthEnd);
|
|
|
|
|
|
|
|
|
|
int? idSection = wellLastFactSection?.Id;
|
2024-02-20 15:54:18 +05:00
|
|
|
|
ProcessMapPlanDrillingDto? processMapPlanWellDrilling = null;
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2023-05-22 10:20:45 +05:00
|
|
|
|
if (idSection.HasValue)
|
2023-02-16 16:27:14 +05:00
|
|
|
|
{
|
2023-10-12 15:47:26 +05:00
|
|
|
|
processMapPlanWellDrilling = wellProcessMaps.FirstOrDefault(p => p.IdWellSectionType == idSection);
|
2023-02-16 16:27:14 +05:00
|
|
|
|
}
|
2023-10-09 13:12:45 +05:00
|
|
|
|
else if (currentDepth.HasValue)
|
2023-02-16 16:27:14 +05:00
|
|
|
|
{
|
2023-10-12 15:47:26 +05:00
|
|
|
|
processMapPlanWellDrilling = wellProcessMaps.FirstOrDefault(p => p.DepthStart <= currentDepth.Value && p.DepthEnd >= currentDepth.Value);
|
2023-02-16 16:27:14 +05:00
|
|
|
|
}
|
|
|
|
|
|
2023-06-29 15:54:54 +05:00
|
|
|
|
double? planTotalDepth = null;
|
2023-05-22 10:52:27 +05:00
|
|
|
|
planTotalDepth = wellDepthByProcessMap.FirstOrDefault(p => p.Id == well.Id)?.DepthEnd;
|
|
|
|
|
planTotalDepth ??= wellOperationsStat?.Total.Plan?.WellDepthEnd;
|
|
|
|
|
|
2023-08-24 10:50:34 +05:00
|
|
|
|
wellMapInfo.Section = wellLastFactSection?.Caption;
|
2023-10-09 13:12:45 +05:00
|
|
|
|
|
|
|
|
|
wellMapInfo.FirstFactOperationDateStart = wellOperationsStat?.Total.Fact?.Start
|
2023-06-29 15:54:54 +05:00
|
|
|
|
?? wellOperationsStat?.Total.Plan?.Start;
|
|
|
|
|
|
2023-08-02 11:34:42 +05:00
|
|
|
|
wellMapInfo.LastPredictOperationDateEnd = wellOperationsStat?.Total.Plan?.End;
|
2023-06-29 15:54:54 +05:00
|
|
|
|
|
2023-08-18 15:51:58 +05:00
|
|
|
|
wellMapInfo.AxialLoad = new()
|
|
|
|
|
{
|
2024-02-20 15:54:18 +05:00
|
|
|
|
Plan = processMapPlanWellDrilling?.AxialLoadPlan,
|
2023-08-18 15:51:58 +05:00
|
|
|
|
Fact = lastSaubTelemetry?.AxialLoad
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
wellMapInfo.TopDriveSpeed = new()
|
|
|
|
|
{
|
2024-02-20 15:54:18 +05:00
|
|
|
|
Plan = processMapPlanWellDrilling?.TopDriveSpeedPlan,
|
2023-08-18 15:51:58 +05:00
|
|
|
|
Fact = lastSaubTelemetry?.RotorSpeed
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
wellMapInfo.TopDriveTorque = new()
|
|
|
|
|
{
|
2024-02-20 15:54:18 +05:00
|
|
|
|
Plan = processMapPlanWellDrilling?.TopDriveTorquePlan,
|
2023-08-18 15:51:58 +05:00
|
|
|
|
Fact = lastSaubTelemetry?.RotorTorque
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
wellMapInfo.Pressure = new()
|
|
|
|
|
{
|
2024-02-20 15:54:18 +05:00
|
|
|
|
Plan = processMapPlanWellDrilling?.DeltaPressurePlan,
|
2023-08-18 15:51:58 +05:00
|
|
|
|
Fact = lastSaubTelemetry?.Pressure
|
|
|
|
|
};
|
2023-10-09 13:12:45 +05:00
|
|
|
|
|
2023-08-18 15:51:58 +05:00
|
|
|
|
wellMapInfo.PressureSp = lastSaubTelemetry?.PressureSp;
|
|
|
|
|
|
2023-02-16 16:27:14 +05:00
|
|
|
|
wellMapInfo.WellDepth = new()
|
|
|
|
|
{
|
2023-05-22 10:52:27 +05:00
|
|
|
|
Plan = planTotalDepth,
|
2023-02-16 16:27:14 +05:00
|
|
|
|
Fact = currentDepth,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
wellMapInfo.ROP = new()
|
|
|
|
|
{
|
2023-10-12 15:47:26 +05:00
|
|
|
|
Plan = processMapPlanWellDrilling?.RopPlan,
|
2023-02-16 16:27:14 +05:00
|
|
|
|
Fact = wellOperationsStat?.Total.Fact?.Rop,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
wellMapInfo.RaceSpeed = new()
|
|
|
|
|
{
|
|
|
|
|
Plan = wellOperationsStat?.Total.Plan?.RouteSpeed,
|
|
|
|
|
Fact = wellOperationsStat?.Total.Fact?.RouteSpeed,
|
|
|
|
|
};
|
|
|
|
|
|
2023-05-22 10:20:45 +05:00
|
|
|
|
var wellSubsystemStat = subsystemStat.FirstOrDefault(s => s.Well.Id == well.Id);
|
2023-12-16 13:22:23 +05:00
|
|
|
|
wellMapInfo.SaubUsage = wellSubsystemStat?.SubsystemAPD?.KUsage ?? 0d;
|
2023-12-18 13:53:51 +05:00
|
|
|
|
wellMapInfo.SpinUsage = wellSubsystemStat?.SubsystemOscillation?.KUsage ?? 0d;
|
2023-06-29 15:54:54 +05:00
|
|
|
|
wellMapInfo.TorqueKUsage = wellSubsystemStat?.SubsystemTorqueMaster?.KUsage ?? 0d;
|
2023-09-27 13:49:55 +05:00
|
|
|
|
wellMapInfo.TvdLagDays = wellOperationsStat?.TvdLagDays;
|
|
|
|
|
wellMapInfo.TvdDrillingDays = wellOperationsStat?.TvdDrillingDays;
|
2023-02-16 16:27:14 +05:00
|
|
|
|
wellMapInfo.IdsCompanies = well.Companies.Select(c => c.Id);
|
2023-06-29 15:54:54 +05:00
|
|
|
|
|
2023-02-16 16:27:14 +05:00
|
|
|
|
return wellMapInfo;
|
|
|
|
|
}).ToArray();
|
2023-08-16 17:30:03 +05:00
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
var updateWellInfoEventTasks = wellsIds.Select(idWell =>
|
2023-08-24 10:50:34 +05:00
|
|
|
|
messageHub.HandleAsync(new UpdateWellInfoEvent(idWell), token));
|
2023-10-09 13:12:45 +05:00
|
|
|
|
|
2023-08-21 17:37:17 +05:00
|
|
|
|
await Task.WhenAll(updateWellInfoEventTasks);
|
2023-02-16 16:27:14 +05:00
|
|
|
|
}
|
2023-10-09 13:12:45 +05:00
|
|
|
|
}
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
class WellMapInfoWithComanies : WellMapInfoDto
|
|
|
|
|
{
|
|
|
|
|
public int? IdTelemetry { get; set; }
|
|
|
|
|
public IEnumerable<int> IdsCompanies { get; set; } = null!;
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-24 09:23:07 +05:00
|
|
|
|
private readonly ITelemetryDataCache<TelemetryDataSaubDto> telemetryDataSaubCache;
|
|
|
|
|
private readonly ITelemetryDataCache<TelemetryDataSpinDto> telemetryDataSpinCache;
|
2023-10-09 13:12:45 +05:00
|
|
|
|
private readonly IWitsRecordRepository<Record7Dto> witsRecord7Repository;
|
|
|
|
|
private readonly IWitsRecordRepository<Record1Dto> witsRecord1Repository;
|
|
|
|
|
private readonly IGtrRepository gtrRepository;
|
|
|
|
|
private static IEnumerable<WellMapInfoWithComanies> WellMapInfo = Enumerable.Empty<WellMapInfoWithComanies>();
|
|
|
|
|
|
|
|
|
|
public WellInfoService(
|
2023-10-24 09:23:07 +05:00
|
|
|
|
ITelemetryDataCache<TelemetryDataSaubDto> telemetryDataSaubCache,
|
|
|
|
|
ITelemetryDataCache<TelemetryDataSpinDto> telemetryDataSpinCache,
|
2023-10-09 13:12:45 +05:00
|
|
|
|
IWitsRecordRepository<Record7Dto> witsRecord7Repository,
|
|
|
|
|
IWitsRecordRepository<Record1Dto> witsRecord1Repository,
|
|
|
|
|
IGtrRepository gtrRepository)
|
|
|
|
|
{
|
|
|
|
|
this.telemetryDataSaubCache = telemetryDataSaubCache;
|
|
|
|
|
this.telemetryDataSpinCache = telemetryDataSpinCache;
|
|
|
|
|
|
|
|
|
|
this.witsRecord7Repository = witsRecord7Repository;
|
|
|
|
|
this.witsRecord1Repository = witsRecord1Repository;
|
|
|
|
|
this.gtrRepository = gtrRepository;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private WellMapInfoWithTelemetryStat Convert(WellMapInfoWithComanies wellInfo)
|
|
|
|
|
{
|
|
|
|
|
var result = wellInfo.Adapt<WellMapInfoWithTelemetryStat>();
|
|
|
|
|
if (wellInfo.IdTelemetry.HasValue)
|
2023-06-29 15:54:54 +05:00
|
|
|
|
{
|
2023-10-09 13:12:45 +05:00
|
|
|
|
var idTelemetry = wellInfo.IdTelemetry.Value;
|
|
|
|
|
result.LastDataSaub = telemetryDataSaubCache.GetLastOrDefault(idTelemetry);
|
|
|
|
|
result.LastDataSpin = telemetryDataSpinCache.GetLastOrDefault(idTelemetry);
|
|
|
|
|
result.LastDataDdsDate = GetLastOrDefaultDdsTelemetry(idTelemetry);
|
|
|
|
|
result.LastDataGtrDate = gtrRepository.GetLastData(wellInfo.Id)
|
|
|
|
|
.MaxOrDefault(item => item.Date);
|
|
|
|
|
result.LastDataDpcsDate = null;
|
|
|
|
|
result.LastDataDpcsDate = null;
|
2023-06-29 15:54:54 +05:00
|
|
|
|
}
|
2023-02-16 16:27:14 +05:00
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
return result;
|
|
|
|
|
}
|
2023-04-18 16:22:53 +05:00
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
private DateTime? GetLastOrDefaultDdsTelemetry(int idTelemetry)
|
|
|
|
|
{
|
|
|
|
|
var lastDdsRecord1Date = witsRecord1Repository.GetLastOrDefault(idTelemetry)?.DateTime;
|
|
|
|
|
var lastDdsRecord7Date = witsRecord7Repository.GetLastOrDefault(idTelemetry)?.DateTime;
|
2023-06-29 15:54:54 +05:00
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
if (lastDdsRecord1Date.HasValue && lastDdsRecord7Date.HasValue)
|
|
|
|
|
if (lastDdsRecord1Date.Value > lastDdsRecord7Date.Value)
|
|
|
|
|
return lastDdsRecord1Date.Value;
|
|
|
|
|
else
|
|
|
|
|
return lastDdsRecord7Date.Value;
|
2023-06-29 15:54:54 +05:00
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
return lastDdsRecord1Date ?? lastDdsRecord7Date;
|
|
|
|
|
}
|
2023-06-29 15:54:54 +05:00
|
|
|
|
|
2023-10-09 13:12:45 +05:00
|
|
|
|
public WellMapInfoWithTelemetryStat? FirstOrDefault(Func<WellMapInfoDto, bool> predicate)
|
2024-02-20 15:54:18 +05:00
|
|
|
|
{
|
2023-10-09 13:12:45 +05:00
|
|
|
|
var first = WellMapInfo.FirstOrDefault(predicate);
|
|
|
|
|
if (first is WellMapInfoWithComanies wellMapInfoWithComanies)
|
|
|
|
|
return Convert(wellMapInfoWithComanies);
|
|
|
|
|
|
|
|
|
|
return null;
|
2023-06-29 15:54:54 +05:00
|
|
|
|
}
|
2023-02-16 16:27:14 +05:00
|
|
|
|
}
|