using AsbCloudDb.Model; using System; using System.Collections.Generic; using System.Linq; namespace AsbCloudInfrastructure.Services.DetectOperations.Detectors { #nullable enable internal abstract class DetectorAbstract { private readonly int idOperation; private readonly int stepLength = 3; protected DetectorAbstract(int idOperation) { this.idOperation = idOperation; } public bool TryDetect(int idTelemetry, DetectableTelemetry[] telemetry, int begin, int end, DetectedOperation? previousOperation, out OperationDetectorResult? result) { // Проверка соответствия критерию начала операции if (DetectBegin(telemetry, begin, previousOperation)) { // Поиск окончания соответствия критерию var positionEnd = begin; while (positionEnd < end) { positionEnd += stepLength; if ((positionEnd > end)) break; if (DetectEnd(telemetry, positionEnd, previousOperation)) { result = MakeOperation(idTelemetry, telemetry, begin, positionEnd); return true; } } } result = null; return false; } protected abstract bool DetectBegin(DetectableTelemetry[] telemetry, int position, DetectedOperation? previousOperation); protected virtual bool DetectEnd(DetectableTelemetry[] telemetry, int position, DetectedOperation? previousOperation) => !DetectBegin(telemetry, position, previousOperation); private OperationDetectorResult MakeOperation(int idTelemetry, DetectableTelemetry[] telemetry, int begin, int end) { var pBegin = telemetry[begin]; var pEnd = telemetry[end]; var result = new OperationDetectorResult { TelemetryBegin = begin, TelemetryEnd = end, Operation = new DetectedOperation { IdTelemetry = idTelemetry, IdCategory = idOperation, IdUsersAtStart = pBegin.IdUser ?? -1, DateStart = pBegin.DateTime, DateEnd = pEnd.DateTime, DepthStart = (double)pBegin.WellDepth, DepthEnd = (double)pEnd.WellDepth, Value = CalcValue(telemetry, begin, end), }, }; return result; } protected abstract bool IsValid(DetectableTelemetry[] telemetry, int begin, int end); protected abstract double CalcValue(DetectableTelemetry[] telemetry, int begin, int end); public static InterpolationLine MakeInterpolationLine( Func yGetter, DetectableTelemetry[] telemetry, int begin, int fragmentLength) { DetectableTelemetry[] data; if (fragmentLength > 0 && (begin + fragmentLength) < telemetry.Length) data = telemetry[begin..(begin + fragmentLength)]; else data = telemetry[begin..]; var line = new InterpolationLine(data.Select(d => (yGetter(d), (d.DateTime - telemetry[0].DateTime).TotalHours))); return line; } /// /// расчет продолжительности операции /// /// /// /// /// protected static double CalcDeltaTime(DetectableTelemetry[] telemetry, int begin, int end) { var pBegin = telemetry[begin]; var pEnd = telemetry[end]; var result = (pEnd.DateTime - pBegin.DateTime).TotalHours; return result; } /// /// часто используемый предикат для определения отсутствия изменения глубины ствола скважины /// /// /// /// /// protected static bool IsValidByWellDepthDoesNotChange(DetectableTelemetry[] telemetry, int begin, int end) { var pBegin = telemetry[begin]; var pEnd = telemetry[end]; if (Math.Abs((double)(pBegin.WellDepth - pEnd.WellDepth)) > 0.01) return false; return true; } protected static bool IsValidByWellDepthIncreasing(DetectableTelemetry[] telemetry, int begin, int end) { var pBegin = telemetry[begin]; var pEnd = telemetry[end]; if (pBegin.WellDepth >= pEnd.WellDepth) return false; return true; } protected static double CalcRop(DetectableTelemetry[] telemetry, int begin, int end) { var pBegin = telemetry[begin]; var pEnd = telemetry[end]; var result = (double)(pEnd.WellDepth - pBegin.WellDepth) / (pEnd.DateTime - pBegin.DateTime).TotalHours; return result; } /// /// Расчет статистики по массиву данных за интервал /// /// /// /// /// /// protected static (double min, double max, double sum, int count) CalcStat( DetectableTelemetry[] telemetry, Func getter, int begin, int count) { var sum = 0d; var min = double.MaxValue; var max = double.MinValue; var end = begin + count; end = end < telemetry.Length ? end : telemetry.Length; for (var i = begin; i < end; i++) { var item = telemetry[i]; var itemValue = getter(item); if (min > itemValue) min = itemValue; if (max < itemValue) max = itemValue; sum += itemValue; } return (min, max, sum, end - begin); } /// /// Максимальное отклонение от среднего за интервал /// /// /// /// /// /// protected static double CalcMaxDeviation( DetectableTelemetry[] telemetry, Func getter, int begin, int count) { var stat = CalcStat(telemetry, getter, begin, count); var avg = stat.sum / stat.count; var dev1 = avg - stat.min; var dev2 = stat.max - avg; var dev = dev1 > dev2 ? dev1 : dev2; return dev; } /// /// Определяет наличие разброса значений в интервале большего указанного значения. /// /// /// /// /// /// /// protected static bool ContainsDeviation( DetectableTelemetry[] telemetry, Func getter, int begin, int count, double deviation) { var min = double.MaxValue; var max = double.MinValue; var end = begin + count; end = end < telemetry.Length ? end : telemetry.Length; for (var i = begin; i < end; i ++) { var item = telemetry[i]; var itemValue = getter(item); if (min > itemValue) min = itemValue; if (max < itemValue) max = itemValue; if(max - min > deviation) return true; } return false; } /// /// Определяет наличие разброса значений в интервале большего указанного значения. По нескольким значениям из интервала. /// /// /// /// /// /// /// protected static bool ContainsDeviationApprox( DetectableTelemetry[] telemetry, Func getter, int begin, int count, double deviation) { var min = double.MaxValue; var max = double.MinValue; var end = begin + count; end = end < telemetry.Length ? end : telemetry.Length; var step = count > 15 ? count / 5 : count > 3 ? 3: 1; for (var i = begin; i < end; i+= step) { var item = telemetry[i]; var itemValue = getter(item); if (min > itemValue) min = itemValue; if (max < itemValue) max = itemValue; if (max - min > deviation) return true; } return false; } } #nullable disable }