OOPS
HybridGain.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2021 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_HYBRIDGAIN_H_
9 #define OOPS_RUNS_HYBRIDGAIN_H_
10 
11 #include <memory>
12 #include <string>
13 #include <vector>
14 
15 
16 #include "eckit/config/LocalConfiguration.h"
17 #include "oops/base/Geometry.h"
18 #include "oops/base/Increment.h"
19 #include "oops/base/State.h"
20 #include "oops/base/Variables.h"
21 #include "oops/mpi/mpi.h"
22 #include "oops/runs/Application.h"
23 #include "oops/util/abor1_cpp.h"
24 #include "oops/util/DateTime.h"
25 
26 namespace oops {
27 
28 template <typename MODEL> class HybridGain : public Application {
32 
33  public:
34  // -----------------------------------------------------------------------------
35  explicit HybridGain(const eckit::mpi::Comm & comm = oops::mpi::world()) : Application(comm) {}
36  // -----------------------------------------------------------------------------
37  virtual ~HybridGain() {}
38  // -----------------------------------------------------------------------------
39  int execute(const eckit::Configuration & fullConfig) const {
40  // Setup Geometry
41  const eckit::LocalConfiguration resolConfig(fullConfig, "geometry");
42  const Geometry_ resol(resolConfig, this->getComm());
43 
44  // Read averaging weights
45  double alphaControl = fullConfig.getDouble("hybrid weights.control");
46  double alphaEnsemble = fullConfig.getDouble("hybrid weights.ensemble");
47 
48  // Read hybrid type
49  std::string hybridType = fullConfig.getString("hybrid type", "average analysis");
50 
51  // Get control state
52  const eckit::LocalConfiguration bkgConfig(fullConfig, "control");
53  State_ xaControl(resol, bkgConfig);
54  Log::test() << "Control: " << std::endl << xaControl << std::endl;
55  const Variables vars = xaControl.variables();
56 
57  // Get posterior ens mean
58  const eckit::LocalConfiguration emeanConfig(fullConfig, "ensemble mean posterior");
59  State_ xaEmeanPost(resol, emeanConfig);
60  Log::test() << "Ensemble mean posterior: " << std::endl << xaEmeanPost << std::endl;
61 
62  // Compute new center
63  State_ xNewCenter(resol, vars, xaControl.validTime());
64  if (hybridType == "average analysis") {
65  // using average of analysis (following Bonavita)
66  // xa_hybrid = a1*xa1 + a2*xa2
67  // a1+a2 have to equal to 1
68 
69  ASSERT(alphaControl + alphaEnsemble == 1.0);
70  xNewCenter.zero();
71  xNewCenter.accumul(alphaControl, xaControl);
72  xNewCenter.accumul(alphaEnsemble, xaEmeanPost);
73  } else if (hybridType == "average increment") {
74  // using average of analysis incerments (following Whitaker)
75  // xa_hybrid = xf_prior + a1*xinc1 + a2*xinc2
76  // Note: a1+a2 no longer need to add to one
77 
78  // Get prior ens mean
79  const eckit::LocalConfiguration emeanConfigPrior(fullConfig, "ensemble mean prior");
80  State_ xfEmeanPrior(resol, emeanConfig);
81  Log::test() << "Ensemble mean prior: " << std::endl << xfEmeanPrior << std::endl;
82  // compute ensemble mean increment
83  Increment_ pertEns(resol, vars, xaControl.validTime());
84  pertEns.diff(xaEmeanPost, xfEmeanPrior);
85  pertEns *= alphaEnsemble;
86  // compute control increment
87  Increment_ pertControl(resol, vars, xaControl.validTime());
88  pertControl.diff(xaControl, xfEmeanPrior);
89  pertControl *= alphaControl;
90  // compute hybrid posterior
91  xNewCenter = xfEmeanPrior;
92  xNewCenter += pertEns;
93  xNewCenter += pertControl;
94  } else {
95  ABORT("Unknown hybrid gain type: " + hybridType);
96  }
97  // Output new center
98  eckit::LocalConfiguration centerOut(fullConfig, "recentered output");
99  centerOut.set("member", static_cast<int>(0) );
100  xNewCenter.write(centerOut);
101  Log::test() << "new center : " << xNewCenter << std::endl;
102 
103  // Get ensemble configuration
104  std::vector<eckit::LocalConfiguration> ensConfig;
105  fullConfig.get("ensemble", ensConfig);
106  unsigned nens = ensConfig.size();
107 
108  // Recenter ensemble around new center and save
109  for (unsigned jj = 0; jj < nens; ++jj) {
110  State_ x(resol, ensConfig[jj]);
111  Increment_ pert(resol, vars, x.validTime());
112  pert.diff(x, xaEmeanPost);
113  x = xNewCenter;
114  x += pert;
115 
116  // Save recentered member
117  eckit::LocalConfiguration recenterout(fullConfig, "recentered output");
118  recenterout.set("member", static_cast<int>(jj+1) );
119  x.write(recenterout);
120  Log::test() << "Recentered member " << jj << " : " << x << std::endl;
121  }
122 
123  return 0;
124  }
125  // -----------------------------------------------------------------------------
126  private:
127  std::string appname() const {
128  return "oops::HybridGain<" + MODEL::name() + ">";
129  }
130  // -----------------------------------------------------------------------------
131 };
132 
133 } // namespace oops
134 
135 #endif // OOPS_RUNS_HYBRIDGAIN_H_
const eckit::mpi::Comm & getComm() const
Definition: Application.h:36
Geometry class used in oops; subclass of interface class interface::Geometry.
Geometry< MODEL > Geometry_
Definition: HybridGain.h:29
HybridGain(const eckit::mpi::Comm &comm=oops::mpi::world())
Definition: HybridGain.h:35
State< MODEL > State_
Definition: HybridGain.h:31
int execute(const eckit::Configuration &fullConfig) const
Definition: HybridGain.h:39
virtual ~HybridGain()
Definition: HybridGain.h:37
std::string appname() const
Definition: HybridGain.h:127
Increment< MODEL > Increment_
Definition: HybridGain.h:30
Increment class used in oops.
State class used in oops; subclass of interface class interface::State.
void diff(const State_ &state1, const State_ &state2)
Set this Increment to be difference between state1 and state2.
void zero()
Zero out this State.
const util::DateTime validTime() const
Accessor to the time of this State.
const Variables & variables() const
Accessor to variables associated with this State.
void accumul(const double &w, const State &x)
Accumulate (add w * x to the state)
void write(const eckit::Configuration &) const
Write this State out to file.
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.