2023-10-20 11:03:14 +05:00
|
|
|
using System;
|
2023-12-05 10:56:49 +05:00
|
|
|
using System.Collections.Generic;
|
2023-10-20 11:03:14 +05:00
|
|
|
using System.Linq;
|
|
|
|
using AsbCloudDb.Model;
|
|
|
|
|
|
|
|
namespace AsbCloudInfrastructure.Services.DetectOperations.Detectors;
|
|
|
|
|
2023-11-22 14:47:17 +05:00
|
|
|
public class DetectorDrilling : DetectorAbstract
|
2023-10-20 11:03:14 +05:00
|
|
|
{
|
2023-12-04 17:36:00 +05:00
|
|
|
private const double dispersionOfNormalizedRotorSpeedThreshold = 0.2d;
|
|
|
|
public const string ExtraDataKeyHasOscillation = "hasOscillation";
|
|
|
|
public const string ExtraDataKeyDispersionOfNormalizedRotorSpeed = "dispersionOfNormalizedRotorSpeed";
|
|
|
|
public const string ExtraDataKeyAvgRotorSpeed = "avgRotorSpeed";
|
2023-11-28 11:23:30 +05:00
|
|
|
|
2023-12-04 17:36:00 +05:00
|
|
|
protected override bool DetectBegin(DetectableTelemetry[] telemetry, int position, DetectedOperation? previousOperation)
|
2023-10-20 11:03:14 +05:00
|
|
|
{
|
|
|
|
var point0 = telemetry[position];
|
|
|
|
var delta = point0.WellDepth - point0.BitDepth;
|
|
|
|
if (delta > 0.03d)
|
|
|
|
return false;
|
|
|
|
|
2023-11-23 09:47:10 +05:00
|
|
|
if (point0.Pressure < 18)
|
2023-10-20 11:03:14 +05:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override int DetectEnd(DetectableTelemetry[] telemetry, int position, DetectedOperation? previousOperation)
|
|
|
|
{
|
|
|
|
var point0 = telemetry[position];
|
|
|
|
var delta = point0.WellDepth - point0.BitDepth;
|
|
|
|
|
|
|
|
if (delta > 0.03d)
|
|
|
|
return IdReasonOfEnd_DeltaDepthIsHi;
|
|
|
|
|
2023-11-23 09:47:10 +05:00
|
|
|
if (point0.Pressure < 18)
|
2023-10-20 11:03:14 +05:00
|
|
|
return IdReasonOfEnd_PressureIsLo;
|
|
|
|
|
|
|
|
return IdReasonOfEnd_NotDetected;
|
|
|
|
}
|
2023-11-22 14:47:17 +05:00
|
|
|
|
2023-10-20 11:03:14 +05:00
|
|
|
protected override double CalcValue(DetectableTelemetry[] telemetry, int begin, int end)
|
|
|
|
=> CalcRop(telemetry, begin, end);
|
|
|
|
|
2023-11-22 14:47:17 +05:00
|
|
|
protected override bool IsValidOperationDetectorResult(OperationDetectorResult operationDetectorResult) =>
|
2023-12-05 16:15:42 +05:00
|
|
|
base.IsValidOperationDetectorResult(operationDetectorResult)
|
|
|
|
&& (operationDetectorResult.Operation.DepthEnd - operationDetectorResult.Operation.DepthStart) > 0.01;
|
2023-11-22 14:47:17 +05:00
|
|
|
|
2023-12-04 17:36:00 +05:00
|
|
|
protected override (int Begin, int End) RefineEdges(DetectableTelemetry[] telemetry, int begin, int end)
|
|
|
|
{
|
|
|
|
var i = end;
|
|
|
|
for (; i > begin + 1; i--)
|
|
|
|
if (telemetry[i].WellDepth - telemetry[i - 1].WellDepth > 0.001d)
|
|
|
|
break;
|
|
|
|
|
|
|
|
return (begin, i);
|
|
|
|
}
|
|
|
|
|
2023-12-05 10:56:49 +05:00
|
|
|
protected override (int IdCategory, IDictionary<string, object> ExtraData) GetSpecificInformation(DetectableTelemetry[] telemetry, int begin, int end)
|
2023-12-04 17:36:00 +05:00
|
|
|
{
|
2023-12-05 10:56:49 +05:00
|
|
|
var (avgRotorSpeed, dispersionOfNormalizedRotorSpeed) = CalcCriteries(telemetry, begin, end);
|
|
|
|
var idCategory = GetIdOperation(avgRotorSpeed, dispersionOfNormalizedRotorSpeed);
|
2023-12-05 14:48:56 +05:00
|
|
|
var extraData = new Dictionary<string, object>
|
|
|
|
{
|
|
|
|
[ExtraDataKeyAvgRotorSpeed] = avgRotorSpeed,
|
|
|
|
[ExtraDataKeyDispersionOfNormalizedRotorSpeed] = dispersionOfNormalizedRotorSpeed,
|
2023-12-18 13:51:40 +05:00
|
|
|
[ExtraDataKeyHasOscillation] = avgRotorSpeed > 1 && dispersionOfNormalizedRotorSpeed > dispersionOfNormalizedRotorSpeedThreshold
|
2023-12-05 14:48:56 +05:00
|
|
|
};
|
2023-12-05 10:56:49 +05:00
|
|
|
return (idCategory, extraData);
|
2023-12-04 17:36:00 +05:00
|
|
|
}
|
|
|
|
|
2023-12-05 10:56:49 +05:00
|
|
|
private static (double avgRotorSpeed, double dispersionOfNormalizedRotorSpeed) CalcCriteries(DetectableTelemetry[] telemetry, int begin, int end)
|
2023-12-04 17:36:00 +05:00
|
|
|
{
|
2023-12-19 00:41:02 +05:00
|
|
|
var telemetryRange = telemetry[begin..end];
|
2023-12-18 13:51:40 +05:00
|
|
|
var avgRotorSpeed = telemetryRange.Average(t => t.RotorSpeed);
|
2023-12-04 17:36:00 +05:00
|
|
|
var dispersion = telemetryRange.Average(t => Math.Pow(t.RotorSpeed / avgRotorSpeed - 1, 2));
|
2023-12-05 10:56:49 +05:00
|
|
|
return (avgRotorSpeed, dispersion);
|
2023-12-04 17:36:00 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
private static int GetIdOperation(double avgRotorSpeed, double dispersionOfNormalizedRotorSpeed)
|
2023-10-20 11:03:14 +05:00
|
|
|
{
|
2023-12-04 17:36:00 +05:00
|
|
|
const int idSlideWithOscillation = WellOperationCategory.IdSlide;
|
|
|
|
|
|
|
|
if (avgRotorSpeed < 5)
|
|
|
|
return WellOperationCategory.IdSlide;
|
|
|
|
|
2023-12-16 12:31:42 +05:00
|
|
|
if(dispersionOfNormalizedRotorSpeed < dispersionOfNormalizedRotorSpeedThreshold)
|
2023-12-04 17:36:00 +05:00
|
|
|
return WellOperationCategory.IdRotor;
|
2023-12-16 12:31:42 +05:00
|
|
|
|
|
|
|
return idSlideWithOscillation;
|
|
|
|
}
|
2023-12-05 10:56:49 +05:00
|
|
|
|
2023-10-20 11:03:14 +05:00
|
|
|
}
|