///////////////////////////////////////////////////////////////////////////////// // // 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 // for more details. // ///////////////////////////////////////////////////////////////////////////////// #include "RifHdf5SummaryExporter.h" #include "RiaFilePathTools.h" #include "RiaLogging.h" #include "RiaPreferencesSummary.h" #include "RiaStdStringTools.h" #include "RifHdf5Exporter.h" #include "RifOpmCommonSummary.h" #include "RifSummaryReaderInterface.h" #ifdef _MSC_VER // Disable warning from external library to make sure treat warnings as error works #pragma warning( disable : 4267 ) #endif #include "opm/common/utility/FileSystem.hpp" #include "opm/common/utility/TimeService.hpp" #include "opm/io/eclipse/ESmry.hpp" #include #include #include #include //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RifHdf5SummaryExporter::ensureHdf5FileIsCreatedMultithreaded( const std::vector& smspecFileNames, const std::vector& h5FileNames, bool createHdfIfNotPresent, int threadCount ) { if ( smspecFileNames.empty() ) return true; if ( smspecFileNames.size() != h5FileNames.size() ) return false; size_t hdfFilesCreatedCount = 0; bool useMultipleThreads = threadCount > 1; #pragma omp parallel for schedule( dynamic ) if ( useMultipleThreads ) num_threads( threadCount ) for ( int cIdx = 0; cIdx < static_cast( smspecFileNames.size() ); ++cIdx ) { const auto& smspecFileName = smspecFileNames[cIdx]; const auto& h5FileName = h5FileNames[cIdx]; RifHdf5SummaryExporter::ensureHdf5FileIsCreated( smspecFileName, h5FileName, createHdfIfNotPresent, hdfFilesCreatedCount ); } if ( hdfFilesCreatedCount > 0 ) { QString txt = QString( "Created [ %1 ] h5 files from a total of [ %2 ] summary files" ) .arg( static_cast( hdfFilesCreatedCount ) ) .arg( static_cast( smspecFileNames.size() ) ); RiaLogging::info( txt ); } return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RifHdf5SummaryExporter::ensureHdf5FileIsCreated( const std::string& smspecFileName, const std::string& h5FileName, bool createHdfIfNotPresent, size_t& hdfFilesCreatedCount ) { // If an H5 file is present, and the SMSPEC file is newer than the H5 file, the H5 file will be recreated. // If no H5 file is present, the H5 file will be created if the flag createHdfIfNotPresent is set to true. // // NB! Always make sure the logic is consistent with the logic in RifOpmCommonEclipseSummary::open if ( !std::filesystem::exists( smspecFileName ) ) return false; bool exportIsRequired = false; bool h5FileExists = std::filesystem::exists( h5FileName ); if ( !h5FileExists ) { if ( createHdfIfNotPresent ) { exportIsRequired = true; } } else if ( RiaFilePathTools::isFirstOlderThanSecond( h5FileName, smspecFileName ) ) { // If both files are present, check if the SMSPEC file is newer than the H5 file. If the SMSPEC file is newer, we abort if it is not // possible to write to the H5 file // Check if we have write permission in the folder QFileInfo info( QString::fromStdString( smspecFileName ) ); if ( !info.isWritable() ) { QString txt = QString( "HDF is older than SMSPEC, but export to file %1 failed due to missing write permissions. Aborting operation." ) .arg( QString::fromStdString( h5FileName ) ); RiaLogging::error( txt ); return false; } exportIsRequired = true; } if ( exportIsRequired ) { try { Opm::EclIO::ESmry sourceSummaryData( smspecFileName ); // Read all data summary data before starting export to HDF. Loading one and one summary vector causes huge // performance penalty sourceSummaryData.loadData(); #pragma omp critical( critical_section_HDF5_export ) { // HDF5 file access is not thread-safe, always make sure we use the HDF5 library from a single thread RifHdf5Exporter exporter( h5FileName ); writeGeneralSection( exporter, sourceSummaryData ); writeSummaryVectors( exporter, sourceSummaryData ); hdfFilesCreatedCount++; } } catch ( std::exception& e ) { QString txt = QString( "HDF export to file %1 failed : %2" ).arg( QString::fromStdString( smspecFileName ), e.what() ); RiaLogging::error( txt ); return false; } } return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RifHdf5SummaryExporter::writeGeneralSection( RifHdf5Exporter& exporter, Opm::EclIO::ESmry& sourceSummaryData ) { auto timesteps = sourceSummaryData.dates(); auto group = exporter.createGroup( nullptr, "general" ); { std::vector values( 1 ); values[0] = -1; exporter.writeDataset( group, "checksum", values ); } { auto startDate = sourceSummaryData.startdate(); // Make sure we convert time identically as the inverse of make_date in ESmry.cpp time_t firstTimeStep = std::chrono::system_clock::to_time_t( startDate ); Opm::TimeStampUTC opmTimeStamp( firstTimeStep ); int day = opmTimeStamp.day(); int month = opmTimeStamp.month(); int year = opmTimeStamp.year(); int hour = opmTimeStamp.hour(); int minute = opmTimeStamp.minutes(); int second = opmTimeStamp.seconds(); std::vector timeValues( 7 ); timeValues[0] = day; timeValues[1] = month; timeValues[2] = year; timeValues[3] = hour; timeValues[4] = minute; timeValues[5] = second; timeValues[6] = 0; // Unknown value, could be millisec exporter.writeDataset( group, "start_date", timeValues ); } { std::vector values( 2 ); values[0] = 1; values[1] = 7; exporter.writeDataset( group, "version", values ); } return true; } //-------------------------------------------------------------------------------------------------- /// //-------------------------------------------------------------------------------------------------- bool RifHdf5SummaryExporter::writeSummaryVectors( RifHdf5Exporter& exporter, Opm::EclIO::ESmry& sourceSummaryData ) { size_t valueCount = sourceSummaryData.numberOfTimeSteps(); if ( valueCount == 0 ) return false; const std::string datasetName( "values" ); std::map> mapVectorNameToSummaryAddresses; auto [addresses, addressToKeywordMap] = RifOpmCommonSummaryTools::buildAddressesAndKeywordMap( sourceSummaryData.keywordList() ); for ( const auto& adr : addresses ) { auto vectorName = adr.vectorName(); if ( mapVectorNameToSummaryAddresses.find( vectorName ) == mapVectorNameToSummaryAddresses.end() ) { mapVectorNameToSummaryAddresses[vectorName] = {}; } auto it = mapVectorNameToSummaryAddresses.find( vectorName ); if ( it != mapVectorNameToSummaryAddresses.end() ) { it->second.push_back( adr ); } } auto summaryVectorsGroup = exporter.createGroup( nullptr, "summary_vectors" ); std::set exportErrorKeywords; for ( const auto& [vectorName, addresses] : mapVectorNameToSummaryAddresses ) { auto keywordGroup = exporter.createGroup( &summaryVectorsGroup, vectorName ); for ( const auto& address : addresses ) { auto keyword = addressToKeywordMap[address]; auto smspecKeywordIndex = sourceSummaryData.getSmspecIndexForKeyword( keyword ); const QString& smspecKeywordText = QString( "%1" ).arg( smspecKeywordIndex ); try { const std::vector& values = sourceSummaryData.get( keyword ); auto dataValuesGroup = exporter.createGroup( &keywordGroup, smspecKeywordText.toStdString() ); exporter.writeDataset( dataValuesGroup, datasetName, values ); dataValuesGroup.close(); } catch ( ... ) { exportErrorKeywords.insert( keyword ); } } keywordGroup.close(); } if ( !exportErrorKeywords.empty() ) { std::vector keywordVector; for ( const auto& k : exportErrorKeywords ) { keywordVector.push_back( k ); } auto txt = RiaStdStringTools::joinStrings( keywordVector, ',' ); QString errorTxt = QString( "Failed to export keywords %1 " ).arg( QString::fromStdString( txt ) ); RiaLogging::error( errorTxt ); } return true; }