/* Copyright 2019 SINTEF Digital, Mathematics and Cybernetics. This file is part of the Open Porous Media project (OPM). OPM 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. OPM 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 for more details. You should have received a copy of the GNU General Public License along with OPM. If not, see . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define BOOST_TEST_MODULE TestRestartSerialization #define BOOST_TEST_NO_MAIN #include #include #include template std::tuple PackUnpack(T& in) { Opm::Serialization::MemPacker packer; Opm::Serializer ser(packer); ser.pack(in); const size_t pos1 = ser.position(); T out{}; ser.unpack(out); const size_t pos2 = ser.position(); return std::make_tuple(std::move(out), pos1, pos2); } #define TEST_FOR_TYPE_NAMED_OBJ(TYPE, NAME, OBJ) \ BOOST_AUTO_TEST_CASE(NAME) \ { \ auto val1 = Opm::TYPE::OBJ(); \ auto val2 = PackUnpack(val1); \ BOOST_CHECK_MESSAGE(std::get<1>(val2) == std::get<2>(val2), "Packed size differ from unpack size for " #TYPE); \ BOOST_CHECK_MESSAGE(val1 == std::get<0>(val2), "Deserialized " #TYPE " differ"); \ } #define TEST_FOR_TYPE_NAMED(TYPE, NAME) \ TEST_FOR_TYPE_NAMED_OBJ(TYPE, NAME, serializationTestObject) #define TEST_FOR_TYPE(TYPE) \ TEST_FOR_TYPE_NAMED(TYPE, TYPE) TEST_FOR_TYPE(ALQState) TEST_FOR_TYPE(GroupState) TEST_FOR_TYPE(HardcodedTimeStepControl) TEST_FOR_TYPE(PIDAndIterationCountTimeStepControl) TEST_FOR_TYPE(PIDTimeStepControl) TEST_FOR_TYPE(SegmentState) TEST_FOR_TYPE(SimpleIterationCountTimeStepControl) TEST_FOR_TYPE(SimulatorTimer) namespace Opm { using ATE = AdaptiveTimeSteppingEbos; } TEST_FOR_TYPE_NAMED_OBJ(ATE, AdaptiveTimeSteppingEbosHardcoded, serializationTestObjectHardcoded) TEST_FOR_TYPE_NAMED_OBJ(ATE, AdaptiveTimeSteppingEbosPID, serializationTestObjectPID) TEST_FOR_TYPE_NAMED_OBJ(ATE, AdaptiveTimeSteppingEbosPIDIt, serializationTestObjectPIDIt) TEST_FOR_TYPE_NAMED_OBJ(ATE, AdaptiveTimeSteppingEbosSimple, serializationTestObjectSimple) namespace Opm { using BPV = BlackOilPrimaryVariables; } TEST_FOR_TYPE_NAMED(BPV, BlackoilPrimaryVariables) namespace Opm { using Disc = Opm::FvBaseDiscretization; using BVec = typename Disc::BlockVectorWrapper; } TEST_FOR_TYPE_NAMED(BVec, BlockVectorWrapper) BOOST_AUTO_TEST_CASE(SingleWellState) { Opm::ParallelWellInfo dummy; auto data_out = Opm::SingleWellState::serializationTestObject(dummy); Opm::Serialization::MemPacker packer; Opm::Serializer ser(packer); ser.pack(data_out); const size_t pos1 = ser.position(); decltype(data_out) data_in("", dummy, false, 0.0, {}, Opm::PhaseUsage{}, 0.0); ser.unpack(data_in); const size_t pos2 = ser.position(); BOOST_CHECK_MESSAGE(pos1 == pos2, "Packed size differ from unpack size for SingleWellState"); BOOST_CHECK_MESSAGE(data_out == data_in, "Deserialized SingleWellState differ"); } BOOST_AUTO_TEST_CASE(WellContainer) { auto data_out = Opm::WellContainer::serializationTestObject(1.0); Opm::Serialization::MemPacker packer; Opm::Serializer ser(packer); ser.pack(data_out); const size_t pos1 = ser.position(); decltype(data_out) data_in; ser.unpack(data_in); const size_t pos2 = ser.position(); BOOST_CHECK_MESSAGE(pos1 == pos2, "Packed size differ from unpack size for WellContainer"); BOOST_CHECK_MESSAGE(data_out == data_in, "Deserialized WellContainer differ"); } BOOST_AUTO_TEST_CASE(EclGenericVanguard) { auto in_params = Opm::EclGenericVanguard::serializationTestParams(); Opm::EclGenericVanguard val1(std::move(in_params)); Opm::Serialization::MemPacker packer; Opm::Serializer ser(packer); ser.pack(val1); const size_t pos1 = ser.position(); Opm::EclGenericVanguard::SimulationModelParams out_params; out_params.setupTime_ = 0.0; out_params.actionState_ = std::make_unique(); out_params.udqState_ = std::make_unique(); out_params.eclSchedule_ = std::make_shared(); out_params.summaryState_ = std::make_unique(); Opm::EclGenericVanguard val2(std::move(out_params)); ser.unpack(val2); const size_t pos2 = ser.position(); BOOST_CHECK_MESSAGE(pos1 == pos2, "Packed size differ from unpack size for EclGenericVanguard"); BOOST_CHECK_MESSAGE(val1 == val2, "Deserialized EclGenericVanguard differ"); } BOOST_AUTO_TEST_CASE(EclGenericProblem) { Opm::EclipseState eclState; Opm::Schedule schedule; Dune::CpGrid grid; #if HAVE_DUNE_FEM using GridPart = Dune::Fem::AdaptiveLeafGridPart; using GridView = Dune::GridView>; auto gridPart = GridPart(grid); auto gridView = GridView(static_cast(gridPart)); #else using GridView = Dune::GridView>; auto gridView = grid.leafGridView(); #endif // HAVE_DUNE_FEM auto data_out = Opm::EclGenericProblem, double>:: serializationTestObject(eclState, schedule, gridView); Opm::Serialization::MemPacker packer; Opm::Serializer ser(packer); ser.pack(data_out); const size_t pos1 = ser.position(); decltype(data_out) data_in(eclState, schedule, gridView); ser.unpack(data_in); const size_t pos2 = ser.position(); BOOST_CHECK_MESSAGE(pos1 == pos2, "Packed size differ from unpack size for EclGenericProblem"); BOOST_CHECK_MESSAGE(data_out == data_in, "Deserialized EclGenericProblem differ"); } template class EclGenericTracerModelTest : public Opm::EclGenericTracerModel { using Base = Opm::EclGenericTracerModel; public: EclGenericTracerModelTest(const GridView& gridView, const Opm::EclipseState& eclState, const Dune::CartesianIndexMapper& cartMapper, const DofMapper& dofMapper, const std::function(int)> centroids) : Base(gridView, eclState, cartMapper, dofMapper, centroids) {} static EclGenericTracerModelTest serializationTestObject(const GridView& gridView, const Opm::EclipseState& eclState, const Dune::CartesianIndexMapper& cartMapper, const DofMapper& dofMapper, const std::function(int)> centroids) { EclGenericTracerModelTest result(gridView, eclState, cartMapper, dofMapper, centroids); result.tracerConcentration_ = {{1.0}, {2.0}, {3.0}}; result.wellTracerRate_.insert({{"foo", "bar"}, 4.0}); return result; } bool operator==(const EclGenericTracerModelTest& rhs) const { if (this->tracerConcentration_.size() != rhs.tracerConcentration_.size()) { return false; } for (size_t i = 0; i < this->tracerConcentration_.size(); ++i) { if (!std::equal(this->tracerConcentration_[i].begin(), this->tracerConcentration_[i].end(), rhs.tracerConcentration_[i].begin(), rhs.tracerConcentration_[i].end())) { return false; } } return this->wellTracerRate_ == rhs.wellTracerRate_; } }; BOOST_AUTO_TEST_CASE(EclGenericTracerModel) { Dune::CpGrid grid; Opm::EclipseState eclState; Dune::CartesianIndexMapper mapper(grid); auto centroids = [](int) { return std::array{}; }; #if HAVE_DUNE_FEM using GridPart = Dune::Fem::AdaptiveLeafGridPart; using GridView = Dune::GridView>; auto gridPart = GridPart(grid); auto gridView = GridView(static_cast(gridPart)); #else using GridView = Dune::GridView>; auto gridView = grid.leafGridView(); #endif // HAVE_DUNE_FEM Dune::MultipleCodimMultipleGeomTypeMapper dofMapper(gridView, Dune::mcmgElementLayout()); auto data_out = EclGenericTracerModelTest, Opm::EcfvStencil, double> ::serializationTestObject(gridView, eclState, mapper, dofMapper, centroids); Opm::Serialization::MemPacker packer; Opm::Serializer ser(packer); ser.pack(data_out); const size_t pos1 = ser.position(); decltype(data_out) data_in(gridView, eclState, mapper, dofMapper, centroids); ser.unpack(data_in); const size_t pos2 = ser.position(); BOOST_CHECK_MESSAGE(pos1 == pos2, "Packed size differ from unpack size for EclGenericTracerModel"); BOOST_CHECK_MESSAGE(data_out == data_in, "Deserialized EclGenericTracerModel differ"); } namespace Opm { class TBatchExport : public EclTracerModel { public: using TBatch = TracerBatch; }; } TEST_FOR_TYPE_NAMED(TBatchExport::TBatch, TracerBatch) namespace { struct AquiferFixture { AquiferFixture() { using TT = Opm::Properties::TTag::EbosTypeTag; const char* argv[] = { "test_RestartSerialization", "--ecl-deck-file-name=GLIFT1.DATA" }; Opm::setupParameters_(2, argv, /*registerParams=*/true); Opm::EclGenericVanguard::setCommunication(std::make_unique()); } }; } BOOST_GLOBAL_FIXTURE(AquiferFixture); #define TEST_FOR_AQUIFER(TYPE) \ BOOST_AUTO_TEST_CASE(TYPE) \ { \ using TT = Opm::Properties::TTag::EbosTypeTag; \ Opm::EclGenericVanguard::readDeck("GLIFT1.DATA"); \ using Simulator = Opm::GetPropType; \ Simulator sim; \ auto data_out = Opm::TYPE::serializationTestObject(sim); \ Opm::Serialization::MemPacker packer; \ Opm::Serializer ser(packer); \ ser.pack(data_out); \ const size_t pos1 = ser.position(); \ decltype(data_out) data_in({}, sim, {}); \ ser.unpack(data_in); \ const size_t pos2 = ser.position(); \ BOOST_CHECK_MESSAGE(pos1 == pos2, "Packed size differ from unpack size for " #TYPE); \ BOOST_CHECK_MESSAGE(data_out == data_in, "Deserialized " #TYPE " differ"); \ } TEST_FOR_AQUIFER(AquiferCarterTracy) TEST_FOR_AQUIFER(AquiferFetkovich) BOOST_AUTO_TEST_CASE(AquiferNumerical) { using TT = Opm::Properties::TTag::EbosTypeTag; Opm::EclGenericVanguard::readDeck("GLIFT1.DATA"); using Simulator = Opm::GetPropType; Simulator sim; auto data_out = Opm::AquiferNumerical::serializationTestObject(sim); Opm::Serialization::MemPacker packer; Opm::Serializer ser(packer); ser.pack(data_out); const size_t pos1 = ser.position(); decltype(data_out) data_in({}, sim); ser.unpack(data_in); const size_t pos2 = ser.position(); BOOST_CHECK_MESSAGE(pos1 == pos2, "Packed size differ from unpack size for AquiferNumerical"); BOOST_CHECK_MESSAGE(data_out == data_in, "Deserialized AquiferNumerical differ"); } bool init_unit_test_func() { return true; } int main(int argc, char** argv) { // MPI setup. int argcDummy = 1; const char *tmp[] = {"test_RestartSerialization"}; char **argvDummy = const_cast(tmp); #if HAVE_DUNE_FEM Dune::Fem::MPIManager::initialize(argcDummy, argvDummy); #else Dune::MPIHelper::instance(argcDummy, argvDummy); #endif return boost::unit_test::unit_test_main(&init_unit_test_func, argc, argv); }