OOPS
LocalEnsembleDA.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2019-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_LOCALENSEMBLEDA_H_
9 #define OOPS_RUNS_LOCALENSEMBLEDA_H_
10 
11 #include <cmath>
12 #include <memory>
13 #include <string>
14 #include <vector>
15 
16 #include "eckit/config/LocalConfiguration.h"
19 #include "oops/base/Departures.h"
20 #include "oops/base/Geometry.h"
21 #include "oops/base/Increment.h"
24 #include "oops/base/Observations.h"
25 #include "oops/base/ObsSpaces.h"
26 #include "oops/base/State4D.h"
31 #include "oops/mpi/mpi.h"
32 #include "oops/runs/Application.h"
33 #include "oops/util/DateTime.h"
34 #include "oops/util/Duration.h"
35 #include "oops/util/Logger.h"
36 
37 
38 namespace oops {
39 
40 /// \brief Application for local ensemble data assimilation
41 template <typename MODEL, typename OBS> class LocalEnsembleDA : public Application {
52 
53  public:
54 // -----------------------------------------------------------------------------
55 
56  explicit LocalEnsembleDA(const eckit::mpi::Comm & comm = oops::mpi::world()) : Application(comm) {
57  instantiateLocalEnsembleSolverFactory<MODEL, OBS>();
58  instantiateObsErrorFactory<OBS>();
59  instantiateObsFilterFactory<OBS>();
60  instantiateVariableChangeFactory<MODEL>();
61  }
62 
63 // -----------------------------------------------------------------------------
64 
65  virtual ~LocalEnsembleDA() = default;
66 
67 // -----------------------------------------------------------------------------
68 
69  int execute(const eckit::Configuration & fullConfig) const {
70  // Setup observation window
71  const util::DateTime winbgn(fullConfig.getString("window begin"));
72  const util::Duration winlen(fullConfig.getString("window length"));
73  const util::DateTime winend(winbgn + winlen);
74  Log::info() << "Observation window from " << winbgn << " to " << winend << std::endl;
75 
76  // Setup geometry
77  const eckit::LocalConfiguration geometryConfig(fullConfig, "geometry");
78  const Geometry_ geometry(geometryConfig, this->getComm());
79 
80  // Get observations configuration
81  eckit::LocalConfiguration obsConfig(fullConfig, "observations");
82  // Get driver configuration
83  const eckit::LocalConfiguration driverConfig(fullConfig, "driver");
84 
85  // if any of the obs. spaces uses Halo distribution it will need to know the geometry
86  // of the local grid on this PE
87  bool update_obsconfig = driverConfig.getBool("update obs config with geometry info", false);
88  if (update_obsconfig) updateConfigWithPatchGeometry(geometry, obsConfig);
89 
90  // Setup observations
91  const eckit::mpi::Comm & time = oops::mpi::myself();
92  ObsSpaces_ obsdb(obsConfig, this->getComm(), winbgn, winend, time);
93  Observations_ yobs(obsdb, "ObsValue");
94 
95  // Get background configurations
96  const eckit::LocalConfiguration bgConfig(fullConfig, "background");
97 
98  // Read all ensemble members and compute the mean
99  StateEnsemble4D_ ens_xx(geometry, bgConfig);
100  const size_t nens = ens_xx.size();
101  const Variables statevars = ens_xx.variables();
102  State4D_ bkg_mean = ens_xx.mean();
103 
104  // set up solver
105  std::unique_ptr<LocalSolver_> solver =
106  LocalEnsembleSolverFactory<MODEL, OBS>::create(obsdb, geometry, fullConfig,
107  nens, bkg_mean);
108  // test prints for the prior ensemble
109  bool do_test_prints = driverConfig.getBool("do test prints", true);
110  if (do_test_prints) {
111  for (size_t jj = 0; jj < nens; ++jj) {
112  Log::test() << "Initial state for member " << jj+1 << ":" << ens_xx[jj] << std::endl;
113  }
114  }
115 
116  // compute H(x)
117  bool readFromDisk = driverConfig.getBool("read HX from disk", false);
118  Observations_ yb_mean = solver->computeHofX(ens_xx, 0, readFromDisk);
119  Log::test() << "H(x) ensemble background mean: " << std::endl << yb_mean << std::endl;
120 
121  Departures_ ombg(yobs - yb_mean);
122  ombg.save("ombg");
123  Log::test() << "background y - H(x): " << std::endl << ombg << std::endl;
124 
125  // quit early if running in observer-only mode
126  bool observerOnly = driverConfig.getBool("run as observer only", false);
127  if (observerOnly) {
128  obsdb.save();
129  return 0;
130  }
131 
132  // print background mean
133  if (do_test_prints) {
134  Log::test() << "Background mean :" << bkg_mean << std::endl;
135  }
136 
137  // calculate background ensemble perturbations
138  IncrementEnsemble4D_ bkg_pert(ens_xx, bkg_mean, statevars);
139 
140  // initialize empty analysis perturbations
141  IncrementEnsemble4D_ ana_pert(geometry, statevars, ens_xx[0].validTimes(), bkg_pert.size());
142 
143  // run the solver at each gridpoint
144  Log::info() << "Beginning core local solver..." << std::endl;
145  for (GeometryIterator_ i = geometry.begin(); i != geometry.end(); ++i) {
146  solver->measurementUpdate(bkg_pert, i, ana_pert);
147  }
148  Log::info() << "Local solver completed." << std::endl;
149 
150  // wait all tasks to finish their solution, so the timing for functions below reports
151  // time which truly used (not from mpi_wait(), as all tasks need to sync before write).
152  oops::mpi::world().barrier();
153 
154  // calculate final analysis states
155  for (size_t jj = 0; jj < nens; ++jj) {
156  ens_xx[jj] = bkg_mean;
157  ens_xx[jj] += ana_pert[jj];
158  }
159 
160  // save the posterior mean and ensemble first (since they are needed for the next cycle)
161  // save the posterior mean
162  State4D_ ana_mean = ens_xx.mean(); // calculate analysis mean
163  if (do_test_prints) {
164  Log::test() << "Analysis mean :" << ana_mean << std::endl;
165  }
166  bool save_xamean = driverConfig.getBool("save posterior mean", false);
167  if (save_xamean) {
168  eckit::LocalConfiguration outConfig(fullConfig, "output");
169  outConfig.set("member", 0);
170  ana_mean.write(outConfig);
171  }
172 
173  // save the posterior ensemble
174  bool save_ens = driverConfig.getBool("save posterior ensemble", true);
175  if (save_ens) {
176  size_t mymember;
177  for (size_t jj=0; jj < nens; ++jj) {
178  mymember = jj+1;
179  eckit::LocalConfiguration outConfig(fullConfig, "output");
180  outConfig.set("member", mymember);
181  ens_xx[jj].write(outConfig);
182  }
183  }
184 
185  // below is the diagnostic output -----------------------------
186  // save the background mean
187  bool save_xbmean = driverConfig.getBool("save prior mean", false);
188  if (save_xbmean) {
189  eckit::LocalConfiguration outConfig(fullConfig, "output mean prior");
190  outConfig.set("member", 0);
191  bkg_mean.write(outConfig);
192  }
193 
194  // save the analysis increment
195  bool save_mean_increment = driverConfig.getBool("save posterior mean increment", false);
196  if (save_mean_increment) {
197  eckit::LocalConfiguration outConfig(fullConfig, "output increment");
198  outConfig.set("member", 0);
199  for (size_t itime = 0; itime < ana_mean.size(); ++itime) {
200  Increment_ ana_increment(ana_pert[0][itime], false);
201  ana_increment.diff(ana_mean[itime], bkg_mean[itime]);
202  ana_increment.write(outConfig);
203  if (do_test_prints) {
204  Log::test() << "Analysis mean increment :" << ana_increment << std::endl;
205  }
206  }
207  }
208 
209  // save the prior variance
210  bool save_variance = driverConfig.getBool("save prior variance", false);
211  if (save_variance) {
212  eckit::LocalConfiguration outConfig(fullConfig, "output variance prior");
213  outConfig.set("member", 0);
214  std::string strOut("Forecast variance :");
215  saveVariance(outConfig, bkg_pert, do_test_prints, strOut);
216  }
217 
218  // save the posterior variance
219  save_variance = driverConfig.getBool("save posterior variance", false);
220  if (save_variance) {
221  eckit::LocalConfiguration outConfig(fullConfig, "output variance posterior");
222  outConfig.set("member", 0);
223  std::string strOut("Analysis variance :");
224  saveVariance(outConfig, ana_pert, do_test_prints, strOut);
225  }
226 
227  // posterior observer
228  // note: if H(X) is read from file, it might have used different time slots for observation
229  // than LETKF background/analysis perturbations.
230  // hence one might not expect that oman and omaf are comparable
231  bool do_posterior_observer = driverConfig.getBool("do posterior observer", true);
232  if (do_posterior_observer) {
233  Observations_ ya_mean = solver->computeHofX(ens_xx, 1, false);
234  Log::test() << "H(x) ensemble analysis mean: " << std::endl << ya_mean << std::endl;
235 
236  // calculate analysis obs departures
237  Departures_ oman(yobs - ya_mean);
238  oman.save("oman");
239  Log::test() << "analysis y - H(x): " << std::endl << oman << std::endl;
240 
241  // display overall background/analysis RMS stats
242  Log::test() << "ombg RMS: " << ombg.rms() << std::endl
243  << "oman RMS: " << oman.rms() << std::endl;
244  }
245  obsdb.save();
246 
247  return 0;
248  }
249 
250 // -----------------------------------------------------------------------------
251 
252  private:
253  std::string appname() const {
254  return "oops::LocalEnsembleDA<" + MODEL::name() + ", " + OBS::name() + ">";
255  }
256 
258  eckit::LocalConfiguration & obsConfig) const {
259  std::vector<double> patchCenter(2, 0.0);
260  double patchRadius = 0.0;
261 
262  // since Halo distribution is only implemented in ioda, we can assume that
263  // x is lon and y is lat on Earth. then we need to compute average of x on a circle
264  eckit::geometry::Point2 gptmp;
265  const double radius_earth = 6.371e6;
266  const double deg2rad = 3.14159265/180.0;
267 
268  // compute patch center
269  double sinXmean = 0;
270  double cosXmean = 0;
271  double ymean = 0;
272  int n = 0;
273  for (GeometryIterator_ i = geometry.begin(); i != geometry.end(); ++i) {
274  gptmp = *i;
275  cosXmean += cos(gptmp.x()*deg2rad);
276  sinXmean += sin(gptmp.x()*deg2rad);
277  ymean += gptmp.y();
278  ++n;
279  }
280  cosXmean = cosXmean/static_cast<double>(n);
281  sinXmean = sinXmean/static_cast<double>(n);
282  patchCenter[0] = atan2(sinXmean, cosXmean)/deg2rad;
283  patchCenter[1] = ymean/static_cast<double>(n);
284 
285  // compute radius
286  eckit::geometry::Point2 center(patchCenter[0], patchCenter[1]);
287  patchRadius = 0;
288  for (GeometryIterator_ i = geometry.begin(); i != geometry.end(); ++i) {
289  double dist = eckit::geometry::Sphere::distance(radius_earth, center, *i);
290  patchRadius = fmax(patchRadius, dist);
291  }
292  Log::debug() << "patch center=" << patchCenter
293  << " patch radius=" << patchRadius << std::endl;
294 
295  // update observations configs with information on patch center and radius
296  std::vector<eckit::LocalConfiguration> obsConfigs = obsConfig.getSubConfigurations();
297  for (auto & conf : obsConfigs) {
298  conf.set("obs space.center", patchCenter);
299  conf.set("obs space.radius", patchRadius);
300  }
301  eckit::LocalConfiguration tmp;
302  tmp.set("observations", obsConfigs);
303  obsConfig = tmp.getSubConfiguration("observations");
304  }
305 
306  void saveVariance(const eckit::LocalConfiguration & outConfig, const IncrementEnsemble4D_ & perts,
307  const bool do_test_prints, const std::string & strOut) const {
308  // save and optionaly print varaince of an IncrementEnsemble4D_ object
309  size_t nens = perts.size();
310  const double nc = 1.0/(static_cast<double>(nens) - 1.0);
311  for (size_t itime = 0; itime < perts[0].size(); ++itime) {
312  Increment_ var(perts[0][itime], false);
313  for (size_t iens = 0; iens < nens; ++iens) {
314  Increment_ tmp(perts[iens][itime], true);
315  tmp.schur_product_with(perts[iens][itime]);
316  var += tmp;
317  }
318  var *= nc;
319  var.write(outConfig);
320  if (do_test_prints) {
321  Log::test() << strOut << var << std::endl;
322  }
323  }
324  }
325 
326 // -----------------------------------------------------------------------------
327 };
328 
329 } // namespace oops
330 #endif // OOPS_RUNS_LOCALENSEMBLEDA_H_
const eckit::mpi::Comm & getComm() const
Definition: Application.h:36
Difference between two observation vectors.
Definition: Departures.h:44
void save(const std::string &) const
Save departures values.
Definition: Departures.h:249
double rms() const
Definition: Departures.h:193
Geometry class used in oops; subclass of interface class interface::Geometry.
Ensemble of 4D increments.
size_t size() const
Accessors.
Increment class used in oops.
Application for local ensemble data assimilation.
Departures< OBS > Departures_
virtual ~LocalEnsembleDA()=default
StateEnsemble4D< MODEL > StateEnsemble4D_
int execute(const eckit::Configuration &fullConfig) const
IncrementEnsemble4D< MODEL > IncrementEnsemble4D_
GeometryIterator< MODEL > GeometryIterator_
Observations< OBS > Observations_
void saveVariance(const eckit::LocalConfiguration &outConfig, const IncrementEnsemble4D_ &perts, const bool do_test_prints, const std::string &strOut) const
void updateConfigWithPatchGeometry(const Geometry_ &geometry, eckit::LocalConfiguration &obsConfig) const
LocalEnsembleDA(const eckit::mpi::Comm &comm=oops::mpi::world())
ObsSpaces< OBS > ObsSpaces_
std::string appname() const
Increment< MODEL > Increment_
LocalEnsembleSolver< MODEL, OBS > LocalSolver_
Geometry< MODEL > Geometry_
State4D< MODEL > State4D_
static std::unique_ptr< LocalEnsembleSolver< MODEL, OBS > > create(ObsSpaces_ &, const Geometry_ &, const eckit::Configuration &, size_t, const State4D_ &)
Base class for LETKF-type solvers.
void save() const
Save files.
Definition: ObsSpaces.h:114
Observations Class.
Definition: Observations.h:35
Four dimensional state (vector of 3D States)
Definition: State4D.h:29
size_t size() const
Get 3D model state.
Definition: State4D.h:43
void write(const eckit::Configuration &) const
I/O.
Definition: State4D.h:87
Ensemble of 4D states.
const Variables & variables() const
Information.
State4D_ mean() const
calculate ensemble mean
unsigned int size() const
Accessors.
GeometryIterator_ end() const
Iterator to the past-the-end gridpoint of Geometry (only used in LocalEnsembleDA)
GeometryIterator_ begin() const
Iterator to the first gridpoint of Geometry (only used in LocalEnsembleDA)
void schur_product_with(const Increment &other)
Compute Schur product of this Increment with other, assign to this Increment.
void diff(const State_ &state1, const State_ &state2)
Set this Increment to be difference between state1 and state2.
void write(const eckit::Configuration &) const
Write this Increment out to file.
const eckit::mpi::Comm & myself()
Default communicator with each MPI task by itself.
Definition: oops/mpi/mpi.cc:90
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.