2020-10-16 04:27:46 -05:00
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Copyright (C) 2020- Equinor ASA
|
|
|
|
//
|
|
|
|
// ResInsight is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// ResInsight is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
//
|
|
|
|
// See the GNU General Public License at <http://www.gnu.org/licenses/gpl.html>
|
|
|
|
// for more details.
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
2020-11-04 06:46:17 -06:00
|
|
|
#include "RimStimPlanModelStressCalculator.h"
|
2020-10-16 04:27:46 -05:00
|
|
|
|
|
|
|
#include "RiaDefines.h"
|
2021-01-21 05:58:46 -06:00
|
|
|
#include "RiaEclipseUnitTools.h"
|
2021-03-23 08:56:02 -05:00
|
|
|
#include "RiaInterpolationTools.h"
|
2020-10-28 07:33:56 -05:00
|
|
|
#include "RiaLogging.h"
|
2020-11-04 06:46:17 -06:00
|
|
|
#include "RiaStimPlanModelDefines.h"
|
2020-10-16 04:27:46 -05:00
|
|
|
|
|
|
|
#include "RigEclipseCaseData.h"
|
|
|
|
#include "RigEclipseWellLogExtractor.h"
|
|
|
|
#include "RigWellPath.h"
|
|
|
|
|
|
|
|
#include "RigWellPathGeometryTools.h"
|
|
|
|
#include "RimCase.h"
|
|
|
|
#include "RimEclipseCase.h"
|
|
|
|
#include "RimModeledWellPath.h"
|
2020-11-04 06:46:17 -06:00
|
|
|
#include "RimStimPlanModel.h"
|
|
|
|
#include "RimStimPlanModelCalculator.h"
|
|
|
|
#include "RimStimPlanModelStressCalculator.h"
|
2020-10-16 04:27:46 -05:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-11-04 06:46:17 -06:00
|
|
|
RimStimPlanModelStressCalculator::RimStimPlanModelStressCalculator( RimStimPlanModelCalculator* stimPlanModelCalculator )
|
|
|
|
: m_stimPlanModelCalculator( stimPlanModelCalculator )
|
2020-10-16 04:27:46 -05:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-11-04 06:46:17 -06:00
|
|
|
bool RimStimPlanModelStressCalculator::isMatching( RiaDefines::CurveProperty curveProperty ) const
|
2020-10-16 04:27:46 -05:00
|
|
|
{
|
2020-11-06 03:46:38 -06:00
|
|
|
std::vector<RiaDefines::CurveProperty> matching = { RiaDefines::CurveProperty::INITIAL_STRESS,
|
|
|
|
RiaDefines::CurveProperty::STRESS,
|
|
|
|
RiaDefines::CurveProperty::STRESS_GRADIENT,
|
|
|
|
RiaDefines::CurveProperty::TEMPERATURE };
|
2020-10-16 04:27:46 -05:00
|
|
|
|
|
|
|
return std::find( matching.begin(), matching.end(), curveProperty ) != matching.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-11-04 06:46:17 -06:00
|
|
|
bool RimStimPlanModelStressCalculator::calculate( RiaDefines::CurveProperty curveProperty,
|
|
|
|
const RimStimPlanModel* stimPlanModel,
|
2020-10-16 04:27:46 -05:00
|
|
|
int timeStep,
|
|
|
|
std::vector<double>& values,
|
|
|
|
std::vector<double>& measuredDepthValues,
|
|
|
|
std::vector<double>& tvDepthValues,
|
|
|
|
double& rkbDiff ) const
|
|
|
|
{
|
2021-02-05 08:25:52 -06:00
|
|
|
RimEclipseCase* eclipseCase = stimPlanModel->eclipseCaseForProperty( curveProperty );
|
2020-10-16 04:27:46 -05:00
|
|
|
if ( !eclipseCase )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-11-04 06:46:17 -06:00
|
|
|
if ( !stimPlanModel->thicknessDirectionWellPath() )
|
2020-10-28 07:33:56 -05:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-11-04 06:46:17 -06:00
|
|
|
RigWellPath* wellPathGeometry = stimPlanModel->thicknessDirectionWellPath()->wellPathGeometry();
|
2020-10-28 07:33:56 -05:00
|
|
|
if ( !wellPathGeometry )
|
|
|
|
{
|
|
|
|
RiaLogging::error( "No well path geometry found for stress data exctration." );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-11-04 06:46:17 -06:00
|
|
|
std::vector<double> tvDepthInFeet = m_stimPlanModelCalculator->calculateTrueVerticalDepth();
|
2020-10-16 04:27:46 -05:00
|
|
|
for ( double f : tvDepthInFeet )
|
|
|
|
{
|
|
|
|
tvDepthValues.push_back( RiaEclipseUnitTools::feetToMeter( f ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( curveProperty == RiaDefines::CurveProperty::STRESS )
|
|
|
|
{
|
2020-11-04 06:46:17 -06:00
|
|
|
values = m_stimPlanModelCalculator->calculateStress();
|
|
|
|
std::vector<double> stressGradients = m_stimPlanModelCalculator->calculateStressGradient();
|
2020-10-16 04:27:46 -05:00
|
|
|
addDatapointsForBottomOfLayers( tvDepthValues, values, stressGradients );
|
|
|
|
}
|
|
|
|
else if ( curveProperty == RiaDefines::CurveProperty::INITIAL_STRESS )
|
|
|
|
{
|
2020-11-04 06:46:17 -06:00
|
|
|
values = m_stimPlanModelCalculator->calculateInitialStress();
|
|
|
|
std::vector<double> stressGradients = m_stimPlanModelCalculator->calculateStressGradient();
|
2020-10-16 04:27:46 -05:00
|
|
|
addDatapointsForBottomOfLayers( tvDepthValues, values, stressGradients );
|
|
|
|
}
|
|
|
|
else if ( curveProperty == RiaDefines::CurveProperty::STRESS_GRADIENT )
|
|
|
|
{
|
2020-11-04 06:46:17 -06:00
|
|
|
values = m_stimPlanModelCalculator->calculateStressGradient();
|
2021-03-23 08:56:02 -05:00
|
|
|
addDatapointsForBottomOfLayers( tvDepthValues, values );
|
2020-10-16 04:27:46 -05:00
|
|
|
}
|
|
|
|
else if ( curveProperty == RiaDefines::CurveProperty::TEMPERATURE )
|
|
|
|
{
|
2020-11-04 06:46:17 -06:00
|
|
|
m_stimPlanModelCalculator->calculateTemperature( values );
|
2021-03-23 08:56:02 -05:00
|
|
|
addDatapointsForBottomOfLayers( tvDepthValues, values );
|
2020-10-16 04:27:46 -05:00
|
|
|
}
|
|
|
|
|
2022-10-19 04:22:49 -05:00
|
|
|
if ( eclipseCase && eclipseCase->eclipseCaseData() )
|
2020-10-16 04:27:46 -05:00
|
|
|
{
|
|
|
|
RigEclipseWellLogExtractor eclExtractor( eclipseCase->eclipseCaseData(), wellPathGeometry, "fracture model" );
|
|
|
|
|
2020-10-23 02:56:49 -05:00
|
|
|
rkbDiff = wellPathGeometry->rkbDiff();
|
2020-10-16 04:27:46 -05:00
|
|
|
|
|
|
|
// Generate MD data by interpolation
|
2020-10-23 02:56:49 -05:00
|
|
|
const std::vector<double>& mdValuesOfWellPath = wellPathGeometry->measuredDepths();
|
2021-03-23 08:56:02 -05:00
|
|
|
const std::vector<double>& tvdValuesOfWellPath = wellPathGeometry->trueVerticalDepths();
|
2020-11-02 12:13:56 -06:00
|
|
|
if ( mdValuesOfWellPath.empty() )
|
|
|
|
{
|
|
|
|
RiaLogging::error( "Well path geometry had no MD values." );
|
|
|
|
return false;
|
|
|
|
}
|
2020-10-16 04:27:46 -05:00
|
|
|
|
2021-03-23 08:56:02 -05:00
|
|
|
// The thickness direction "fake" well path is always a straight line:
|
|
|
|
// measured depth can be interpolated linearly
|
|
|
|
measuredDepthValues.clear();
|
|
|
|
for ( double tvd : tvDepthValues )
|
|
|
|
{
|
|
|
|
double md = RiaInterpolationTools::linear( tvdValuesOfWellPath, mdValuesOfWellPath, tvd );
|
|
|
|
measuredDepthValues.push_back( md );
|
|
|
|
}
|
2020-10-16 04:27:46 -05:00
|
|
|
CVF_ASSERT( measuredDepthValues.size() == tvDepthValues.size() );
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-11-04 06:46:17 -06:00
|
|
|
void RimStimPlanModelStressCalculator::addDatapointsForBottomOfLayers( std::vector<double>& tvDepthValues,
|
2020-10-16 04:27:46 -05:00
|
|
|
std::vector<double>& stress,
|
|
|
|
const std::vector<double>& stressGradients )
|
|
|
|
{
|
|
|
|
std::vector<double> tvdWithBottomLayers;
|
|
|
|
std::vector<double> valuesWithBottomLayers;
|
|
|
|
for ( size_t i = 0; i < stress.size(); i++ )
|
|
|
|
{
|
|
|
|
// Add the data point at top of the layer
|
|
|
|
double topLayerDepth = tvDepthValues[i];
|
|
|
|
double stressValue = stress[i];
|
|
|
|
tvdWithBottomLayers.push_back( topLayerDepth );
|
|
|
|
valuesWithBottomLayers.push_back( stressValue );
|
|
|
|
|
|
|
|
// Add extra data points for bottom part of the layer
|
|
|
|
if ( i < stress.size() - 1 )
|
|
|
|
{
|
|
|
|
double bottomLayerDepth = tvDepthValues[i + 1];
|
|
|
|
double diffDepthFeet = RiaEclipseUnitTools::meterToFeet( bottomLayerDepth - topLayerDepth );
|
|
|
|
double bottomStress = stressValue + diffDepthFeet * stressGradients[i];
|
|
|
|
|
|
|
|
tvdWithBottomLayers.push_back( bottomLayerDepth );
|
|
|
|
valuesWithBottomLayers.push_back( bottomStress );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stress = valuesWithBottomLayers;
|
|
|
|
tvDepthValues = tvdWithBottomLayers;
|
|
|
|
}
|
2021-03-23 08:56:02 -05:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2023-02-26 03:48:40 -06:00
|
|
|
void RimStimPlanModelStressCalculator::addDatapointsForBottomOfLayers( std::vector<double>& tvDepthValues, std::vector<double>& values )
|
2021-03-23 08:56:02 -05:00
|
|
|
|
|
|
|
{
|
|
|
|
std::vector<double> tvdWithBottomLayers;
|
|
|
|
std::vector<double> valuesWithBottomLayers;
|
|
|
|
for ( size_t i = 0; i < values.size(); i++ )
|
|
|
|
{
|
|
|
|
// Add the data point at top of the layer
|
|
|
|
double topLayerDepth = tvDepthValues[i];
|
|
|
|
double value = values[i];
|
|
|
|
tvdWithBottomLayers.push_back( topLayerDepth );
|
|
|
|
valuesWithBottomLayers.push_back( value );
|
|
|
|
|
|
|
|
// Add extra data points for bottom part of the layer
|
|
|
|
if ( i < values.size() - 1 )
|
|
|
|
{
|
|
|
|
double bottomLayerDepth = tvDepthValues[i + 1];
|
|
|
|
double bottomValue = value;
|
|
|
|
|
|
|
|
tvdWithBottomLayers.push_back( bottomLayerDepth );
|
|
|
|
valuesWithBottomLayers.push_back( bottomValue );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
values = valuesWithBottomLayers;
|
|
|
|
tvDepthValues = tvdWithBottomLayers;
|
|
|
|
}
|