/* Copyright 2013 Statoil ASA. 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "Well/injection.hpp" #include "MSW/Compsegs.hpp" namespace Opm { namespace { bool name_match(const std::string& pattern, const std::string& name) { int flags = 0; return (fnmatch(pattern.c_str(), name.c_str(), flags) == 0); } std::pair restart_info(const RestartIO::RstState * rst) { if (!rst) return std::make_pair(std::time_t{0}, std::size_t{0}); else return rst->header.restart_info(); } } Schedule::Schedule( const Deck& deck, const EclipseGrid& grid, const FieldPropsManager& fp, const Runspec &runspec, const ParseContext& parseContext, ErrorGuard& errors, [[maybe_unused]] std::shared_ptr python, const RestartIO::RstState * rst) try : m_static( python, deck, runspec ), m_sched_deck(deck, restart_info(rst) ), m_timeMap( deck , restart_info( rst )), rft_config(this->m_timeMap), restart_config(m_timeMap, deck, parseContext, errors) { if (rst) { auto restart_step = rst->header.restart_info().second; this->iterateScheduleSection( 0, restart_step, parseContext, errors, false, nullptr, &grid, &fp); this->load_rst(*rst, grid, fp); this->iterateScheduleSection( restart_step, this->m_sched_deck.size(), parseContext, errors, false, nullptr, &grid, &fp); } else this->iterateScheduleSection( 0, this->m_sched_deck.size(), parseContext, errors, false, nullptr, &grid, &fp); /* The code in the #ifdef SCHEDULE_DEBUG is an enforced integration test to assert the sanity of the ongoing Schedule refactoring. This will be removed when the Schedule refactoring is complete. */ #ifdef SCHEDULE_DEBUG if (this->size() == 0) return; // Verify that the time schedule is correct. for (std::size_t report_step = 0; report_step < this->size() - 1; report_step++) { const auto& this_block = this->m_sched_deck[report_step]; if (this_block.start_time() != std::chrono::system_clock::from_time_t(this->m_timeMap[report_step])) { auto msg = fmt::format("Block: Bug in start_time for report_step: {} ", report_step); throw std::logic_error(msg); } const auto& next_block = this->m_sched_deck[report_step + 1]; if (this_block.end_time() != next_block.start_time()) throw std::logic_error("Block: Internal bug in sched_block start / end inconsistent"); const auto& this_step = this->operator[](report_step); if (this_step.start_time() != std::chrono::system_clock::from_time_t(this->m_timeMap[report_step])) { auto msg = fmt::format("Bug in start_time for report_step: {} ", report_step); throw std::logic_error(msg); } const auto& next_step = this->operator[](report_step + 1); if (this_step.end_time() != next_step.start_time()) throw std::logic_error( fmt::format("Internal bug in sched_step start / end inconsistent report:{}", report_step)); } #endif } catch (const OpmInputError& opm_error) { throw; } catch (const std::exception& std_error) { OpmLog::error(fmt::format("An error occured while creating the reservoir schedule\n" "Internal error: {}", std_error.what())); throw; } template Schedule::Schedule( const Deck& deck, const EclipseGrid& grid, const FieldPropsManager& fp, const Runspec &runspec, const ParseContext& parseContext, T&& errors, std::shared_ptr python, const RestartIO::RstState * rst) : Schedule(deck, grid, fp, runspec, parseContext, errors, python, rst) {} Schedule::Schedule( const Deck& deck, const EclipseGrid& grid, const FieldPropsManager& fp, const Runspec &runspec, std::shared_ptr python, const RestartIO::RstState * rst) : Schedule(deck, grid, fp, runspec, ParseContext(), ErrorGuard(), python, rst) {} Schedule::Schedule(const Deck& deck, const EclipseState& es, const ParseContext& parse_context, ErrorGuard& errors, std::shared_ptr python, const RestartIO::RstState * rst) : Schedule(deck, es.getInputGrid(), es.fieldProps(), es.runspec(), parse_context, errors, python, rst) {} template Schedule::Schedule(const Deck& deck, const EclipseState& es, const ParseContext& parse_context, T&& errors, std::shared_ptr python, const RestartIO::RstState * rst) : Schedule(deck, es.getInputGrid(), es.fieldProps(), es.runspec(), parse_context, errors, python, rst) {} Schedule::Schedule(const Deck& deck, const EclipseState& es, std::shared_ptr python, const RestartIO::RstState * rst) : Schedule(deck, es, ParseContext(), ErrorGuard(), python, rst) {} Schedule::Schedule(const Deck& deck, const EclipseState& es, const RestartIO::RstState * rst) : Schedule(deck, es, ParseContext(), ErrorGuard(), std::make_shared(), rst) {} Schedule::Schedule(std::shared_ptr python_handle) : m_static( python_handle ) { } /* In general the serializeObject() instances are used as targets for deserialization, i.e. the serialized buffer is unpacked into this instance. However the Schedule object is a top level object, and the simulator will instantiate and manage a Schedule object to unpack into, so the instance created here is only for testing. */ Schedule Schedule::serializeObject() { Schedule result; result.m_static = ScheduleStatic::serializeObject(); result.m_timeMap = TimeMap::serializeObject(); result.rft_config = RFTConfig::serializeObject(); result.restart_config = RestartConfig::serializeObject(); result.snapshots = { ScheduleState::serializeObject() }; return result; } std::time_t Schedule::getStartTime() const { return this->posixStartTime( ); } time_t Schedule::posixStartTime() const { return m_timeMap.getStartTime( 0 ); } time_t Schedule::posixEndTime() const { return this->m_timeMap.getEndTime(); } void Schedule::handleKeyword(std::size_t currentStep, const ScheduleBlock& block, const DeckKeyword& keyword, const ParseContext& parseContext, ErrorGuard& errors, const EclipseGrid* grid, const FieldPropsManager* fp, const std::vector& matching_wells, bool runtime, const std::unordered_map * target_wellpi, std::vector >& rftProperties) { static const std::unordered_set require_grid = { "COMPDAT", "COMPSEGS" }; HandlerContext handlerContext { block, keyword, currentStep, matching_wells, runtime , target_wellpi}; /* The grid and fieldProps members create problems for reiterating the Schedule section. We therefor single them out very clearly here. */ if (require_grid.count(keyword.name()) > 0) { handlerContext.grid_ptr = grid; handlerContext.fp_ptr = fp; } if (handleNormalKeyword(handlerContext, parseContext, errors)) return; else if (keyword.name() == "WRFT") rftProperties.push_back( std::make_pair( &keyword , currentStep )); else if (keyword.name() == "WRFTPLT") rftProperties.push_back( std::make_pair( &keyword , currentStep )); else if (keyword.name() == "PYACTION") handlePYACTION(keyword); } namespace { class ScheduleLogger { public: explicit ScheduleLogger(bool restart_skip) { if (restart_skip) this->log_function = &OpmLog::note; else this->log_function = &OpmLog::info; } void operator()(const std::string& msg) { this->log_function(msg); } void info(const std::string& msg) { OpmLog::info(msg); } void complete_step(const std::string& msg) { this->step_count += 1; if (this->step_count == this->max_print) { this->log_function(msg); OpmLog::info("Report limit reached, see PRT-file for remaining Schedule initialization.\n"); this->log_function = &OpmLog::note; } else this->log_function( msg + "\n"); }; void restart() { this->step_count = 0; this->log_function = &OpmLog::info; } private: std::size_t step_count = 0; std::size_t max_print = 5; void (*log_function)(const std::string&); }; } void Schedule::iterateScheduleSection(std::size_t load_start, std::size_t load_end, const ParseContext& parseContext , ErrorGuard& errors, bool runtime, const std::unordered_map * target_wellpi, const EclipseGrid* grid, const FieldPropsManager* fp) { std::vector > rftProperties; std::string time_unit = this->m_static.m_unit_system.name(UnitSystem::measure::time); auto deck_time = [this](double seconds) { return this->m_static.m_unit_system.from_si(UnitSystem::measure::time, seconds); }; std::string current_file; const auto& time_map = this->m_timeMap; /* The keywords in the skiprest_whitelist set are loaded from the SCHEDULE section even though the SKIPREST keyword is in action. The full list includes some additional keywords which we do not support at all. */ std::unordered_set skiprest_whitelist = {"VFPPROD", "VFPINJ", "RPTSCHED", "RPTRST", "TUNING", "MESSAGES"}; /* The behavior of variable restart_skip is more lenient than the SKIPREST keyword. If this is a restarted[1] run the loop iterating over keywords will skip the all keywords[2] until DATES keyword with the restart date is encountered - irrespective of whether the SKIPREST keyword is present in the deck or not. [1]: opm/flow can restart in a mode where all the keywords from the historical part of the Schedule section is internalized, and only the solution fields are read from the restart file. In this case we will have TimeMap::restart_offset() == 0. [2]: With the exception of the keywords in the skiprest_whitelist; these keywords will be assigned to report step 0. */ auto restart_skip = load_start < this->m_timeMap.restart_offset(); ScheduleLogger logger(restart_skip); { const auto& location = this->m_sched_deck.location(); current_file = location.filename; logger.info(fmt::format("\nProcessing dynamic information from\n{} line {}", current_file, location.lineno)); if (restart_skip) logger.info(fmt::format("This is a restarted run - skipping until report step {} at {}", time_map.restart_offset(), Schedule::formatDate(time_map.restart_time()))); logger(fmt::format("Initializing report step {}/{} at {} {} {} line {}", load_start + 1, this->size(), Schedule::formatDate(this->getStartTime()), deck_time(time_map.getTimePassedUntil(load_start)), time_unit, location.lineno)); } for (auto report_step = load_start; report_step < load_end; report_step++) { std::size_t keyword_index = 0; auto& block = this->m_sched_deck[report_step]; auto time_type = block.time_type(); if (time_type == ScheduleTimeType::DATES || time_type == ScheduleTimeType::TSTEP) { const auto& start_date = Schedule::formatDate(std::chrono::system_clock::to_time_t(block.start_time())); const auto& days = deck_time(this->stepLength(report_step - 1)); const auto& days_total = deck_time(time_map.getTimePassedUntil(report_step)); logger.complete_step(fmt::format("Complete report step {0} ({1} {2}) at {3} ({4} {2})", report_step, days, time_unit, start_date, days_total)); logger(fmt::format("Initializing report step {}/{} at {} ({} {}) - line {}", report_step + 1, this->size(), start_date, days_total, time_unit, block.location().lineno)); } this->create_next(block); while (true) { if (keyword_index == block.size()) break; const auto& keyword = block[keyword_index]; const auto& location = keyword.location(); if (location.filename != current_file) { logger(fmt::format("Reading from: {} line {}", location.filename, location.lineno)); current_file = location.filename; } if (keyword.name() == "ACTIONX") { Action::ActionX action(keyword, this->m_timeMap.getStartTime(report_step)); while (true) { keyword_index++; if (keyword_index == block.size()) throw OpmInputError("Missing keyword ENDACTIO", keyword.location()); const auto& action_keyword = block[keyword_index]; if (action_keyword.name() == "ENDACTIO") break; if (Action::ActionX::valid_keyword(action_keyword.name())) action.addKeyword(action_keyword); else { std::string msg_fmt = "The keyword {keyword} is not supported in the ACTIONX block\n" "In {file} line {line}."; parseContext.handleError( ParseContext::ACTIONX_ILLEGAL_KEYWORD, msg_fmt, action_keyword.location(), errors); } } this->addACTIONX(action); keyword_index++; continue; } logger(fmt::format("Processing keyword {} at line {}", location.keyword, location.lineno)); this->handleKeyword(report_step, block, keyword, parseContext, errors, grid, fp, {}, runtime, target_wellpi, rftProperties); keyword_index++; } checkIfAllConnectionsIsShut(report_step); } for (auto rftPair = rftProperties.begin(); rftPair != rftProperties.end(); ++rftPair) { const DeckKeyword& keyword = *rftPair->first; std::size_t timeStep = rftPair->second; if (keyword.name() == "WRFT") applyWRFT(keyword, timeStep); if (keyword.name() == "WRFTPLT") applyWRFTPLT(keyword, timeStep); } } void Schedule::addACTIONX(const Action::ActionX& action) { auto new_actions = this->snapshots.back().actions.get(); new_actions.add( action ); this->snapshots.back().actions.update( std::move(new_actions) ); } void Schedule::handlePYACTION(const DeckKeyword& keyword) { if (!this->m_static.m_python_handle->enabled()) { //Must have a real Python instance here - to ensure that IMPORT works const auto& loc = keyword.location(); OpmLog::warning("This version of flow is built without support for Python. Keyword PYACTION in file: " + loc.filename + " line: " + std::to_string(loc.lineno) + " is ignored."); return; } const auto& name = keyword.getRecord(0).getItem().get(0); const auto& run_count = Action::PyAction::from_string( keyword.getRecord(0).getItem().get(0) ); const auto& module_arg = keyword.getRecord(1).getItem().get(0); std::string module; if (this->m_static.m_input_path.empty()) module = module_arg; else module = this->m_static.m_input_path + "/" + module_arg; Action::PyAction pyaction(this->m_static.m_python_handle, name, run_count, module); auto new_actions = this->snapshots.back().actions.get(); new_actions.add(pyaction); this->snapshots.back().actions.update( std::move(new_actions) ); } void Schedule::applyEXIT(const DeckKeyword& keyword, std::size_t report_step) { int status = keyword.getRecord(0).getItem().get(0); OpmLog::info("Simulation exit with status: " + std::to_string(status) + " requested as part of ACTIONX at report_step: " + std::to_string(report_step)); this->exit_status = status; } void Schedule::shut_well(const std::string& well_name, std::size_t report_step) { this->updateWellStatus(well_name, report_step, Well::Status::SHUT); } void Schedule::open_well(const std::string& well_name, std::size_t report_step) { this->updateWellStatus(well_name, report_step, Well::Status::OPEN); } void Schedule::stop_well(const std::string& well_name, std::size_t report_step) { this->updateWellStatus(well_name, report_step, Well::Status::STOP); } /* Function is quite dangerous - because if this is called while holding a Well pointer that will go stale and needs to be refreshed. */ bool Schedule::updateWellStatus( const std::string& well_name, std::size_t reportStep , Well::Status status, std::optional location) { auto well2 = this->snapshots[reportStep].wells.get(well_name); if (well2.getConnections().empty() && status == Well::Status::OPEN) { if (location) { auto msg = fmt::format("Problem with{}\n", "In {} line{}\n" "Well {} has no connections to grid and will remain SHUT", location->keyword, location->filename, location->lineno, well_name); OpmLog::warning(msg); } else OpmLog::warning(fmt::format("Well {} has no connections to grid and will remain SHUT", well_name)); return false; } auto old_status = well2.getStatus(); bool update = false; if (well2.updateStatus(status)) { if (status == Well::Status::OPEN) this->rft_config.addWellOpen(well_name, reportStep); /* The Well::updateStatus() will always return true because a new WellStatus object should be created. But the new object might have the same value as the previous object; therefor we need to check for an actual status change before we emit a WELL_STATUS_CHANGE event. */ if (old_status != status) { this->snapshots.back().events().addEvent( ScheduleEvents::WELL_STATUS_CHANGE); this->snapshots.back().wellgroup_events().addEvent( well2.name(), ScheduleEvents::WELL_STATUS_CHANGE); } this->snapshots[reportStep].wells.update( std::move(well2) ); update = true; } return update; } bool Schedule::updateWPAVE(const std::string& wname, std::size_t report_step, const PAvg& pavg) { const auto& well = this->getWell(wname, report_step); if (well.pavg() != pavg) { auto new_well = this->snapshots[report_step].wells.get(wname); new_well.updateWPAVE( pavg ); this->snapshots[report_step].wells.update( std::move(new_well) ); return true; } return false; } void Schedule::applyWELOPEN(const DeckKeyword& keyword, std::size_t currentStep, bool runtime, const ParseContext& parseContext, ErrorGuard& errors, const std::vector& matching_wells) { auto conn_defaulted = []( const DeckRecord& rec ) { auto defaulted = []( const DeckItem& item ) { return item.defaultApplied( 0 ); }; return std::all_of( rec.begin() + 2, rec.end(), defaulted ); }; constexpr auto open = Well::Status::OPEN; for (const auto& record : keyword) { const auto& wellNamePattern = record.getItem( "WELL" ).getTrimmedString(0); const auto& status_str = record.getItem( "STATUS" ).getTrimmedString( 0 ); const auto well_names = this->wellNames(wellNamePattern, currentStep, matching_wells); if (well_names.empty()) invalidNamePattern( wellNamePattern, currentStep, parseContext, errors, keyword); /* if all records are defaulted or just the status is set, only * well status is updated */ if (conn_defaulted( record )) { const auto well_status = Well::StatusFromString( status_str ); for (const auto& wname : well_names) { { const auto& well = this->getWell(wname, currentStep); if( well_status == open && !well.canOpen() ) { auto days = m_timeMap.getTimePassedUntil( currentStep ) / (60 * 60 * 24); std::string msg = "Well " + wname + " where crossflow is banned has zero total rate." + " This well is prevented from opening at " + std::to_string( days ) + " days"; OpmLog::note(msg); } else { this->updateWellStatus( wname, currentStep, well_status); if (well_status == open) this->rft_config.addWellOpen(wname, currentStep); } } } continue; } /* Some of the connection information has been entered, in this case we *only* update the status of the connections, and not the well itself. Unless all connections are shut - then the well is also shut. */ for (const auto& wname : well_names) { if (!runtime) { auto well = this->snapshots[currentStep].wells.get(wname); this->snapshots[currentStep].wells.update( std::move(well) ); } const auto connection_status = Connection::StateFromString( status_str ); { auto well = this->snapshots[currentStep].wells.get(wname); well.handleWELOPENConnections(record, connection_status, runtime); this->snapshots[currentStep].wells.update( std::move(well) ); } this->snapshots.back().events().addEvent( ScheduleEvents::COMPLETION_CHANGE); } } } void Schedule::applyWRFT(const DeckKeyword& keyword, std::size_t currentStep) { /* Rule for handling RFT: Request current RFT data output for specified wells, plus output when * any well is subsequently opened */ for (const auto& record : keyword) { const std::string& wellNamePattern = record.getItem("WELL").getTrimmedString(0); const auto well_names = wellNames(wellNamePattern, currentStep); for (const auto& well_name : well_names) this->rft_config.updateRFT(well_name, currentStep, RFTConfig::RFT::YES); } this->rft_config.setWellOpenRFT(currentStep); } void Schedule::applyWRFTPLT(const DeckKeyword& keyword, std::size_t currentStep) { for (const auto& record : keyword) { const std::string& wellNamePattern = record.getItem("WELL").getTrimmedString(0); RFTConfig::RFT RFTKey = RFTConfig::RFTFromString(record.getItem("OUTPUT_RFT").getTrimmedString(0)); RFTConfig::PLT PLTKey = RFTConfig::PLTFromString(record.getItem("OUTPUT_PLT").getTrimmedString(0)); const auto well_names = wellNames(wellNamePattern, currentStep); for (const auto& well_name : well_names) { this->rft_config.updateRFT(well_name, currentStep, RFTKey); this->rft_config.updatePLT(well_name, currentStep, PLTKey); } } } const RFTConfig& Schedule::rftConfig() const { return this->rft_config; } void Schedule::invalidNamePattern( const std::string& namePattern, std::size_t, const ParseContext& parseContext, ErrorGuard& errors, const DeckKeyword& keyword ) const { std::string msg_fmt = fmt::format("No wells/groups match the pattern: \'{}\'", namePattern); parseContext.handleError( ParseContext::SCHEDULE_INVALID_NAME, msg_fmt, keyword.location(), errors ); } const TimeMap& Schedule::getTimeMap() const { return this->m_timeMap; } GTNode Schedule::groupTree(const std::string& root_node, std::size_t report_step, std::size_t level, const std::optional& parent_name) const { auto root_group = this->getGroup(root_node, report_step); GTNode tree(root_group, level, parent_name); for (const auto& wname : root_group.wells()) { const auto& well = this->getWell(wname, report_step); tree.add_well(well); } for (const auto& gname : root_group.groups()) { auto child_group = this->groupTree(gname, report_step, level + 1, root_node); tree.add_group(child_group); } return tree; } GTNode Schedule::groupTree(const std::string& root_node, std::size_t report_step) const { return this->groupTree(root_node, report_step, 0, {}); } GTNode Schedule::groupTree(std::size_t report_step) const { return this->groupTree("FIELD", report_step); } void Schedule::addWell(const std::string& wellName, const DeckRecord& record, std::size_t timeStep, Connection::Order wellConnectionOrder) { // We change from eclipse's 1 - n, to a 0 - n-1 solution int headI = record.getItem("HEAD_I").get< int >(0) - 1; int headJ = record.getItem("HEAD_J").get< int >(0) - 1; Phase preferredPhase; { const std::string phaseStr = record.getItem("PHASE").getTrimmedString(0); if (phaseStr == "LIQ") { // We need a workaround in case the preferred phase is "LIQ", // which is not proper phase and will cause the get_phase() // function to throw. In that case we choose to treat it as OIL. preferredPhase = Phase::OIL; OpmLog::warning("LIQ_PREFERRED_PHASE", "LIQ preferred phase not supported for well " + wellName + ", using OIL instead"); } else { preferredPhase = get_phase(phaseStr); } } const auto& refDepthItem = record.getItem("REF_DEPTH"); std::optional ref_depth; if (refDepthItem.hasValue( 0 )) ref_depth = refDepthItem.getSIDouble( 0 ); double drainageRadius = record.getItem( "D_RADIUS" ).getSIDouble(0); bool allowCrossFlow = true; const std::string& allowCrossFlowStr = record.getItem().getTrimmedString(0); if (allowCrossFlowStr == "NO") allowCrossFlow = false; bool automaticShutIn = true; const std::string& automaticShutInStr = record.getItem().getTrimmedString(0); if (automaticShutInStr == "STOP") { automaticShutIn = false; } const std::string& group = record.getItem().getTrimmedString(0); auto pvt_table = record.getItem().get(0); auto gas_inflow = Well::GasInflowEquationFromString( record.getItem().get(0) ); this->addWell(wellName, group, headI, headJ, preferredPhase, ref_depth, drainageRadius, allowCrossFlow, automaticShutIn, pvt_table, gas_inflow, timeStep, wellConnectionOrder); } void Schedule::addWell(Well well) { const std::string wname = well.name(); auto& sched_state = this->snapshots.back(); sched_state.events().addEvent( ScheduleEvents::NEW_WELL ); sched_state.wellgroup_events().addWell( wname ); { auto wo = sched_state.well_order.get(); wo.add( wname ); sched_state.well_order.update( std::move(wo) ); } well.setInsertIndex(sched_state.wells.size()); sched_state.wells.update( std::move(well) ); } void Schedule::addWell(const std::string& wellName, const std::string& group, int headI, int headJ, Phase preferredPhase, const std::optional& ref_depth, double drainageRadius, bool allowCrossFlow, bool automaticShutIn, int pvt_table, Well::GasInflowEquation gas_inflow, std::size_t timeStep, Connection::Order wellConnectionOrder) { const auto& sched_state = this->operator[](timeStep); Well well(wellName, group, timeStep, 0, headI, headJ, ref_depth, WellType(preferredPhase), sched_state.whistctl(), wellConnectionOrder, this->m_static.m_unit_system, this->getUDQConfig(timeStep).params().undefinedValue(), drainageRadius, allowCrossFlow, automaticShutIn, pvt_table, gas_inflow); this->addWell( std::move(well) ); const auto& ts = this->operator[](timeStep); this->updateWPAVE( wellName, timeStep, ts.pavg.get() ); } std::size_t Schedule::numWells() const { return this->snapshots.back().wells.size(); } std::size_t Schedule::numWells(std::size_t timestep) const { auto well_names = this->wellNames(timestep); return well_names.size(); } bool Schedule::hasWell(const std::string& wellName) const { return this->snapshots.back().wells.has(wellName); } bool Schedule::hasWell(const std::string& wellName, std::size_t timeStep) const { return this->snapshots[timeStep].wells.has(wellName); } std::vector< const Group* > Schedule::getChildGroups2(const std::string& group_name, std::size_t timeStep) const { const auto& sched_state = this->snapshots[timeStep]; const auto& group = sched_state.groups.get(group_name); std::vector child_groups; for (const auto& child_name : group.groups()) child_groups.push_back( std::addressof(this->getGroup(child_name, timeStep))); return child_groups; } std::vector< Well > Schedule::getChildWells2(const std::string& group_name, std::size_t timeStep) const { const auto& sched_state = this->snapshots[timeStep]; const auto& group = sched_state.groups.get(group_name); std::vector wells; if (group.groups().size()) { for (const auto& child_name : group.groups()) { const auto& child_wells = this->getChildWells2(child_name, timeStep); wells.insert(wells.end(), child_wells.begin(), child_wells.end()); } } else { for (const auto& well_name : group.wells()) { wells.push_back( this->getWell(well_name, timeStep)); } } return wells; } /* This function will return a list of wells which have changed *structurally* in the last report_step; wells where only production settings have changed will not be included. */ std::vector Schedule::changed_wells(std::size_t report_step) const { std::vector wells; const auto& state = this->snapshots[report_step]; const auto& all_wells = state.wells(); if (report_step == 0) std::transform( all_wells.begin(), all_wells.end(), std::back_inserter(wells), [] (const auto& well_ref) { return well_ref.get().name(); }); else { const auto& prev_state = this->snapshots[report_step - 1]; for (const auto& well_ref : all_wells) { const auto& wname = well_ref.get().name(); if (prev_state.wells.has(wname)) { const auto& prev_well = prev_state.wells.get( wname ); if (!prev_well.cmp_structure(well_ref.get())) wells.push_back( wname ); } else wells.push_back( wname ); } } return wells; } std::vector Schedule::getWells(std::size_t timeStep) const { std::vector wells; if (timeStep >= this->m_timeMap.size()) throw std::invalid_argument("timeStep argument beyond the length of the simulation"); const auto& well_order = this->snapshots[timeStep].well_order(); for (const auto& wname : well_order) wells.push_back( this->snapshots[timeStep].wells.get(wname) ); return wells; } std::vector Schedule::getWellsatEnd() const { return this->getWells(this->snapshots.size() - 1); } const Well& Schedule::getWellatEnd(const std::string& well_name) const { return this->getWell(well_name, this->snapshots.size() - 1); } const Well& Schedule::getWell(const std::string& wellName, std::size_t timeStep) const { return this->snapshots[timeStep].wells.get(wellName); } const Well& Schedule::getWell(std::size_t well_index, std::size_t timeStep) const { const auto find_pred = [well_index] (const auto& well_pair) -> bool { return well_pair.second->seqIndex() == well_index; }; auto well_ptr = this->snapshots[timeStep].wells.find( find_pred ); if (well_ptr == nullptr) throw std::invalid_argument(fmt::format("There is no well with well_index:{} at report_step:{}", well_index, timeStep)); return *well_ptr; } const Group& Schedule::getGroup(const std::string& groupName, std::size_t timeStep) const { return this->snapshots[timeStep].groups.get(groupName); } void Schedule::updateGuideRateModel(const GuideRateModel& new_model, std::size_t report_step) { auto new_config = this->snapshots[report_step].guide_rate(); if (new_config.update_model(new_model)) this->snapshots[report_step].guide_rate.update( std::move(new_config) ); } /* There are many SCHEDULE keyword which take a wellname as argument. In addition to giving a fully qualified name like 'W1' you can also specify shell wildcard patterns like like 'W*', you can get all the wells in the well-list '*WL'[1] and the wellname '?' is used to get all the wells which already have matched a condition in a ACTIONX keyword. This function should be one-stop function to get all well names according to a input pattern. The timestep argument is used to check that the wells have indeed been defined at the point in time we are considering. [1]: The leading '*' in a WLIST name should not be interpreted as a shell wildcard! */ std::vector Schedule::wellNames(const std::string& pattern, std::size_t timeStep, const std::vector& matching_wells) const { // ACTIONX handler if (pattern == "?") return { matching_wells.begin(), matching_wells.end() }; auto wm = this->wellMatcher(timeStep); return wm.wells(pattern); } WellMatcher Schedule::wellMatcher(std::size_t report_step) const { const ScheduleState * sched_state; if (report_step < this->snapshots.size()) sched_state = &this->snapshots[report_step]; else sched_state = &this->snapshots.back(); return WellMatcher(sched_state->well_order.get(), sched_state->wlist_manager.get()); } std::vector Schedule::wellNames(const std::string& pattern) const { return this->wellNames(pattern, this->size() - 1); } std::vector Schedule::wellNames(std::size_t timeStep) const { const auto& well_order = this->snapshots[timeStep].well_order(); return well_order.names(); } std::vector Schedule::wellNames() const { const auto& well_order = this->snapshots.back().well_order(); return well_order.names(); } std::vector Schedule::groupNames(const std::string& pattern, std::size_t timeStep) const { if (pattern.size() == 0) return {}; const auto& group_order = this->snapshots[timeStep].group_order(); // Normal pattern matching auto star_pos = pattern.find('*'); if (star_pos != std::string::npos) { std::vector names; for (const auto& gname : group_order) { if (name_match(pattern, gname)) names.push_back(gname); } return names; } // Normal group name without any special characters if (group_order.has(pattern)) return { pattern }; return {}; } std::vector Schedule::groupNames(std::size_t timeStep) const { const auto& group_order = this->snapshots[timeStep].group_order(); return group_order.names(); } std::vector Schedule::groupNames(const std::string& pattern) const { return this->groupNames(pattern, this->snapshots.size() - 1); } std::vector Schedule::groupNames() const { const auto& group_order = this->snapshots.back().group_order(); return group_order.names(); } std::vector Schedule::restart_groups(std::size_t timeStep) const { const auto& restart_groups = this->snapshots[timeStep].group_order().restart_groups(); std::vector rst_groups(restart_groups.size() , nullptr ); for (std::size_t restart_index = 0; restart_index < restart_groups.size(); restart_index++) { const auto& group_name = restart_groups[restart_index]; if (group_name.has_value()) rst_groups[restart_index] = &this->getGroup(group_name.value(), timeStep); } return rst_groups; } void Schedule::addGroup(const std::string& groupName, std::size_t timeStep) { auto udq_undefined = this->getUDQConfig(timeStep).params().undefinedValue(); auto& sched_state = this->snapshots.back(); auto insert_index = sched_state.groups.size(); sched_state.groups.update( Group{ groupName, insert_index, udq_undefined, this->m_static.m_unit_system} ); sched_state.events().addEvent( ScheduleEvents::NEW_GROUP ); sched_state.wellgroup_events().addGroup(groupName); { auto go = sched_state.group_order.get(); go.add( groupName ); sched_state.group_order.update( std::move(go) ); } // All newly created groups are attached to the field group, // can then be relocated with the GRUPTREE keyword. if (groupName != "FIELD") this->addGroupToGroup("FIELD", groupName, timeStep); } void Schedule::addGroupToGroup( const std::string& parent_name, const std::string& child_name, std::size_t timeStep) { auto parent_group = this->snapshots.back().groups.get(parent_name); if (parent_group.addGroup(child_name)) this->snapshots.back().groups.update( std::move(parent_group) ); // Check and update backreference in child const auto& child_group = this->getGroup(child_name, timeStep); if (child_group.parent() != parent_name) { auto old_parent = this->snapshots.back().groups.get(child_group.parent()); old_parent.delGroup(child_group.name()); this->snapshots.back().groups.update( std::move(old_parent) ); auto new_child_group = Group{ child_group }; new_child_group.updateParent(parent_name); this->snapshots.back().groups.update( std::move(new_child_group) ); } } void Schedule::addWellToGroup( const std::string& group_name, const std::string& well_name , std::size_t timeStep) { auto well = this->getWell(well_name, timeStep); const auto old_gname = well.groupName(); if (old_gname != group_name) { well.updateGroup(group_name); this->snapshots.back().wells.update( std::move(well) ); this->snapshots.back().wellgroup_events().addEvent( well_name, ScheduleEvents::WELL_WELSPECS_UPDATE ); // Remove well child reference from previous group auto group = this->snapshots.back().groups.get( old_gname ); group.delWell(well_name); this->snapshots.back().groups.update( std::move(group) ); } // Add well child reference to new group auto group = this->snapshots.back().groups.get( group_name ); group.addWell(well_name); this->snapshots.back().groups.update( std::move(group) ); this->snapshots.back().events().addEvent( ScheduleEvents::GROUP_CHANGE ); } Well::ProducerCMode Schedule::getGlobalWhistctlMmode(std::size_t timestep) const { return this->operator[](timestep).whistctl(); } void Schedule::checkIfAllConnectionsIsShut(std::size_t timeStep) { const auto& well_names = this->wellNames(timeStep); for (const auto& wname : well_names) { const auto& well = this->getWell(wname, timeStep); const auto& connections = well.getConnections(); if (connections.allConnectionsShut() && well.getStatus() != Well::Status::SHUT) { std::string msg = "All completions in well " + well.name() + " is shut at " + std::to_string ( m_timeMap.getTimePassedUntil(timeStep) / (60*60*24) ) + " days. \n" + "The well is therefore also shut."; OpmLog::note(msg); this->updateWellStatus( well.name(), timeStep, Well::Status::SHUT); } } } void Schedule::filterConnections(const ActiveGridCells& grid) { for (auto& sched_state : this->snapshots) { for (auto& well : sched_state.wells()) { well.get().filterConnections(grid); } } } const UDQConfig& Schedule::getUDQConfig(std::size_t timeStep) const { return this->snapshots[timeStep].udq.get(); } const GuideRateConfig& Schedule::guideRateConfig(std::size_t timeStep) const { return this->snapshots[timeStep].guide_rate(); } std::optional Schedule::exitStatus() const { return this->exit_status; } std::size_t Schedule::size() const { return this->m_timeMap.size(); } double Schedule::seconds(std::size_t timeStep) const { return this->m_timeMap.seconds(timeStep); } time_t Schedule::simTime(std::size_t timeStep) const { return this->m_timeMap[timeStep]; } double Schedule::stepLength(std::size_t timeStep) const { return this->m_timeMap.getTimeStepLength(timeStep); } void Schedule::applyAction(std::size_t reportStep, const std::chrono::system_clock::time_point&, const Action::ActionX& action, const Action::Result& result, const std::unordered_map& target_wellpi) { ParseContext parseContext; ErrorGuard errors; std::vector > ignored_rftProperties; this->snapshots.resize(reportStep + 1); auto& input_block = this->m_sched_deck[reportStep]; for (const auto& keyword : action) { input_block.push_back(keyword); this->handleKeyword(reportStep, input_block, keyword, parseContext, errors, nullptr, nullptr, result.wells(), true, &target_wellpi, ignored_rftProperties); } if (reportStep < this->m_sched_deck.size() - 1) iterateScheduleSection(reportStep + 1, this->m_sched_deck.size(), parseContext, errors, true, &target_wellpi, nullptr, nullptr); } void Schedule::applyWellProdIndexScaling(const std::string& well_name, const std::size_t reportStep, const double newWellPI) { if (reportStep >= this->snapshots.size()) return; if (!this->snapshots[reportStep].wells.has(well_name)) return; std::vector unique_wells; for (std::size_t step = reportStep; step < this->snapshots.size(); step++) { auto& well = this->snapshots[step].wells.get(well_name); if (unique_wells.empty() || (!(*unique_wells.back() == well))) unique_wells.push_back( &well ); } std::vector scalingApplicable; const auto targetPI = this->snapshots[reportStep].target_wellpi.at(well_name); auto prev_well = unique_wells[0]; auto scalingFactor = prev_well->convertDeckPI(targetPI) / newWellPI; prev_well->applyWellProdIndexScaling(scalingFactor, scalingApplicable); for (std::size_t well_index = 1; well_index < unique_wells.size(); well_index++) { auto wellPtr = unique_wells[well_index]; if (! wellPtr->hasSameConnectionsPointers(*prev_well)) { wellPtr->applyWellProdIndexScaling(scalingFactor, scalingApplicable); prev_well = wellPtr; } } } RestartConfig& Schedule::restart() { return this->restart_config; } const RestartConfig& Schedule::restart() const { return this->restart_config; } bool Schedule::operator==(const Schedule& data) const { return this->m_timeMap == data.m_timeMap && this->m_static == data.m_static && this->rft_config == data.rft_config && this->restart_config == data.restart_config && this->snapshots == data.snapshots; } std::string Schedule::formatDate(std::time_t t) { const auto ts { TimeStampUTC(t) } ; return fmt::format("{:04d}-{:02d}-{:02d}" , ts.year(), ts.month(), ts.day()); } std::string Schedule::simulationDays(std::size_t currentStep) const { const double sim_time { this->m_static.m_unit_system.from_si(UnitSystem::measure::time, simTime(currentStep)) } ; return fmt::format("{} {}", sim_time, this->m_static.m_unit_system.name(UnitSystem::measure::time)); } namespace { // Duplicated from Well.cpp Connection::Order order_from_int(int int_value) { switch(int_value) { case 0: return Connection::Order::TRACK; case 1: return Connection::Order::DEPTH; case 2: return Connection::Order::INPUT; default: throw std::invalid_argument("Invalid integer value: " + std::to_string(int_value) + " encountered when determining connection ordering"); } } } void Schedule::load_rst(const RestartIO::RstState& rst_state, const EclipseGrid& grid, const FieldPropsManager& fp) { const auto report_step = rst_state.header.report_step - 1; double udq_undefined = 0; for (const auto& rst_group : rst_state.groups) { this->addGroup(rst_group.name, report_step); const auto& group = this->snapshots.back().groups.get( rst_group.name ); if (group.isProductionGroup()) { // Was originally at report_step + 1 this->snapshots.back().events().addEvent(ScheduleEvents::GROUP_PRODUCTION_UPDATE ); this->snapshots.back().wellgroup_events().addEvent(rst_group.name, ScheduleEvents::GROUP_PRODUCTION_UPDATE); } if (group.isInjectionGroup()) { // Was originally at report_step + 1 this->snapshots.back().events().addEvent(ScheduleEvents::GROUP_INJECTION_UPDATE ); this->snapshots.back().wellgroup_events().addEvent(rst_group.name, ScheduleEvents::GROUP_INJECTION_UPDATE); } } for (std::size_t group_index = 0; group_index < rst_state.groups.size(); group_index++) { const auto& rst_group = rst_state.groups[group_index]; if (rst_group.parent_group == 0) continue; if (rst_group.parent_group == rst_state.header.max_groups_in_field) continue; const auto& parent_group = rst_state.groups[rst_group.parent_group - 1]; this->addGroupToGroup(parent_group.name, rst_group.name, report_step); } for (const auto& rst_well : rst_state.wells) { Opm::Well well(rst_well, report_step, this->m_static.m_unit_system, udq_undefined); std::vector rst_connections; for (const auto& rst_conn : rst_well.connections) rst_connections.emplace_back(rst_conn, grid, fp); if (rst_well.segments.empty()) { Opm::WellConnections connections(order_from_int(rst_well.completion_ordering), rst_well.ij[0], rst_well.ij[1], rst_connections); well.updateConnections( std::make_shared( std::move(connections) ), grid, fp.get_int("PVTNUM")); } else { std::unordered_map rst_segments; for (const auto& rst_segment : rst_well.segments) { Opm::Segment segment(rst_segment); rst_segments.insert(std::make_pair(rst_segment.segment, std::move(segment))); } auto [connections, segments] = Compsegs::rstUpdate(rst_well, rst_connections, rst_segments); well.updateConnections( std::make_shared(std::move(connections)), grid, fp.get_int("PVTNUM")); well.updateSegments( std::make_shared(std::move(segments) )); } this->addWell(well); this->addWellToGroup(well.groupName(), well.name(), report_step); } this->snapshots[report_step + 1].update_tuning(rst_state.tuning); // Originally at report_step + 1 this->snapshots.back().events().addEvent( ScheduleEvents::TUNING_CHANGE ); { const auto& header = rst_state.header; bool time_interval = 0; GuideRateModel::Target target = GuideRateModel::Target::OIL; bool allow_increase = true; bool use_free_gas = false; if (GuideRateModel::rst_valid(time_interval, header.guide_rate_a, header.guide_rate_b, header.guide_rate_c, header.guide_rate_d, header.guide_rate_e, header.guide_rate_f, header.guide_rate_damping)) { auto guide_rate_model = GuideRateModel(time_interval, target, header.guide_rate_a, header.guide_rate_b, header.guide_rate_c, header.guide_rate_d, header.guide_rate_e, header.guide_rate_f, allow_increase, header.guide_rate_damping, use_free_gas); this->updateGuideRateModel(guide_rate_model, report_step); } } } std::shared_ptr Schedule::python() const { return this->m_static.m_python_handle; } const GasLiftOpt& Schedule::glo(std::size_t report_step) const { return this->snapshots[report_step].glo(); } namespace { /* The insane trickery here (thank you Stackoverflow!) is to be able to provide a simple templated comparison function template int not_equal(const T& arg1, const T& arg2, const std::string& msg); which will print arg1 and arg2 on stderr *if* T supports operator<<, otherwise it will just print the typename of T. */ template struct cmpx { int neq(const T& arg1, const T& arg2, const std::string& msg) { if (arg1 == arg2) return 0; std::cerr << "Error when comparing <" << typeid(arg1).name() << ">: " << msg << std::endl; return 1; } }; template struct cmpx { int neq(const T& arg1, const T& arg2, const std::string& msg) { if (arg1 == arg2) return 0; std::cerr << "Error when comparing: " << msg << " " << arg1 << " != " << arg2 << std::endl; return 1; } }; template int not_equal(const T& arg1, const T& arg2, const std::string& msg) { return cmpx().neq(arg1, arg2, msg); } template <> int not_equal(const double& arg1, const double& arg2, const std::string& msg) { if (Opm::cmp::scalar_equal(arg1, arg2)) return 0; std::cerr << "Error when comparing: " << msg << " " << arg1 << " != " << arg2 << std::endl; return 1; } template <> int not_equal(const UDAValue& arg1, const UDAValue& arg2, const std::string& msg) { if (arg1.is()) return not_equal( arg1.get(), arg2.get(), msg); else return not_equal( arg1.get(), arg2.get(), msg); } std::string well_msg(const std::string& well, const std::string& msg) { return "Well: " + well + " " + msg; } std::string well_segment_msg(const std::string& well, int segment_number, const std::string& msg) { return "Well: " + well + " Segment: " + std::to_string(segment_number) + " " + msg; } std::string well_connection_msg(const std::string& well, const Connection& conn, const std::string& msg) { return "Well: " + well + " Connection: " + std::to_string(conn.getI()) + ", " + std::to_string(conn.getJ()) + ", " + std::to_string(conn.getK()) + " " + msg; } } bool Schedule::cmp(const Schedule& sched1, const Schedule& sched2, std::size_t report_step) { int count = not_equal(sched1.wellNames(report_step), sched2.wellNames(report_step), "Wellnames"); if (count != 0) return false; { const auto& tm1 = sched1.getTimeMap(); const auto& tm2 = sched2.getTimeMap(); if (not_equal(tm1.size(), tm2.size(), "TimeMap: size()")) count += 1; for (auto& step_index = report_step; step_index < std::min(tm1.size(), tm2.size()) - 1; step_index++) { if (not_equal(tm1[step_index], tm2[step_index], "TimePoint[" + std::to_string(step_index) + "]")) count += 1; } } for (const auto& wname : sched1.wellNames(report_step)) { const auto& well1 = sched1.getWell(wname, report_step); const auto& well2 = sched2.getWell(wname, report_step); int well_count = 0; { const auto& connections2 = well2.getConnections(); const auto& connections1 = well1.getConnections(); well_count += not_equal( connections1.ordering(), connections2.ordering(), well_msg(well1.name(), "Connection: ordering")); for (std::size_t icon = 0; icon < connections1.size(); icon++) { const auto& conn1 = connections1[icon]; const auto& conn2 = connections2[icon]; well_count += not_equal( conn1.getI(), conn2.getI(), well_connection_msg(well1.name(), conn1, "I")); well_count += not_equal( conn1.getJ() , conn2.getJ() , well_connection_msg(well1.name(), conn1, "J")); well_count += not_equal( conn1.getK() , conn2.getK() , well_connection_msg(well1.name(), conn1, "K")); well_count += not_equal( conn1.state() , conn2.state(), well_connection_msg(well1.name(), conn1, "State")); well_count += not_equal( conn1.dir() , conn2.dir(), well_connection_msg(well1.name(), conn1, "dir")); well_count += not_equal( conn1.complnum() , conn2.complnum(), well_connection_msg(well1.name(), conn1, "complnum")); well_count += not_equal( conn1.segment() , conn2.segment(), well_connection_msg(well1.name(), conn1, "segment")); well_count += not_equal( conn1.kind() , conn2.kind(), well_connection_msg(well1.name(), conn1, "CFKind")); well_count += not_equal( conn1.sort_value(), conn2.sort_value(), well_connection_msg(well1.name(), conn1, "sort_value")); well_count += not_equal( conn1.CF(), conn2.CF(), well_connection_msg(well1.name(), conn1, "CF")); well_count += not_equal( conn1.Kh(), conn2.Kh(), well_connection_msg(well1.name(), conn1, "Kh")); well_count += not_equal( conn1.rw(), conn2.rw(), well_connection_msg(well1.name(), conn1, "rw")); well_count += not_equal( conn1.depth(), conn2.depth(), well_connection_msg(well1.name(), conn1, "depth")); //well_count += not_equal( conn1.r0(), conn2.r0(), well_connection_msg(well1.name(), conn1, "r0")); well_count += not_equal( conn1.skinFactor(), conn2.skinFactor(), well_connection_msg(well1.name(), conn1, "skinFactor")); } } if (not_equal(well1.isMultiSegment(), well2.isMultiSegment(), well_msg(well1.name(), "Is MSW"))) return false; if (well1.isMultiSegment()) { const auto& segments1 = well1.getSegments(); const auto& segments2 = well2.getSegments(); if (not_equal(segments1.size(), segments2.size(), "Segments: size")) return false; for (std::size_t iseg=0; iseg < segments1.size(); iseg++) { const auto& segment1 = segments1[iseg]; const auto& segment2 = segments2[iseg]; //const auto& segment2 = segments2.getFromSegmentNumber(segment1.segmentNumber()); well_count += not_equal(segment1.segmentNumber(), segment2.segmentNumber(), well_segment_msg(well1.name(), segment1.segmentNumber(), "segmentNumber")); well_count += not_equal(segment1.branchNumber(), segment2.branchNumber(), well_segment_msg(well1.name(), segment1.segmentNumber(), "branchNumber")); well_count += not_equal(segment1.outletSegment(), segment2.outletSegment(), well_segment_msg(well1.name(), segment1.segmentNumber(), "outletSegment")); well_count += not_equal(segment1.totalLength(), segment2.totalLength(), well_segment_msg(well1.name(), segment1.segmentNumber(), "totalLength")); well_count += not_equal(segment1.depth(), segment2.depth(), well_segment_msg(well1.name(), segment1.segmentNumber(), "depth")); well_count += not_equal(segment1.internalDiameter(), segment2.internalDiameter(), well_segment_msg(well1.name(), segment1.segmentNumber(), "internalDiameter")); well_count += not_equal(segment1.roughness(), segment2.roughness(), well_segment_msg(well1.name(), segment1.segmentNumber(), "roughness")); well_count += not_equal(segment1.crossArea(), segment2.crossArea(), well_segment_msg(well1.name(), segment1.segmentNumber(), "crossArea")); well_count += not_equal(segment1.volume(), segment2.volume(), well_segment_msg(well1.name(), segment1.segmentNumber(), "volume")); } } well_count += not_equal(well1.getStatus(), well2.getStatus(), well_msg(well1.name(), "status")); { const auto& prod1 = well1.getProductionProperties(); const auto& prod2 = well2.getProductionProperties(); well_count += not_equal(prod1.name, prod2.name , well_msg(well1.name(), "Prod: name")); well_count += not_equal(prod1.OilRate, prod2.OilRate, well_msg(well1.name(), "Prod: OilRate")); well_count += not_equal(prod1.GasRate, prod2.GasRate, well_msg(well1.name(), "Prod: GasRate")); well_count += not_equal(prod1.WaterRate, prod2.WaterRate, well_msg(well1.name(), "Prod: WaterRate")); well_count += not_equal(prod1.LiquidRate, prod2.LiquidRate, well_msg(well1.name(), "Prod: LiquidRate")); well_count += not_equal(prod1.ResVRate, prod2.ResVRate, well_msg(well1.name(), "Prod: ResVRate")); well_count += not_equal(prod1.BHPTarget, prod2.BHPTarget, well_msg(well1.name(), "Prod: BHPTarget")); well_count += not_equal(prod1.THPTarget, prod2.THPTarget, well_msg(well1.name(), "Prod: THPTarget")); well_count += not_equal(prod1.VFPTableNumber, prod2.VFPTableNumber, well_msg(well1.name(), "Prod: VFPTableNumber")); well_count += not_equal(prod1.ALQValue, prod2.ALQValue, well_msg(well1.name(), "Prod: ALQValue")); well_count += not_equal(prod1.predictionMode, prod2.predictionMode, well_msg(well1.name(), "Prod: predictionMode")); if (!prod1.predictionMode) { well_count += not_equal(prod1.bhp_hist_limit, prod2.bhp_hist_limit, well_msg(well1.name(), "Prod: bhp_hist_limit")); well_count += not_equal(prod1.thp_hist_limit, prod2.thp_hist_limit, well_msg(well1.name(), "Prod: thp_hist_limit")); well_count += not_equal(prod1.BHPH, prod2.BHPH, well_msg(well1.name(), "Prod: BHPH")); well_count += not_equal(prod1.THPH, prod2.THPH, well_msg(well1.name(), "Prod: THPH")); } well_count += not_equal(prod1.productionControls(), prod2.productionControls(), well_msg(well1.name(), "Prod: productionControls")); if (well1.getStatus() == Well::Status::OPEN) well_count += not_equal(prod1.controlMode, prod2.controlMode, well_msg(well1.name(), "Prod: controlMode")); well_count += not_equal(prod1.whistctl_cmode, prod2.whistctl_cmode, well_msg(well1.name(), "Prod: whistctl_cmode")); } { const auto& inj1 = well1.getInjectionProperties(); const auto& inj2 = well2.getInjectionProperties(); well_count += not_equal(inj1.name, inj2.name, well_msg(well1.name(), "Well::Inj: name")); well_count += not_equal(inj1.surfaceInjectionRate, inj2.surfaceInjectionRate, well_msg(well1.name(), "Well::Inj: surfaceInjectionRate")); well_count += not_equal(inj1.reservoirInjectionRate, inj2.reservoirInjectionRate, well_msg(well1.name(), "Well::Inj: reservoirInjectionRate")); well_count += not_equal(inj1.BHPTarget, inj2.BHPTarget, well_msg(well1.name(), "Well::Inj: BHPTarget")); well_count += not_equal(inj1.THPTarget, inj2.THPTarget, well_msg(well1.name(), "Well::Inj: THPTarget")); well_count += not_equal(inj1.bhp_hist_limit, inj2.bhp_hist_limit, well_msg(well1.name(), "Well::Inj: bhp_hist_limit")); well_count += not_equal(inj1.thp_hist_limit, inj2.thp_hist_limit, well_msg(well1.name(), "Well::Inj: thp_hist_limit")); well_count += not_equal(inj1.BHPH, inj2.BHPH, well_msg(well1.name(), "Well::Inj: BHPH")); well_count += not_equal(inj1.THPH, inj2.THPH, well_msg(well1.name(), "Well::Inj: THPH")); well_count += not_equal(inj1.VFPTableNumber, inj2.VFPTableNumber, well_msg(well1.name(), "Well::Inj: VFPTableNumber")); well_count += not_equal(inj1.predictionMode, inj2.predictionMode, well_msg(well1.name(), "Well::Inj: predictionMode")); well_count += not_equal(inj1.injectionControls, inj2.injectionControls, well_msg(well1.name(), "Well::Inj: injectionControls")); well_count += not_equal(inj1.injectorType, inj2.injectorType, well_msg(well1.name(), "Well::Inj: injectorType")); well_count += not_equal(inj1.controlMode, inj2.controlMode, well_msg(well1.name(), "Well::Inj: controlMode")); } { well_count += well2.firstTimeStep() > report_step; well_count += not_equal( well1.groupName(), well2.groupName(), well_msg(well1.name(), "Well: groupName")); well_count += not_equal( well1.getHeadI(), well2.getHeadI(), well_msg(well1.name(), "Well: getHeadI")); well_count += not_equal( well1.getHeadJ(), well2.getHeadJ(), well_msg(well1.name(), "Well: getHeadJ")); well_count += not_equal( well1.getRefDepth(), well2.getRefDepth(), well_msg(well1.name(), "Well: getRefDepth")); well_count += not_equal( well1.isMultiSegment(), well2.isMultiSegment() , well_msg(well1.name(), "Well: isMultiSegment")); well_count += not_equal( well1.isAvailableForGroupControl(), well2.isAvailableForGroupControl() , well_msg(well1.name(), "Well: isAvailableForGroupControl")); well_count += not_equal( well1.getGuideRate(), well2.getGuideRate(), well_msg(well1.name(), "Well: getGuideRate")); well_count += not_equal( well1.getGuideRatePhase(), well2.getGuideRatePhase(), well_msg(well1.name(), "Well: getGuideRatePhase")); well_count += not_equal( well1.getGuideRateScalingFactor(), well2.getGuideRateScalingFactor(), well_msg(well1.name(), "Well: getGuideRateScalingFactor")); well_count += not_equal( well1.predictionMode(), well2.predictionMode(), well_msg(well1.name(), "Well: predictionMode")); well_count += not_equal( well1.canOpen(), well2.canOpen(), well_msg(well1.name(), "Well: canOpen")); well_count += not_equal( well1.isProducer(), well2.isProducer(), well_msg(well1.name(), "Well: isProducer")); well_count += not_equal( well1.isInjector(), well2.isInjector(), well_msg(well1.name(), "Well: isInjector")); if (well1.isInjector()) well_count += not_equal( well1.injectorType(), well2.injectorType(), well_msg(well1.name(), "Well1: injectorType")); well_count += not_equal( well1.seqIndex(), well2.seqIndex(), well_msg(well1.name(), "Well: seqIndex")); well_count += not_equal( well1.getAutomaticShutIn(), well2.getAutomaticShutIn(), well_msg(well1.name(), "Well: getAutomaticShutIn")); well_count += not_equal( well1.getAllowCrossFlow(), well2.getAllowCrossFlow(), well_msg(well1.name(), "Well: getAllowCrossFlow")); well_count += not_equal( well1.getSolventFraction(), well2.getSolventFraction(), well_msg(well1.name(), "Well: getSolventFraction")); well_count += not_equal( well1.getStatus(), well2.getStatus(), well_msg(well1.name(), "Well: getStatus")); //well_count += not_equal( well1.getInjectionProperties(), well2.getInjectionProperties(), "Well: getInjectionProperties"); if (well1.isProducer()) well_count += not_equal( well1.getPreferredPhase(), well2.getPreferredPhase(), well_msg(well1.name(), "Well: getPreferredPhase")); well_count += not_equal( well1.getDrainageRadius(), well2.getDrainageRadius(), well_msg(well1.name(), "Well: getDrainageRadius")); well_count += not_equal( well1.getEfficiencyFactor(), well2.getEfficiencyFactor(), well_msg(well1.name(), "Well: getEfficiencyFactor")); } count += well_count; if (well_count > 0) std::cerr << std::endl; } return (count == 0); } const ScheduleState& Schedule::back() const { return this->snapshots.back(); } const ScheduleState& Schedule::operator[](std::size_t index) const { return this->snapshots.at(index); } std::vector::const_iterator Schedule::begin() const { return this->snapshots.begin(); } std::vector::const_iterator Schedule::end() const { return this->snapshots.end(); } void Schedule::create_first(const std::chrono::system_clock::time_point& start_time, const std::optional& end_time) { if (end_time.has_value()) this->snapshots.emplace_back( start_time, end_time.value() ); else this->snapshots.emplace_back(start_time); auto& sched_state = snapshots.back(); sched_state.update_nupcol( this->m_static.m_runspec.nupcol() ); sched_state.update_oilvap( OilVaporizationProperties( this->m_static.m_runspec.tabdims().getNumPVTTables() )); sched_state.update_message_limits( this->m_static.m_deck_message_limits ); sched_state.pavg.update( PAvg() ); sched_state.wtest_config.update( WellTestConfig() ); sched_state.gconsale.update( GConSale() ); sched_state.gconsump.update( GConSump() ); sched_state.wlist_manager.update( WListManager() ); sched_state.network.update( Network::ExtNetwork() ); sched_state.rpt_config.update( RPTConfig() ); sched_state.actions.update( Action::Actions() ); sched_state.udq_active.update( UDQActive() ); sched_state.well_order.update( NameOrder() ); sched_state.group_order.update( GroupOrder( this->m_static.m_runspec.wellDimensions().maxGroupsInField()) ); sched_state.udq.update( UDQConfig( this->m_static.m_runspec.udqParams() )); sched_state.glo.update( GasLiftOpt() ); sched_state.guide_rate.update( GuideRateConfig() ); this->addGroup("FIELD", 0); } void Schedule::create_next(const std::chrono::system_clock::time_point& start_time, const std::optional& end_time) { if (this->snapshots.empty()) this->create_first(start_time, end_time); else { const auto& last = this->snapshots.back(); if (end_time.has_value()) this->snapshots.emplace_back( last, start_time, end_time.value() ); else this->snapshots.emplace_back( last, start_time ); } } void Schedule::create_next(const ScheduleBlock& block) { const auto& start_time = block.start_time(); const auto& end_time = block.end_time(); this->create_next(start_time, end_time); } }