mirror of
https://github.com/OPM/ResInsight.git
synced 2025-01-01 03:37:15 -06:00
ddefd39fc2
Add access to all MSW parameters in Python : "msw_settings = well_path.msw_settings()" Make all parameters of the well completion object available in Python.
284 lines
12 KiB
C++
284 lines
12 KiB
C++
/////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (C) 2021- 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.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "RimcWellPath.h"
|
|
|
|
#include "FractureCommands/RicPlaceThermalFractureUsingTemplateDataFeature.h"
|
|
#include "RiaLogging.h"
|
|
|
|
#include "RimEclipseCase.h"
|
|
#include "RimEclipseCaseTools.h"
|
|
#include "RimMswCompletionParameters.h"
|
|
#include "RimPerforationCollection.h"
|
|
#include "RimPerforationInterval.h"
|
|
#include "RimStimPlanFractureTemplate.h"
|
|
#include "RimStimPlanModel.h"
|
|
#include "RimThermalFractureTemplate.h"
|
|
#include "RimTools.h"
|
|
#include "RimWellPath.h"
|
|
#include "RimWellPathCollection.h"
|
|
#include "RimWellPathCompletionSettings.h"
|
|
#include "RimWellPathFracture.h"
|
|
|
|
#include "RigStimPlanModelTools.h"
|
|
|
|
#include "FractureCommands/RicNewWellPathFractureFeature.h"
|
|
|
|
#include "cafPdmAbstractFieldScriptingCapability.h"
|
|
#include "cafPdmFieldScriptingCapability.h"
|
|
|
|
CAF_PDM_OBJECT_METHOD_SOURCE_INIT( RimWellPath, RimcWellPath_addFracture, "AddFracture" );
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimcWellPath_addFracture::RimcWellPath_addFracture( caf::PdmObjectHandle* self )
|
|
: caf::PdmObjectMethod( self )
|
|
{
|
|
CAF_PDM_InitObject( "Add StimPlan Fracture", "", "", "Add StimPlan Fracture" );
|
|
|
|
CAF_PDM_InitScriptableField( &m_md, "MeasuredDepth", 0.0, "Measured Depth" );
|
|
CAF_PDM_InitScriptableFieldNoDefault( &m_stimPlanFractureTemplate, "StimPlanFractureTemplate", "", "", "", "StimPlan Fracture Template" );
|
|
CAF_PDM_InitScriptableField( &m_alignDip, "AlignDip", false, "Align Dip" );
|
|
CAF_PDM_InitScriptableFieldNoDefault( &m_eclipseCase, "EclipseCase", "", "", "", "Eclipse Case" );
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
caf::PdmObjectHandle* RimcWellPath_addFracture::execute()
|
|
{
|
|
auto wellPath = self<RimWellPath>();
|
|
|
|
RimWellPathFracture* wellPathFracture = RicNewWellPathFractureFeature::addFracture( wellPath, m_md() );
|
|
|
|
if ( m_stimPlanFractureTemplate ) wellPathFracture->setFractureTemplate( m_stimPlanFractureTemplate() );
|
|
|
|
if ( m_alignDip )
|
|
{
|
|
if ( m_eclipseCase && m_eclipseCase->eclipseCaseData() )
|
|
{
|
|
RiaLogging::info( "Computing formation dip for fracture alignment" );
|
|
|
|
double boundingBoxHorizontal = 50.0;
|
|
double boundingBoxVertical = 100.0;
|
|
|
|
cvf::Vec3d position = wellPathFracture->anchorPosition();
|
|
cvf::Vec3d direction = RigStimPlanModelTools::calculateTSTDirection( m_eclipseCase->eclipseCaseData(),
|
|
position,
|
|
boundingBoxHorizontal,
|
|
boundingBoxVertical );
|
|
RiaLogging::info( QString( "Direction: %1 %2 %3" ).arg( direction.x() ).arg( direction.y() ).arg( direction.z() ) );
|
|
cvf::Vec3d fractureDirectionNormal = wellPathFracture->computeFractureDirectionNormal();
|
|
|
|
cvf::Vec3d formationDirection = RimStimPlanModel::projectVectorIntoFracturePlane( position, fractureDirectionNormal, direction );
|
|
if ( !formationDirection.isUndefined() )
|
|
{
|
|
double formationDip = RigStimPlanModelTools::calculateFormationDip( formationDirection ) - 90.0;
|
|
RiaLogging::info( QString( "Computed formation dip: %1" ).arg( formationDip ) );
|
|
|
|
wellPathFracture->setDip( formationDip );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
RiaLogging::error( "No eclipse case found. Fracture not aligned with formation dip." );
|
|
}
|
|
}
|
|
|
|
return wellPathFracture;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RimcWellPath_addFracture::resultIsPersistent() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
std::unique_ptr<caf::PdmObjectHandle> RimcWellPath_addFracture::defaultResult() const
|
|
{
|
|
return std::unique_ptr<caf::PdmObjectHandle>( new RimWellPathFracture );
|
|
}
|
|
|
|
CAF_PDM_OBJECT_METHOD_SOURCE_INIT( RimWellPath, RimcWellPath_addThermalFracture, "AddThermalFracture" );
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimcWellPath_addThermalFracture::RimcWellPath_addThermalFracture( caf::PdmObjectHandle* self )
|
|
: caf::PdmObjectMethod( self )
|
|
{
|
|
CAF_PDM_InitObject( "Add Thermal Fracture", "", "", "Add Thermal Fracture" );
|
|
|
|
CAF_PDM_InitScriptableField( &m_md, "MeasuredDepth", 0.0, "Measured Depth" );
|
|
CAF_PDM_InitScriptableFieldNoDefault( &m_fractureTemplate, "FractureTemplate", "", "", "", "Thermal Fracture Template" );
|
|
CAF_PDM_InitScriptableField( &m_placeUsingTemplateData, "PlaceUsingTemplateData", true, "Place using template data" );
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
caf::PdmObjectHandle* RimcWellPath_addThermalFracture::execute()
|
|
{
|
|
auto wellPath = self<RimWellPath>();
|
|
|
|
RimWellPathFracture* wellPathFracture = RicNewWellPathFractureFeature::addFracture( wellPath, m_md() );
|
|
|
|
if ( m_fractureTemplate )
|
|
{
|
|
wellPathFracture->setFractureTemplate( m_fractureTemplate() );
|
|
}
|
|
|
|
if ( m_placeUsingTemplateData )
|
|
{
|
|
RicPlaceThermalFractureUsingTemplateDataFeature::placeUsingTemplateData( wellPathFracture );
|
|
}
|
|
|
|
return wellPathFracture;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RimcWellPath_addThermalFracture::resultIsPersistent() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
std::unique_ptr<caf::PdmObjectHandle> RimcWellPath_addThermalFracture::defaultResult() const
|
|
{
|
|
return std::unique_ptr<caf::PdmObjectHandle>( new RimWellPathFracture );
|
|
}
|
|
|
|
CAF_PDM_OBJECT_METHOD_SOURCE_INIT( RimWellPath, RimcWellPath_appendPerforationInterval, "AppendPerforationInterval" );
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimcWellPath_appendPerforationInterval::RimcWellPath_appendPerforationInterval( caf::PdmObjectHandle* self )
|
|
: caf::PdmObjectMethod( self )
|
|
{
|
|
CAF_PDM_InitObject( "Append Perforation Interval", "", "", "Append Perforation Interval" );
|
|
CAF_PDM_InitScriptableField( &m_startMD, "StartMd", 0.0, "", "", "", "Start Measured Depth" );
|
|
CAF_PDM_InitScriptableField( &m_endMD, "EndMd", 0.0, "", "", "", "End Measured Depth" );
|
|
CAF_PDM_InitScriptableField( &m_diameter, "Diameter", 0.0, "", "", "", "Diameter" );
|
|
CAF_PDM_InitScriptableField( &m_skinFactor, "SkinFactor", 0.0, "", "", "", "Skin Factor" );
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
caf::PdmObjectHandle* RimcWellPath_appendPerforationInterval::execute()
|
|
{
|
|
auto wellPath = self<RimWellPath>();
|
|
|
|
auto perforationInterval = new RimPerforationInterval;
|
|
perforationInterval->setStartAndEndMD( m_startMD, m_endMD );
|
|
perforationInterval->setSkinFactor( m_skinFactor );
|
|
perforationInterval->setDiameter( m_diameter );
|
|
|
|
wellPath->perforationIntervalCollection()->appendPerforation( perforationInterval );
|
|
|
|
auto* wellPathCollection = RimTools::wellPathCollection();
|
|
|
|
wellPathCollection->uiCapability()->updateConnectedEditors();
|
|
wellPathCollection->scheduleRedrawAffectedViews();
|
|
|
|
return perforationInterval;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RimcWellPath_appendPerforationInterval::resultIsPersistent() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
std::unique_ptr<caf::PdmObjectHandle> RimcWellPath_appendPerforationInterval::defaultResult() const
|
|
{
|
|
return std::unique_ptr<caf::PdmObjectHandle>( new RimPerforationInterval );
|
|
}
|
|
|
|
CAF_PDM_OBJECT_METHOD_SOURCE_INIT( RimWellPath, RimcWellPath_multiSegmentWellSettings, "MswSettings" );
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimcWellPath_multiSegmentWellSettings::RimcWellPath_multiSegmentWellSettings( caf::PdmObjectHandle* self )
|
|
: caf::PdmObjectMethod( self )
|
|
{
|
|
CAF_PDM_InitObject( "MSW Settings", "", "", "Multi Segment Well Settings" );
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
caf::PdmObjectHandle* RimcWellPath_multiSegmentWellSettings::execute()
|
|
{
|
|
auto wellPath = self<RimWellPath>();
|
|
|
|
// RimMswCompletionParameters is a child object of RimWellPathCompletionSettings. To simplify the Python API, we return
|
|
// RimMswCompletionParameters directly from a well path object in Python. Two parameters are already exposed as part of the completion
|
|
// settings object, see RimWellPathCompletionSettings and the proxy fields liner_diameter and roughness. These fields are kept to
|
|
// ensure backward compatibility with existing scripts.
|
|
//
|
|
// https://github.com/OPM/ResInsight/issues/11901
|
|
|
|
if ( auto completionSettings = wellPath->completionSettings() )
|
|
{
|
|
return completionSettings->mswCompletionParameters();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RimcWellPath_multiSegmentWellSettings::resultIsPersistent() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RimcWellPath_multiSegmentWellSettings::isNullptrValidResult() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
std::unique_ptr<caf::PdmObjectHandle> RimcWellPath_multiSegmentWellSettings::defaultResult() const
|
|
{
|
|
return std::unique_ptr<caf::PdmObjectHandle>( new RimMswCompletionParameters );
|
|
}
|