8 #ifndef OOPS_RUNS_ENSEMBLEAPPLICATION_H_
9 #define OOPS_RUNS_ENSEMBLEAPPLICATION_H_
14 #include "eckit/config/LocalConfiguration.h"
15 #include "eckit/config/YAMLConfiguration.h"
16 #include "eckit/exception/Exceptions.h"
17 #include "eckit/mpi/Comm.h"
20 #include "oops/util/abor1_cpp.h"
21 #include "oops/util/Logger.h"
25 template <
typename APP>
34 int execute(
const eckit::Configuration & fullConfig)
const {
36 std::vector<std::string> listConf;
37 fullConfig.get(
"files", listConf);
38 Log::info() <<
"EnsembleApplication yaml files:" << listConf << std::endl;
41 const int nmembers = listConf.size();
42 const int ntasks = this->
getComm().size();
43 const int mytask = this->
getComm().rank();
44 const int tasks_per_member = ntasks / nmembers;
45 const int mymember = mytask / tasks_per_member + 1;
47 Log::info() <<
"Running " << nmembers <<
" EnsembleApplication members handled by "
48 << ntasks <<
" total MPI tasks and "
49 << tasks_per_member <<
" MPI tasks per member." << std::endl;
51 ASSERT(ntasks%nmembers == 0);
54 std::string commNameStr =
"comm_member_" + std::to_string(mymember);
55 char const *commName = commNameStr.c_str();
56 eckit::mpi::Comm & commMember = this->
getComm().split(mymember, commName);
59 eckit::PathName confPath = listConf[mymember-1];
60 eckit::YAMLConfiguration memberConf(confPath);
62 Log::debug() <<
"EnsembleApplication config for member " << mymember <<
": "
63 << memberConf << std::endl;
65 APP ensapp(commMember);
66 return ensapp.execute(memberConf);
71 return "oops::EnsembleApplication<>";
const eckit::mpi::Comm & getComm() const
virtual ~EnsembleApplication()
EnsembleApplication(const eckit::mpi::Comm &comm=oops::mpi::world())
int execute(const eckit::Configuration &fullConfig) const
std::string appname() const
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
The namespace for the main oops code.