OOPS
HofX4D.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 2020-2020 UCAR.
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  * In applying this licence, ECMWF does not waive the privileges and immunities
8  * granted to it by virtue of its status as an intergovernmental organisation nor
9  * does it submit to any jurisdiction.
10  */
11 
12 #ifndef OOPS_RUNS_HOFX4D_H_
13 #define OOPS_RUNS_HOFX4D_H_
14 
15 #include <string>
16 #include <vector>
17 
18 #include "eckit/config/LocalConfiguration.h"
19 #include "eckit/exception/Exceptions.h"
20 #include "oops/base/Geometry.h"
22 #include "oops/base/Model.h"
24 #include "oops/base/ObsErrors.h"
25 #include "oops/base/Observations.h"
26 #include "oops/base/Observers.h"
27 #include "oops/base/ObsSpaces.h"
29 #include "oops/base/State.h"
30 #include "oops/base/StateInfo.h"
34 #include "oops/mpi/mpi.h"
35 #include "oops/runs/Application.h"
36 #include "oops/util/algorithms.h"
37 #include "oops/util/DateTime.h"
38 #include "oops/util/Duration.h"
39 #include "oops/util/Logger.h"
40 #include "oops/util/parameters/OptionalParameter.h"
41 #include "oops/util/parameters/Parameter.h"
42 #include "oops/util/parameters/Parameters.h"
43 #include "oops/util/parameters/RequiredParameter.h"
44 
45 namespace oops {
46 
47 // -----------------------------------------------------------------------------
48 
49 /// \brief Options controlling the processing of observations from a single obs space.
50 template <typename OBS>
51 class ObsTypeParameters : public oops::Parameters {
52  OOPS_CONCRETE_PARAMETERS(ObsTypeParameters, Parameters)
53 
54  public:
58 
59  /// Options used to configure the observation space.
60  oops::RequiredParameter<ObsSpaceParameters_> obsSpace{"obs space", this};
61 
62  /// Options used to configure the observation operator, observation filters and GetValues.
64 
65  /// Options used to configure the observation error covariance matrix model.
66  oops::Parameter<ObsErrorParameters_> obsError{"obs error", {}, this};
67 
68  /// Options used to configure bias correction.
69  oops::Parameter<ObsAuxControlParameters_> obsBias{"obs bias", {}, this};
70 };
71 
72 // -----------------------------------------------------------------------------
73 
74 /// \brief Top-level options taken by the HofX4D application.
75 template <typename MODEL, typename OBS>
76 class HofX4DParameters : public oops::Parameters {
77  OOPS_CONCRETE_PARAMETERS(HofX4DParameters, Parameters)
78 
79  public:
82 
83  /// Only observations taken at times lying in the (`window begin`, `window begin` + `window
84  /// length`] interval will be included in observation spaces.
85  oops::RequiredParameter<util::DateTime> windowBegin{"window begin", this};
86  oops::RequiredParameter<util::Duration> windowLength{"window length", this};
87 
88  /// Forecast length.
89  oops::RequiredParameter<util::Duration> forecastLength{"forecast length", this};
90 
91  /// A list whose elements determine treatment of observations from individual observation spaces.
92  oops::Parameter<std::vector<ObsTypeParameters<OBS>>> observations{"observations", {}, this};
93 
94  /// Geometry parameters.
95  oops::RequiredParameter<GeometryParameters_> geometry{"geometry", this};
96 
97  /// Model parameters.
98  oops::RequiredParameter<ModelParameters_> model{"model", this};
99 
100  /// Initial state parameters.
101  oops::RequiredParameter<eckit::LocalConfiguration> initialCondition{"initial condition", this};
102 
103  /// Options passed to the object writing out forecast fields.
104  oops::Parameter<eckit::LocalConfiguration> prints{"prints", eckit::LocalConfiguration(), this};
105 
106  /// Whether to perturb the H(x) vector before saving.
107  oops::Parameter<bool> obsPerturbations{"obs perturbations", false, this};
108 
109  /// Whether to save the H(x) vector as ObsValues.
110  oops::Parameter<bool> makeObs{"make obs", false, this};
111 
112  /// Parameters used by regression tests comparing results produced by the application against
113  /// known good outputs.
114  oops::Parameter<eckit::LocalConfiguration> test{"test", eckit::LocalConfiguration(), this};
115 };
116 
117 // -----------------------------------------------------------------------------
118 
119 /// Application runs model forecast from "initial condition" for the "forecast length"
120 /// and computes H(x) on the run. If "obspert" is specified in the config, the resulting
121 /// H(x) is perturbed. It is saved as "hofx" by default, or as specified "hofx group name"
122 template <typename MODEL, typename OBS> class HofX4D : public Application {
132 
139 
140  public:
141 // -----------------------------------------------------------------------------
142  explicit HofX4D(const eckit::mpi::Comm & comm = oops::mpi::world()) : Application(comm) {
143  instantiateObsErrorFactory<OBS>();
144  instantiateObsFilterFactory<OBS>();
145  instantiateVariableChangeFactory<MODEL>();
146  }
147 // -----------------------------------------------------------------------------
148  virtual ~HofX4D() = default;
149 // -----------------------------------------------------------------------------
150  int execute(const eckit::Configuration & fullConfig) const {
151 // Deserialize parameters
152  HofX4DParameters_ params;
153  params.validateAndDeserialize(fullConfig);
154 
155 // Setup observation window
156  const util::Duration &winlen = params.windowLength;
157  const util::DateTime &winbgn = params.windowBegin;
158  const util::DateTime winend(winbgn + winlen);
159  Log::info() << "Observation window from " << winbgn << " to " << winend << std::endl;
160 
161 // Setup geometry
162  const Geometry_ geometry(params.geometry, this->getComm(), oops::mpi::myself());
163 
164 // Setup Model
165  const Model_ model(geometry, params.model.value().modelParameters);
166 
167 // Setup initial state
168  State_ xx(geometry, params.initialCondition);
169  ModelAux_ moderr(geometry, params.initialCondition);
170  const util::Duration &flength = params.forecastLength;
171  Log::test() << "Initial state: " << xx << std::endl;
172 
173 // Check that window specified for forecast is at least the same as obs window
174  if (winbgn < xx.validTime() || winend > xx.validTime() + flength) {
175  Log::error() << "Observation window can not be outside of forecast window." << std::endl;
176  Log::error() << "Obs window: " << winbgn << " to " << winend << std::endl;
177  Log::error() << "Forecast runs from: " << xx.validTime() << " for " << flength << std::endl;
178  throw eckit::BadValue("Observation window can not be outside of forecast window.");
179  }
180 
181 // Setup forecast outputs
183 
184  post.enrollProcessor(new StateInfo<State_>("fc", params.prints));
185 
186 // Setup observations
187 
188  const eckit::LocalConfiguration obsConfig(fullConfig, "observations");
189  const std::vector<ObsSpaceParameters_> obsspaceParams = util::transformVector(
190  params.observations.value(),
191  [](const ObsTypeParameters_ & obsTypeParams) { return obsTypeParams.obsSpace.value(); });
192  ObsSpaces_ obspaces(obsspaceParams, this->getComm(), winbgn, winend);
193 
194  const std::vector<ObsAuxControlParameters_> obsauxParams = util::transformVector(
195  params.observations.value(),
196  [](const ObsTypeParameters_ & obsTypeParams) { return obsTypeParams.obsBias.value(); });
197  ObsAux_ obsaux(obspaces, obsauxParams);
198 
199  const std::vector<ObsErrorParameters_> obsErrorParams = util::transformVector(
200  params.observations.value(),
201  [](const ObsTypeParameters_ & obsTypeParams) { return obsTypeParams.obsError.value(); });
202  ObsErrors_ Rmat(obsErrorParams, obspaces);
203 
204 // Setup and initialize observer
205  std::vector<ObserverParameters_> observerParams = util::transformVector(
206  params.observations.value(),
207  [](const ObsTypeParameters_ & obsTypeParams) { return obsTypeParams.observer; });
208  Observers_ hofx(obspaces, observerParams);
209  hofx.initialize(geometry, obsaux, Rmat, post);
210 
211 // run the model and compute H(x)
212  model.forecast(xx, moderr, flength, post);
213  Log::test() << "Final state: " << xx << std::endl;
214 
215 // Get observations from observer
216  Observations_ yobs(obspaces);
217  hofx.finalize(yobs);
218  Log::test() << "H(x): " << std::endl << yobs << "End H(x)" << std::endl;
219 
220 // Perturb H(x) if needed (can be used for generating obs in OSSE: perturbed H(x) could be saved
221 // as ObsValue if "hofx group name" == ObsValue.
222  if (params.obsPerturbations) {
223  yobs.perturb(Rmat);
224  Log::test() << "Perturbed H(x): " << std::endl << yobs << "End Perturbed H(x)" << std::endl;
225  }
226 
227 // Save H(x) as observations (if "make obs" == true)
228  if (params.makeObs) yobs.save("ObsValue");
229  obspaces.save();
230 
231  return 0;
232  }
233 // -----------------------------------------------------------------------------
234  private:
235  std::string appname() const {
236  return "oops::HofX4D<" + MODEL::name() + ", " + OBS::name() + ">";
237  }
238 // -----------------------------------------------------------------------------
239 };
240 
241 } // namespace oops
242 
243 #endif // OOPS_RUNS_HOFX4D_H_
const eckit::mpi::Comm & getComm() const
Definition: Application.h:36
Geometry class used in oops; subclass of interface class interface::Geometry.
interface::Geometry< MODEL >::Parameters_ Parameters_
HofX4D(const eckit::mpi::Comm &comm=oops::mpi::world())
Definition: HofX4D.h:142
int execute(const eckit::Configuration &fullConfig) const
Definition: HofX4D.h:150
ObserverParameters< OBS > ObserverParameters_
Definition: HofX4D.h:136
Observations< OBS > Observations_
Definition: HofX4D.h:127
Observers< MODEL, OBS > Observers_
Definition: HofX4D.h:129
Model< MODEL > Model_
Definition: HofX4D.h:124
ObsSpace< OBS >::Parameters_ ObsSpaceParameters_
Definition: HofX4D.h:137
ModelAuxControl< MODEL > ModelAux_
Definition: HofX4D.h:125
ObsSpaces< OBS > ObsSpaces_
Definition: HofX4D.h:130
HofX4DParameters< MODEL, OBS > HofX4DParameters_
Definition: HofX4D.h:133
virtual ~HofX4D()=default
State< MODEL > State_
Definition: HofX4D.h:131
std::string appname() const
Definition: HofX4D.h:235
ObsTypeParameters< OBS > ObsTypeParameters_
Definition: HofX4D.h:138
Geometry< MODEL > Geometry_
Definition: HofX4D.h:123
ObsErrorParametersWrapper< OBS > ObsErrorParameters_
Definition: HofX4D.h:135
ObsErrors< OBS > ObsErrors_
Definition: HofX4D.h:128
ObsAuxControl< OBS >::Parameters_ ObsAuxControlParameters_
Definition: HofX4D.h:134
ObsAuxControls< OBS > ObsAux_
Definition: HofX4D.h:126
Top-level options taken by the HofX4D application.
Definition: HofX4D.h:76
oops::Parameter< std::vector< ObsTypeParameters< OBS > > > observations
A list whose elements determine treatment of observations from individual observation spaces.
Definition: HofX4D.h:92
oops::RequiredParameter< eckit::LocalConfiguration > initialCondition
Initial state parameters.
Definition: HofX4D.h:101
oops::RequiredParameter< util::DateTime > windowBegin
Definition: HofX4D.h:85
oops::RequiredParameter< ModelParameters_ > model
Model parameters.
Definition: HofX4D.h:98
oops::Parameter< eckit::LocalConfiguration > prints
Options passed to the object writing out forecast fields.
Definition: HofX4D.h:104
Geometry< MODEL >::Parameters_ GeometryParameters_
Definition: HofX4D.h:80
oops::RequiredParameter< util::Duration > forecastLength
Forecast length.
Definition: HofX4D.h:89
oops::Parameter< bool > obsPerturbations
Whether to perturb the H(x) vector before saving.
Definition: HofX4D.h:107
oops::RequiredParameter< util::Duration > windowLength
Definition: HofX4D.h:86
ModelParametersWrapper< MODEL > ModelParameters_
Definition: HofX4D.h:81
oops::RequiredParameter< GeometryParameters_ > geometry
Geometry parameters.
Definition: HofX4D.h:95
oops::Parameter< bool > makeObs
Whether to save the H(x) vector as ObsValues.
Definition: HofX4D.h:110
Auxiliary state related to model (could be e.g. model bias), not used at the moment.
Abstract nonlinear forecast model used by high level algorithms and applications.
void forecast(State_ &xx, const ModelAux_ &, const util::Duration &len, PostProcessor< State_ > &post) const
Run the forecast from state xx for len time, with post postprocessors Does not need to be implemented...
Contains a polymorphic parameter holding an instance of a subclass of ModelParametersBase.
ObsAuxControl_::Parameters_ Parameters_
Holds a vector of ObsAuxControl.
Contains a polymorphic parameter holding an instance of a subclass of ObsErrorParametersBase.
Container for ObsErrors for all observation types that are used in DA.
Definition: ObsErrors.h:34
ObsSpace_::Parameters_ Parameters_
void save() const
Save files.
Definition: ObsSpaces.h:114
Options controlling the processing of observations from a single obs space.
Definition: HofX4D.h:51
oops::Parameter< ObsAuxControlParameters_ > obsBias
Options used to configure bias correction.
Definition: HofX4D.h:69
ObsAuxControl< OBS >::Parameters_ ObsAuxControlParameters_
Definition: HofX4D.h:55
ObserverParameters< OBS > observer
Options used to configure the observation operator, observation filters and GetValues.
Definition: HofX4D.h:63
oops::RequiredParameter< ObsSpaceParameters_ > obsSpace
Options used to configure the observation space.
Definition: HofX4D.h:60
ObsErrorParametersWrapper< OBS > ObsErrorParameters_
Definition: HofX4D.h:56
oops::Parameter< ObsErrorParameters_ > obsError
Options used to configure the observation error covariance matrix model.
Definition: HofX4D.h:66
ObsSpace< OBS >::Parameters_ ObsSpaceParameters_
Definition: HofX4D.h:57
Observations Class.
Definition: Observations.h:35
void save(const std::string &) const
Save/read observations values.
Definition: Observations.h:141
void perturb(const ObsErrors_ &)
Perturbations.
Definition: Observations.h:186
Computes observation operator (from GeoVaLs), applies bias correction and runs QC filters.
Definition: Observers.h:36
void initialize(const Geometry_ &, const ObsAuxCtrls_ &, ObsErrors_ &, PostProc_ &, const eckit::Configuration &=eckit::LocalConfiguration())
Initializes variables, obs bias, obs filters (could be different for different iterations.
Definition: Observers.h:98
void finalize(Observations_ &)
Computes H(x) from the filled in GeoVaLs.
Definition: Observers.h:115
Control model post processing.
Definition: PostProcessor.h:30
void enrollProcessor(PostBase_ *pp)
Definition: PostProcessor.h:38
State class used in oops; subclass of interface class interface::State.
Handles writing-out of forecast fields.
Definition: StateInfo.h:28
const util::DateTime validTime() const
Accessor to the time of this State.
int error
Definition: compare.py:168
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.