OOPS
HofX4Dhack.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-2020 UCAR
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  */
7 
8 #ifndef OOPS_RUNS_HOFX4DHACK_H_
9 #define OOPS_RUNS_HOFX4DHACK_H_
10 
11 #include <algorithm>
12 #include <memory>
13 #include <string>
14 #include <vector>
15 
16 #include "eckit/config/LocalConfiguration.h"
17 #include "eckit/exception/Exceptions.h"
18 
20 #include "oops/base/Geometry.h"
22 #include "oops/base/Observations.h"
23 #include "oops/base/ObsSpaces.h"
24 #include "oops/base/State.h"
25 #include "oops/base/Variables.h"
28 #include "oops/interface/GeoVaLs.h"
31 #include "oops/mpi/mpi.h"
32 #include "oops/runs/Application.h"
33 #include "oops/util/ConfigFunctions.h"
34 #include "oops/util/DateTime.h"
35 #include "oops/util/Duration.h"
36 #include "oops/util/Logger.h"
37 
38 // ########################################################################
39 // ########################################################################
40 // ### This file is a temporary hack until all models can work with a ###
41 // ### single variable transform in GetValues. Then it will be removed. ###
42 // ########################################################################
43 // ########################################################################
44 
45 namespace oops {
46 
47 template <typename MODEL, typename OBS> class HofX4Dhack : public Application {
58 
59  typedef std::vector<std::unique_ptr<GeoVaLs_>> GeoVaLsVec_;
60  typedef std::vector<std::unique_ptr<Locations_>> LocationsVec_;
61  typedef std::vector<Variables> VariablesVec_;
62 
63  public:
64 // -----------------------------------------------------------------------------
65  explicit HofX4Dhack(const eckit::mpi::Comm & comm = oops::mpi::world()) : Application(comm) {
66  instantiateObsFilterFactory<OBS>();
67  instantiateVariableChangeFactory<MODEL>();
68  }
69 // -----------------------------------------------------------------------------
70  virtual ~HofX4Dhack() {}
71 // -----------------------------------------------------------------------------
72  int execute(const eckit::Configuration & fullConfig) const {
73  // Setup observation window
74  const util::Duration winlen(fullConfig.getString("window length"));
75  const util::DateTime winbgn(fullConfig.getString("window begin"));
76  const util::DateTime winend(winbgn + winlen);
77  Log::info() << "Observation window from " << winbgn << " to " << winend << std::endl;
78 
79  // Get information about states
80  std::vector<eckit::LocalConfiguration> statesConf;
81  if (fullConfig.has("states")) {
82  statesConf = fullConfig.getSubConfigurations("states");
83  } else {
84  if (fullConfig.has("state")) {
85  statesConf[0] = eckit::LocalConfiguration(fullConfig, "state");
86  } else {
87  eckit::BadParameter("Error in background state(s) configuration");
88  }
89  }
90  size_t nsubwin = statesConf.size();
91 
92  size_t ntasks = this->getComm().size();
93  size_t myrank = this->getComm().rank();
94  ASSERT(ntasks % nsubwin == 0);
95  size_t ntaskpslot = ntasks / nsubwin;
96  size_t mysubwin = myrank / ntaskpslot;
97  Log::debug() << "ntasks = " << ntasks << " nsubwin = " << nsubwin << std::endl;
98 
99  // Define local sub-window
100  util::Duration subWinLength = winlen;
101  if (nsubwin > 1) subWinLength = winlen / (nsubwin - 1);
102  Log::debug() << "Task " << mysubwin << " subWinLength " << subWinLength << std::endl;
103  util::DateTime subWinTime = winbgn + mysubwin * subWinLength;
104  util::DateTime subWinBegin = subWinTime - subWinLength/2;
105  util::DateTime subWinEnd = subWinTime + subWinLength/2;
106  if (mysubwin == 0) subWinBegin = subWinTime;
107  if (mysubwin == nsubwin - 1) subWinEnd = subWinTime;
108  ASSERT(subWinBegin >= winbgn);
109  ASSERT(subWinEnd <= winend);
110  Log::debug() << "Task " << mysubwin << " Obs times " << subWinTime
111  << ", " << subWinBegin << ", " << subWinEnd << std::endl;
112  // Create a communicator for same sub-window, to be used for communications in space
113  std::string sgeom = "comm_geom_" + std::to_string(mysubwin);
114  char const *geomName = sgeom.c_str();
115  eckit::mpi::Comm * commSpace = &this->getComm().split(mysubwin, geomName);
116  // Create a communicator for same local area, to be used for communications in time
117  size_t myarea = commSpace->rank();
118  std::string stime = "comm_time_" + std::to_string(myarea);
119  char const *timeName = stime.c_str();
120  eckit::mpi::Comm * commTime = &this->getComm().split(myarea, timeName);
121  ASSERT(commTime->size() == nsubwin);
122 
123  // Setup geometry
124  const eckit::LocalConfiguration geometryConfig(fullConfig, "geometry");
125  const Geometry_ geometry(geometryConfig, *commSpace);
126 
127  // Setup states for H(x)
128  Log::info() << "State configuration is:" << statesConf[mysubwin] << std::endl;
129  State_ xx(geometry, statesConf[mysubwin]);
130  Log::test() << "State: " << xx << std::endl;
131  Log::debug() << "Task " << mysubwin << " State time " << xx.validTime() << std::endl;
132  ASSERT(xx.validTime() == subWinTime);
133 
134  // Setup observations
135  const eckit::LocalConfiguration obsConfig(fullConfig, "observations");
136  ObsSpaces_ obspaces(obsConfig, *commSpace, subWinBegin, subWinEnd, *commTime);
137  ObsAux_ obsaux(obspaces, obsConfig);
138  CalcHofX_ hofx(obspaces, obsConfig);
139  hofx.initialize(obsaux);
140 
141  // fill in GeoVaLs
142  GeoVaLsVec_ geovals;
143  const LocationsVec_ & locations = hofx.locations();
144  const VariablesVec_ & vars = hofx.requiredVars();
145  Log::debug() << "HofX4Dhack: Required hofx size = " << hofx.requiredVars().size() << std::endl;
146 
147  Variables geovars;
148  Log::debug() << "HofX4Dhack: Required vars size = " << vars.size() << std::endl;
149  for (size_t jj = 0; jj < vars.size(); ++jj) {
150  Log::debug() << "HofX4Dhack: Required vars:" << vars[jj] << std::endl;
151  geovars += vars[jj];
152  }
153  Log::debug() << "HofX4Dhack: Required variables:" << geovars << std::endl;
154  eckit::LocalConfiguration chvarconf; // empty for now
155  ChangeVariables_ chvar(chvarconf, geometry, xx.variables(), geovars);
156 
157  State_ zz(geometry, geovars, xx.validTime());
158  chvar.changeVar(xx, zz);
159 
160  std::vector<eckit::LocalConfiguration> getValuesConfig =
161  util::vectoriseAndFilter(obsConfig, "get values");
162 
163  // loop over all observation types
164  for (size_t jj = 0; jj < obspaces.size(); ++jj) {
165  GetValues_ getvals(geometry, *locations[jj], getValuesConfig[jj]);
166  geovals.emplace_back(new GeoVaLs_(*locations[jj], vars[jj],
167  geometry.variableSizes(vars[jj])));
168  getvals.fillGeoVaLs(zz, subWinBegin, subWinEnd, *geovals[jj]);
169  }
170 
171  // Compute H(x) on filled in geovals and run the filters
172  Observations_ yobs = hofx.compute(geovals);
173  hofx.saveQcFlags("EffectiveQC");
174  hofx.saveObsErrors("EffectiveError");
175 
176  // Save H(x)
177  Log::test() << "H(x): " << std::endl << yobs << "End H(x)" << std::endl;
178  Log::info() << "H(x): " << std::endl << yobs << "End H(x)" << std::endl;
179  yobs.save("hofx");
180  obspaces.save();
181 
182  return 0;
183  }
184 // -----------------------------------------------------------------------------
185  private:
186  std::string appname() const {
187  return "oops::HofX4Dhack<" + MODEL::name() + ", " + OBS::name() + ">";
188  }
189 // -----------------------------------------------------------------------------
190 };
191 
192 } // namespace oops
193 
194 #endif // OOPS_RUNS_HOFX4DHACK_H_
const eckit::mpi::Comm & getComm() const
Definition: Application.h:36
Computes observation operator (from GeoVaLs), applies bias correction and runs QC filters.
Definition: CalcHofX.h:50
void saveObsErrors(const std::string &) const
save obs error variances (modified in QC) to ObsSpaces
Definition: CalcHofX.h:225
void initialize(const ObsAuxCtrls_ &, const int iteration=0)
Initializes variables, obs bias, obs filters (could be different for different iterations.
Definition: CalcHofX.h:147
const LocationsVec_ & locations() const
accessor to the locations
Definition: CalcHofX.h:82
Observations_ compute(const GeoVaLsVec_ &)
Computes H(x) from the filled in GeoVaLs.
Definition: CalcHofX.h:170
void saveQcFlags(const std::string &) const
save QC flags to ObsSpaces
Definition: CalcHofX.h:216
const VariablesVec_ & requiredVars() const
accessor to variables required from the model
Definition: CalcHofX.h:84
Encapsulates the nonlinear variable change There should not be a factory for ChangeVariable,...
void changeVar(const State_ &xin, State_ &xout) const
change variable from state xin to xout
Geometry class used in oops; subclass of interface class interface::Geometry.
Gets values from model State to observation locations (fills GeoVaLs)
void fillGeoVaLs(const State_ &, const util::DateTime &, const util::DateTime &, GeoVaLs_ &) const
Get state values at observation locations.
Geometry< MODEL > Geometry_
Definition: HofX4Dhack.h:49
Locations< OBS > Locations_
Definition: HofX4Dhack.h:52
HofX4Dhack(const eckit::mpi::Comm &comm=oops::mpi::world())
Definition: HofX4Dhack.h:65
ObsAuxControls< OBS > ObsAux_
Definition: HofX4Dhack.h:53
Observations< OBS > Observations_
Definition: HofX4Dhack.h:54
ObsSpaces< OBS > ObsSpaces_
Definition: HofX4Dhack.h:55
std::vector< Variables > VariablesVec_
Definition: HofX4Dhack.h:61
GetValues< MODEL, OBS > GetValues_
Definition: HofX4Dhack.h:51
GeoVaLs< OBS > GeoVaLs_
Definition: HofX4Dhack.h:50
ChangeVariables< MODEL > ChangeVariables_
Definition: HofX4Dhack.h:57
virtual ~HofX4Dhack()
Definition: HofX4Dhack.h:70
std::vector< std::unique_ptr< GeoVaLs_ > > GeoVaLsVec_
Definition: HofX4Dhack.h:59
std::string appname() const
Definition: HofX4Dhack.h:186
State< MODEL > State_
Definition: HofX4Dhack.h:56
std::vector< std::unique_ptr< Locations_ > > LocationsVec_
Definition: HofX4Dhack.h:60
CalcHofX< OBS > CalcHofX_
Definition: HofX4Dhack.h:48
int execute(const eckit::Configuration &fullConfig) const
Definition: HofX4Dhack.h:72
Locations of observations for observation operator.
Holds a vector of ObsAuxControl.
void save() const
Save files.
Definition: ObsSpaces.h:114
std::size_t size() const
Access.
Definition: ObsSpaces.h:61
Observations Class.
Definition: Observations.h:35
void save(const std::string &) const
Save/read observations values.
Definition: Observations.h:141
State class used in oops; subclass of interface class interface::State.
size_t size() const
std::vector< size_t > variableSizes(const Variables &) const
const util::DateTime validTime() const
Accessor to the time of this State.
const Variables & variables() const
Accessor to variables associated with this State.
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
Definition: oops/mpi/mpi.cc:84
The namespace for the main oops code.