2019-05-20 06:21:02 -05:00
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// Copyright (C) 2019- 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.
|
|
|
|
//
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
2021-10-24 03:21:17 -05:00
|
|
|
|
2019-05-20 06:21:02 -05:00
|
|
|
#include "RiaGrpcServiceInterface.h"
|
2021-10-24 03:21:17 -05:00
|
|
|
#include "RiaLogging.h"
|
2019-05-20 06:21:02 -05:00
|
|
|
|
|
|
|
#include "RimCase.h"
|
2019-09-06 03:40:57 -05:00
|
|
|
#include "RimProject.h"
|
2019-05-20 06:21:02 -05:00
|
|
|
|
2020-08-21 04:13:07 -05:00
|
|
|
#include "cafPdmAbstractFieldScriptingCapability.h"
|
2020-01-17 03:35:42 -06:00
|
|
|
#include "cafPdmChildArrayField.h"
|
|
|
|
#include "cafPdmChildField.h"
|
2019-06-10 13:42:20 -05:00
|
|
|
#include "cafPdmDataValueField.h"
|
|
|
|
#include "cafPdmObject.h"
|
2020-08-21 04:13:07 -05:00
|
|
|
#include "cafPdmObjectScriptingCapability.h"
|
|
|
|
#include "cafPdmObjectScriptingCapabilityRegister.h"
|
2020-02-28 07:38:24 -06:00
|
|
|
#include "cafPdmProxyValueField.h"
|
2020-03-04 03:12:59 -06:00
|
|
|
#include "cafPdmScriptIOMessages.h"
|
2019-06-10 13:42:20 -05:00
|
|
|
#include "cafPdmXmlFieldHandle.h"
|
|
|
|
|
2021-10-24 03:21:17 -05:00
|
|
|
#include <PdmObject.pb.h>
|
2019-05-20 06:21:02 -05:00
|
|
|
#include <grpcpp/grpcpp.h>
|
|
|
|
|
2021-08-23 04:58:32 -05:00
|
|
|
#include <QDebug>
|
2019-06-10 13:42:20 -05:00
|
|
|
#include <QXmlStreamReader>
|
2021-08-23 04:58:32 -05:00
|
|
|
|
2019-05-20 06:21:02 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
RimCase* RiaGrpcServiceInterface::findCase( int caseId )
|
2019-05-20 06:21:02 -05:00
|
|
|
{
|
|
|
|
std::vector<RimCase*> cases;
|
2020-05-12 02:50:38 -05:00
|
|
|
RimProject::current()->allCases( cases );
|
2019-05-20 06:21:02 -05:00
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( RimCase* rimCase : cases )
|
2019-05-20 06:21:02 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( caseId == rimCase->caseId() )
|
2019-05-20 06:21:02 -05:00
|
|
|
{
|
|
|
|
return rimCase;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-02-21 01:57:01 -06:00
|
|
|
/// Find the number of data items that will fit in the given bytes.
|
|
|
|
/// The default argument for numBytesWantedInPackage is meant to be a sensible size for GRPC.
|
2019-05-20 06:21:02 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-02-21 01:57:01 -06:00
|
|
|
size_t RiaGrpcServiceInterface::numberOfDataUnitsInPackage( size_t dataUnitSize, size_t packageByteCount /*= 64 * 1024u*/ )
|
2019-05-20 06:21:02 -05:00
|
|
|
{
|
2020-02-21 01:57:01 -06:00
|
|
|
size_t dataUnitCount = packageByteCount / dataUnitSize;
|
|
|
|
return dataUnitCount;
|
2019-05-20 06:21:02 -05:00
|
|
|
}
|
|
|
|
|
2019-06-10 13:42:20 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-02-12 04:43:15 -06:00
|
|
|
void RiaGrpcServiceInterface::copyPdmObjectFromCafToRips( const caf::PdmObjectHandle* source, rips::PdmObject* destination )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2020-01-17 03:35:42 -06:00
|
|
|
CAF_ASSERT( source && destination && source->xmlCapability() );
|
2019-06-10 13:42:20 -05:00
|
|
|
|
2020-02-24 08:42:23 -06:00
|
|
|
QString classKeyword = source->xmlCapability()->classKeyword();
|
2021-09-02 01:55:35 -05:00
|
|
|
|
|
|
|
QString scriptClassName;
|
|
|
|
|
|
|
|
// Find first scriptable object in inheritance stack
|
|
|
|
{
|
|
|
|
auto pdmObject = dynamic_cast<const caf::PdmObject*>( source );
|
|
|
|
auto classKeywordStack = pdmObject->classInheritanceStack();
|
|
|
|
|
|
|
|
// Reverse to get leaf node first
|
|
|
|
classKeywordStack.reverse();
|
|
|
|
|
|
|
|
for ( const auto& candidateClassKeyword : classKeywordStack )
|
|
|
|
{
|
|
|
|
if ( caf::PdmObjectScriptingCapabilityRegister::isScriptable( candidateClassKeyword ) )
|
|
|
|
{
|
|
|
|
scriptClassName =
|
|
|
|
caf::PdmObjectScriptingCapabilityRegister::scriptClassNameFromClassKeyword( candidateClassKeyword );
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fallback to source object class name
|
|
|
|
if ( scriptClassName.isEmpty() ) scriptClassName = classKeyword;
|
|
|
|
}
|
|
|
|
|
|
|
|
destination->set_class_keyword( scriptClassName.toStdString() );
|
2019-09-06 03:40:57 -05:00
|
|
|
destination->set_address( reinterpret_cast<uint64_t>( source ) );
|
2020-01-28 08:48:50 -06:00
|
|
|
|
|
|
|
bool visible = true;
|
|
|
|
if ( source->uiCapability() && source->uiCapability()->objectToggleField() )
|
|
|
|
{
|
2021-09-02 01:55:35 -05:00
|
|
|
const auto* boolField = dynamic_cast<const caf::PdmField<bool>*>( source->uiCapability()->objectToggleField() );
|
2020-01-28 08:48:50 -06:00
|
|
|
if ( boolField )
|
|
|
|
{
|
|
|
|
visible = boolField->value();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
destination->set_visible( visible );
|
|
|
|
|
2019-06-10 13:42:20 -05:00
|
|
|
std::vector<caf::PdmFieldHandle*> fields;
|
2019-09-06 03:40:57 -05:00
|
|
|
source->fields( fields );
|
2019-06-10 13:42:20 -05:00
|
|
|
|
|
|
|
auto parametersMap = destination->mutable_parameters();
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( auto field : fields )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
auto pdmValueField = dynamic_cast<const caf::PdmValueField*>( field );
|
|
|
|
if ( pdmValueField )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
QString keyword = pdmValueField->keyword();
|
2020-08-21 04:13:07 -05:00
|
|
|
auto ricfHandle = field->template capability<caf::PdmAbstractFieldScriptingCapability>();
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( ricfHandle != nullptr )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2020-02-28 07:38:24 -06:00
|
|
|
auto pdmProxyField = dynamic_cast<const caf::PdmProxyFieldHandle*>( field );
|
|
|
|
if ( !( pdmProxyField && pdmProxyField->isStreamingField() ) )
|
|
|
|
{
|
|
|
|
QString text;
|
|
|
|
QTextStream outStream( &text );
|
2020-03-04 04:30:08 -06:00
|
|
|
ricfHandle->readFromField( outStream, false );
|
2020-03-03 03:43:07 -06:00
|
|
|
( *parametersMap )[ricfHandle->scriptFieldName().toStdString()] = text.toStdString();
|
2020-02-28 07:38:24 -06:00
|
|
|
}
|
2019-09-06 03:40:57 -05:00
|
|
|
}
|
2019-06-10 13:42:20 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2020-02-12 04:43:15 -06:00
|
|
|
void RiaGrpcServiceInterface::copyPdmObjectFromRipsToCaf( const rips::PdmObject* source, caf::PdmObjectHandle* destination )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2020-01-17 03:35:42 -06:00
|
|
|
CAF_ASSERT( source && destination && destination->xmlCapability() );
|
2020-01-28 08:48:50 -06:00
|
|
|
|
|
|
|
if ( destination->uiCapability() && destination->uiCapability()->objectToggleField() )
|
|
|
|
{
|
2021-09-02 01:55:35 -05:00
|
|
|
auto* boolField = dynamic_cast<caf::PdmField<bool>*>( destination->uiCapability()->objectToggleField() );
|
2020-01-28 08:48:50 -06:00
|
|
|
if ( boolField )
|
|
|
|
{
|
|
|
|
QVariant oldValue = boolField->toQVariant();
|
|
|
|
boolField->setValue( source->visible() );
|
|
|
|
QVariant newValue = boolField->toQVariant();
|
|
|
|
destination->uiCapability()->fieldChangedByUi( boolField, oldValue, newValue );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-10 13:42:20 -05:00
|
|
|
std::vector<caf::PdmFieldHandle*> fields;
|
2019-09-06 03:40:57 -05:00
|
|
|
destination->fields( fields );
|
2019-06-10 13:42:20 -05:00
|
|
|
|
|
|
|
auto parametersMap = source->parameters();
|
2021-08-23 04:58:32 -05:00
|
|
|
|
|
|
|
bool printContent = false; // Flag to control debug output to debugger
|
|
|
|
if ( printContent )
|
|
|
|
{
|
|
|
|
for ( const auto& p : parametersMap )
|
|
|
|
{
|
|
|
|
qDebug() << QString::fromStdString( p.first ) << " : " << QString::fromStdString( p.second );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-24 03:21:17 -05:00
|
|
|
caf::PdmScriptIOMessages messages;
|
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( auto field : fields )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2020-08-21 04:13:07 -05:00
|
|
|
auto scriptability = field->template capability<caf::PdmAbstractFieldScriptingCapability>();
|
2020-03-10 08:11:22 -05:00
|
|
|
if ( scriptability )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2022-03-15 09:32:48 -05:00
|
|
|
bool isPdmPtrArrayField = ( dynamic_cast<caf::PdmPtrArrayFieldHandle*>( field ) &&
|
|
|
|
!dynamic_cast<caf::PdmChildArrayFieldHandle*>( field ) );
|
|
|
|
|
|
|
|
if ( !isPdmPtrArrayField && !dynamic_cast<caf::PdmValueField*>( field ) )
|
2021-06-25 07:18:36 -05:00
|
|
|
{
|
2022-03-15 09:32:48 -05:00
|
|
|
// Update of child objects and child object arrays are not supported
|
|
|
|
// Update of PdmPtrArrayField is supported, used by RimcSummaryPlotCollection_newSummaryPlot
|
2021-06-25 07:18:36 -05:00
|
|
|
// https://github.com/OPM/ResInsight/issues/7794
|
|
|
|
continue;
|
|
|
|
}
|
2020-03-10 08:11:22 -05:00
|
|
|
QString keyword = scriptability->scriptFieldName();
|
|
|
|
QString value = QString::fromStdString( parametersMap[keyword.toStdString()] );
|
2020-02-21 01:57:01 -06:00
|
|
|
|
2020-03-10 08:11:22 -05:00
|
|
|
QVariant oldValue, newValue;
|
2021-10-24 03:21:17 -05:00
|
|
|
messages.currentCommand = "Assign value to field " + keyword;
|
|
|
|
messages.currentArgument = value;
|
|
|
|
if ( assignFieldValue( value, field, &oldValue, &newValue, &messages ) )
|
2020-03-10 08:11:22 -05:00
|
|
|
{
|
|
|
|
destination->uiCapability()->fieldChangedByUi( field, oldValue, newValue );
|
2020-01-21 04:08:09 -06:00
|
|
|
}
|
2019-06-10 13:42:20 -05:00
|
|
|
}
|
|
|
|
}
|
2021-10-24 03:21:17 -05:00
|
|
|
|
|
|
|
for ( const auto message : messages.m_messages )
|
|
|
|
{
|
|
|
|
RiaLogging::error( message.second );
|
|
|
|
}
|
2019-06-10 13:42:20 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2021-10-24 03:21:17 -05:00
|
|
|
bool RiaGrpcServiceInterface::assignFieldValue( const QString& stringValue,
|
|
|
|
caf::PdmFieldHandle* field,
|
|
|
|
QVariant* oldValue,
|
|
|
|
QVariant* newValue,
|
|
|
|
caf::PdmScriptIOMessages* messages )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2020-01-21 04:08:09 -06:00
|
|
|
CAF_ASSERT( oldValue && newValue );
|
|
|
|
|
2020-08-21 04:13:07 -05:00
|
|
|
auto scriptability = field->template capability<caf::PdmAbstractFieldScriptingCapability>();
|
2022-10-21 03:07:16 -05:00
|
|
|
if ( field && scriptability != nullptr && scriptability->isIOWriteable() )
|
2019-06-10 13:42:20 -05:00
|
|
|
{
|
2021-10-24 03:21:17 -05:00
|
|
|
auto* valueField = dynamic_cast<caf::PdmValueField*>( field );
|
|
|
|
QTextStream stream( stringValue.toLatin1() );
|
2020-03-10 08:11:22 -05:00
|
|
|
if ( valueField ) *oldValue = valueField->toQVariant();
|
2021-10-24 03:21:17 -05:00
|
|
|
scriptability->writeToField( stream, nullptr, messages, false, RimProject::current() );
|
2020-03-10 08:11:22 -05:00
|
|
|
if ( valueField ) *newValue = valueField->toQVariant();
|
2020-01-21 04:08:09 -06:00
|
|
|
return true;
|
2019-06-10 13:42:20 -05:00
|
|
|
}
|
2020-01-21 04:08:09 -06:00
|
|
|
return false;
|
2019-06-10 13:42:20 -05:00
|
|
|
}
|
|
|
|
|
2019-07-19 06:54:15 -05:00
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2021-06-25 07:18:36 -05:00
|
|
|
caf::PdmObjectHandle* RiaGrpcServiceInterface::emplaceChildField( caf::PdmObject* parent,
|
|
|
|
const QString& fieldKeyword,
|
|
|
|
const QString& keywordForClassToCreate )
|
2019-07-19 06:54:15 -05:00
|
|
|
{
|
|
|
|
std::vector<caf::PdmFieldHandle*> fields;
|
2019-09-06 03:40:57 -05:00
|
|
|
parent->fields( fields );
|
2019-07-19 06:54:15 -05:00
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( auto field : fields )
|
2019-07-19 06:54:15 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
auto pdmChildArrayField = dynamic_cast<caf::PdmChildArrayFieldHandle*>( field );
|
2020-01-17 03:35:42 -06:00
|
|
|
auto pdmChildField = dynamic_cast<caf::PdmChildFieldHandle*>( field );
|
2021-06-25 07:18:36 -05:00
|
|
|
if ( pdmChildArrayField )
|
2019-07-19 06:54:15 -05:00
|
|
|
{
|
2021-06-25 07:18:36 -05:00
|
|
|
bool isMatching = false;
|
|
|
|
if ( fieldKeyword.isEmpty() )
|
|
|
|
{
|
|
|
|
// Use first child array field if no fieldKeyword is specified
|
|
|
|
isMatching = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
isMatching = ( pdmChildArrayField->keyword() == fieldKeyword );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( isMatching )
|
|
|
|
{
|
|
|
|
auto objectCreated = emplaceChildArrayField( pdmChildArrayField, keywordForClassToCreate );
|
|
|
|
|
|
|
|
// Notify parent object that a new object has been created
|
|
|
|
if ( objectCreated ) parent->onChildAdded( pdmChildArrayField );
|
|
|
|
|
|
|
|
return objectCreated;
|
|
|
|
}
|
2020-01-17 03:35:42 -06:00
|
|
|
}
|
2021-06-25 07:18:36 -05:00
|
|
|
else if ( pdmChildField && pdmChildField->keyword() == fieldKeyword )
|
2020-01-17 03:35:42 -06:00
|
|
|
{
|
2021-06-25 07:18:36 -05:00
|
|
|
return emplaceChildField( pdmChildField, keywordForClassToCreate );
|
2019-07-19 06:54:15 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
2020-01-17 03:35:42 -06:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2021-06-25 07:18:36 -05:00
|
|
|
caf::PdmObjectHandle* RiaGrpcServiceInterface::emplaceChildField( caf::PdmChildFieldHandle* childField,
|
|
|
|
const QString& keywordForClassToCreate )
|
2020-01-17 03:35:42 -06:00
|
|
|
{
|
2021-06-25 07:18:36 -05:00
|
|
|
QString childClassKeyword;
|
|
|
|
if ( keywordForClassToCreate.isEmpty() )
|
|
|
|
{
|
|
|
|
childClassKeyword = childField->xmlCapability()->dataTypeName();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
childClassKeyword = keywordForClassToCreate;
|
|
|
|
}
|
2020-01-17 03:35:42 -06:00
|
|
|
|
|
|
|
auto pdmObjectHandle = caf::PdmDefaultObjectFactory::instance()->create( childClassKeyword );
|
|
|
|
CAF_ASSERT( pdmObjectHandle );
|
2021-06-25 07:18:36 -05:00
|
|
|
|
|
|
|
{
|
|
|
|
auto childDataTypeName = childField->xmlCapability()->dataTypeName();
|
|
|
|
|
|
|
|
auto isInheritanceValid = pdmObjectHandle->xmlCapability()->inheritsClassWithKeyword( childDataTypeName );
|
|
|
|
if ( !isInheritanceValid )
|
|
|
|
{
|
|
|
|
delete pdmObjectHandle;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-17 03:35:42 -06:00
|
|
|
childField->setChildObject( pdmObjectHandle );
|
|
|
|
return pdmObjectHandle;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2021-06-25 07:18:36 -05:00
|
|
|
caf::PdmObjectHandle* RiaGrpcServiceInterface::emplaceChildArrayField( caf::PdmChildArrayFieldHandle* childArrayField,
|
|
|
|
const QString& keywordForClassToCreate )
|
2020-01-17 03:35:42 -06:00
|
|
|
{
|
2021-06-25 07:18:36 -05:00
|
|
|
QString childClassKeyword;
|
|
|
|
if ( keywordForClassToCreate.isEmpty() )
|
|
|
|
{
|
|
|
|
childClassKeyword = childArrayField->xmlCapability()->dataTypeName();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
childClassKeyword = keywordForClassToCreate;
|
|
|
|
}
|
2020-01-17 03:35:42 -06:00
|
|
|
|
|
|
|
auto pdmObjectHandle = caf::PdmDefaultObjectFactory::instance()->create( childClassKeyword );
|
2021-06-25 07:18:36 -05:00
|
|
|
if ( !pdmObjectHandle ) return nullptr;
|
|
|
|
|
|
|
|
{
|
|
|
|
auto childDataTypeName = childArrayField->xmlCapability()->dataTypeName();
|
|
|
|
|
|
|
|
auto isInheritanceValid = pdmObjectHandle->xmlCapability()->inheritsClassWithKeyword( childDataTypeName );
|
|
|
|
if ( !isInheritanceValid )
|
|
|
|
{
|
|
|
|
delete pdmObjectHandle;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
2020-01-17 03:35:42 -06:00
|
|
|
|
|
|
|
childArrayField->insertAt( -1, pdmObjectHandle );
|
2021-06-25 07:18:36 -05:00
|
|
|
|
2020-01-17 03:35:42 -06:00
|
|
|
return pdmObjectHandle;
|
|
|
|
}
|