OOPS
CalcHofX.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 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 #ifndef OOPS_ASSIMILATION_CALCHOFX_H_
8 #define OOPS_ASSIMILATION_CALCHOFX_H_
9 
10 #include <memory>
11 #include <string>
12 #include <vector>
13 
14 #include "eckit/config/LocalConfiguration.h"
15 
17 #include "oops/base/Observations.h"
18 #include "oops/base/ObsFilters.h"
19 #include "oops/base/ObsSpaces.h"
20 #include "oops/base/ObsVector.h"
21 #include "oops/base/Variables.h"
22 #include "oops/interface/GeoVaLs.h"
27 #include "oops/util/Logger.h"
28 #include "oops/util/parameters/Parameter.h"
29 #include "oops/util/parameters/Parameters.h"
30 #include "oops/util/parameters/RequiredParameter.h"
31 
32 namespace oops {
33 
34 template <typename OBS>
35 class CalcHofXParameters : public Parameters {
36  OOPS_CONCRETE_PARAMETERS(CalcHofXParameters, Parameters)
37 
38  typedef typename OBS::ObsOperator::Parameters_ ObsOperatorParameters_;
39 
40  public:
41  oops::RequiredParameter<ObsOperatorParameters_> obsOperator{"obs operator", this};
42  oops::Parameter<std::vector<ObsFilterParametersWrapper<OBS>>> obsFilters{"obs filters", {}, this};
43 };
44 
45 // -----------------------------------------------------------------------------
46 
47 /// \brief Computes observation operator (from GeoVaLs), applies bias correction
48 /// and runs QC filters
49 template <typename OBS>
50 class CalcHofX {
60  template <typename DATA> using ObsData_ = ObsDataVector<OBS, DATA>;
61  template <typename DATA> using ObsDataVec_ = std::vector<std::shared_ptr<ObsData_<DATA>>>;
62 
63  typedef std::vector<std::unique_ptr<GeoVaLs_>> GeoVaLsVec_;
64  typedef std::vector<std::unique_ptr<Locations_>> LocationsVec_;
65  typedef std::vector<std::unique_ptr<ObsFilters_>> ObsFiltersVec_;
66  typedef std::vector<std::unique_ptr<ObsOperator_>> ObsOperatorVec_;
67  typedef std::vector<std::shared_ptr<ObsVector_>> ObsVectorVec_;
68  typedef std::vector<Variables> VariablesVec_;
69 
70  public:
71 /// \brief Initializes ObsOperators, Locations, and QC data
72  CalcHofX(const ObsSpaces_ &, const eckit::Configuration &);
73 
74 /// \brief Initializes variables, obs bias, obs filters (could be different for
75 /// different iterations
76  void initialize(const ObsAuxCtrls_ &, const int iteration = 0);
77 
78 /// \brief Computes H(x) from the filled in GeoVaLs
80 
81 /// \brief accessor to the locations
82  const LocationsVec_ & locations() const { return locations_; }
83 /// \brief accessor to variables required from the model
84  const VariablesVec_ & requiredVars() const { return geovars_; }
85 /// \brief accessor to QC flags
86  const ObsDataVec_<int> & qcflags() const { return qcflags_; }
87 
88 /// \brief read QC flags from \p qcname variable from file
89  void readQcFlags(const std::string & qcname);
90 /// \brief reset QC flags and ObsErrors
91  void resetQc();
92 /// \brief save QC flags to ObsSpaces
93  void saveQcFlags(const std::string &) const;
94 /// \brief save obs error variances (modified in QC) to ObsSpaces
95  void saveObsErrors(const std::string &) const;
96 /// \brief mask obs errors with QC flags
97  void maskObsErrors();
98 
99  private:
100  eckit::LocalConfiguration obsconfig_;
101  const ObsSpaces_ & obspaces_; // ObsSpaces used in H(x)
102  ObsOperatorVec_ obsops_; // Obs operators
103  LocationsVec_ locations_; // locations
104  const ObsAuxCtrls_ * biascoeff_; // bias coefficients
106  ObsVectorVec_ obserrs_; // Obs error variances (used in QC filters)
107  ObsFiltersVec_ filters_; // QC filters
108  VariablesVec_ geovars_; // variables required from the model
109 };
110 
111 // -----------------------------------------------------------------------------
112 
113 template <typename OBS>
115  const eckit::Configuration & config) :
116  obsconfig_(config), obspaces_(obspaces), obsops_(), locations_(),
117  biascoeff_(nullptr), qcflags_(), obserrs_(), filters_(),
118  geovars_(obspaces_.size())
119 {
120  std::vector<eckit::LocalConfiguration> obsconfs = config.getSubConfigurations();
121  for (size_t jj = 0; jj < obspaces_.size(); ++jj) {
122  // obsconfs[jj] contains not only options controlling the obs operator and filters (known to
123  // CalcHofX) but also those controlling the obs space (unknown to it). So we can't call
124  // validateAndDeserialize() here, since "obs space" would be treated as an unrecognized
125  // keyword. In the long term the code constructing the CalcHofX will probably need to split
126  // the contents of the "observations" vector into two vectors, one containing the "obs space"
127  // sections and the other the "obs operator" and "obs filters" sections, and pass the former to
128  // the constructor of ObsSpaces and the latter to the constructor of CalcHofX.
129  CalcHofXParameters<OBS> observerParams;
130  observerParams.deserialize(obsconfs[jj]);
131  /// Set up observation operators
132  obsops_.emplace_back(new ObsOperator_(obspaces_[jj], observerParams.obsOperator));
133  locations_.emplace_back(new Locations_(obsops_[jj]->locations()));
134 
135  /// Allocate QC flags
136  qcflags_.emplace_back(std::make_shared<ObsData_<int>>(obspaces[jj],
137  obspaces[jj].obsvariables()));
138  /// Allocate and read initial obs error
139  obserrs_.emplace_back(std::make_shared<ObsVector_>(obspaces[jj], "ObsError"));
140  }
141  Log::trace() << "CalcHofX<OBS> constructed" << std::endl;
142 }
143 
144 // -----------------------------------------------------------------------------
145 
146 template <typename OBS>
147 void CalcHofX<OBS>::initialize(const ObsAuxCtrls_ & obsaux, const int iteration) {
148  std::vector<eckit::LocalConfiguration> obsconfs = obsconfig_.getSubConfigurations();
149  biascoeff_ = &obsaux;
150  filters_.clear();
151  for (size_t jj = 0; jj < obspaces_.size(); ++jj) {
152  CalcHofXParameters<OBS> observerParams;
153  observerParams.deserialize(obsconfs[jj]);
154  /// Set up QC filters and run preprocess
155  filters_.emplace_back(new ObsFilters_(obspaces_[jj], observerParams.obsFilters,
156  qcflags_[jj], *obserrs_[jj], iteration));
157  filters_[jj]->preProcess();
158 
159  /// Set up variables requested from the model
160  geovars_[jj] += obsops_[jj]->requiredVars();
161  geovars_[jj] += (*biascoeff_)[jj].requiredVars();
162  geovars_[jj] += filters_[jj]->requiredVars();
163  }
164  Log::trace() << "CalcHofX<OBS>::initialize done" << std::endl;
165 }
166 
167 // -----------------------------------------------------------------------------
168 
169 template <typename OBS>
171  oops::Log::trace() << "CalcHofX<OBS>::compute start" << std::endl;
172 
173  Observations<OBS> yobs(obspaces_);
174  Observations<OBS> ybias(obspaces_);
175  ybias.zero();
176  for (size_t jj = 0; jj < obspaces_.size(); ++jj) {
177  /// call prior filters
178  filters_[jj]->priorFilter(*geovals[jj]);
179  /// compute H(x)
180  oops::Variables vars;
181  vars += filters_[jj]->requiredHdiagnostics();
182  vars += (*biascoeff_)[jj].requiredHdiagnostics();
183  ObsDiags_ ydiags(obspaces_[jj], *locations_[jj], vars);
184  obsops_[jj]->simulateObs(*geovals[jj], yobs[jj], (*biascoeff_)[jj], ybias[jj], ydiags);
185  /// call posterior filters
186  filters_[jj]->postFilter(yobs[jj], ybias[jj], ydiags);
187  }
188  oops::Log::trace() << "CalcHofX<OBS>::compute done" << std::endl;
189  return yobs;
190 }
191 
192 // -----------------------------------------------------------------------------
193 
194 template <typename OBS>
195 void CalcHofX<OBS>::readQcFlags(const std::string & qcname) {
196  for (const auto & qcflag : qcflags_) {
197  qcflag->read(qcname);
198  }
199 }
200 
201 // -----------------------------------------------------------------------------
202 
203 template <typename OBS>
205  for (const auto & qcflag : qcflags_) {
206  qcflag->zero();
207  }
208  for (const auto & obserr : obserrs_) {
209  obserr->read("ObsError");
210  }
211 }
212 
213 // -----------------------------------------------------------------------------
214 
215 template <typename OBS>
216 void CalcHofX<OBS>::saveQcFlags(const std::string & name) const {
217  for (const auto & qcflag : qcflags_) {
218  qcflag->save(name);
219  }
220 }
221 
222 // -----------------------------------------------------------------------------
223 
224 template <typename OBS>
225 void CalcHofX<OBS>::saveObsErrors(const std::string & name) const {
226  for (const auto & obserr : obserrs_) {
227  obserr->save(name);
228  }
229 }
230 
231 // -----------------------------------------------------------------------------
232 
233 template <typename OBS>
235  for (size_t jj = 0; jj < obserrs_.size(); ++jj) {
236  obserrs_[jj]->mask(*qcflags_[jj]);
237  }
238 }
239 
240 // -----------------------------------------------------------------------------
241 
242 } // namespace oops
243 
244 #endif // OOPS_ASSIMILATION_CALCHOFX_H_
Computes observation operator (from GeoVaLs), applies bias correction and runs QC filters.
Definition: CalcHofX.h:50
ObsDataVec_< int > qcflags_
Definition: CalcHofX.h:105
Observations< OBS > Observations_
Definition: CalcHofX.h:55
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
ObsSpaces< OBS > ObsSpaces_
Definition: CalcHofX.h:58
std::vector< std::unique_ptr< ObsOperator_ > > ObsOperatorVec_
Definition: CalcHofX.h:66
std::vector< std::shared_ptr< ObsVector_ > > ObsVectorVec_
Definition: CalcHofX.h:67
const ObsDataVec_< int > & qcflags() const
accessor to QC flags
Definition: CalcHofX.h:86
ObsFilters< OBS > ObsFilters_
Definition: CalcHofX.h:56
std::vector< std::unique_ptr< ObsFilters_ > > ObsFiltersVec_
Definition: CalcHofX.h:65
std::vector< std::unique_ptr< GeoVaLs_ > > GeoVaLsVec_
Definition: CalcHofX.h:63
ObsFiltersVec_ filters_
Definition: CalcHofX.h:107
ObsVector< OBS > ObsVector_
Definition: CalcHofX.h:59
const LocationsVec_ & locations() const
accessor to the locations
Definition: CalcHofX.h:82
eckit::LocalConfiguration obsconfig_
Definition: CalcHofX.h:100
Observations_ compute(const GeoVaLsVec_ &)
Computes H(x) from the filled in GeoVaLs.
Definition: CalcHofX.h:170
ObsVectorVec_ obserrs_
Definition: CalcHofX.h:106
void resetQc()
reset QC flags and ObsErrors
Definition: CalcHofX.h:204
void saveQcFlags(const std::string &) const
save QC flags to ObsSpaces
Definition: CalcHofX.h:216
const ObsSpaces_ & obspaces_
Definition: CalcHofX.h:101
Locations< OBS > Locations_
Definition: CalcHofX.h:52
ObsOperatorVec_ obsops_
Definition: CalcHofX.h:102
LocationsVec_ locations_
Definition: CalcHofX.h:103
const VariablesVec_ & requiredVars() const
accessor to variables required from the model
Definition: CalcHofX.h:84
CalcHofX(const ObsSpaces_ &, const eckit::Configuration &)
Initializes ObsOperators, Locations, and QC data.
Definition: CalcHofX.h:114
GeoVaLs< OBS > GeoVaLs_
Definition: CalcHofX.h:51
void maskObsErrors()
mask obs errors with QC flags
Definition: CalcHofX.h:234
ObsAuxControls< OBS > ObsAuxCtrls_
Definition: CalcHofX.h:53
std::vector< Variables > VariablesVec_
Definition: CalcHofX.h:68
ObsDiagnostics< OBS > ObsDiags_
Definition: CalcHofX.h:54
VariablesVec_ geovars_
Definition: CalcHofX.h:108
std::vector< std::unique_ptr< Locations_ > > LocationsVec_
Definition: CalcHofX.h:64
void readQcFlags(const std::string &qcname)
read QC flags from qcname variable from file
Definition: CalcHofX.h:195
std::vector< std::shared_ptr< ObsData_< DATA > >> ObsDataVec_
Definition: CalcHofX.h:61
const ObsAuxCtrls_ * biascoeff_
Definition: CalcHofX.h:104
ObsOperator< OBS > ObsOperator_
Definition: CalcHofX.h:57
OBS::ObsOperator::Parameters_ ObsOperatorParameters_
Definition: CalcHofX.h:38
oops::Parameter< std::vector< ObsFilterParametersWrapper< OBS > > > obsFilters
Definition: CalcHofX.h:42
oops::RequiredParameter< ObsOperatorParameters_ > obsOperator
Definition: CalcHofX.h:41
Locations of observations for observation operator.
Holds a vector of ObsAuxControl.
ObsDataVector is a vector templated on data type, in the observation space.
Holds observation filters (usually QC) for one observation type.
Definition: ObsFilters.h:36
MODEL-agnostic part of nonlinear observation (forward) operator. The full nonlinear observation opera...
std::size_t size() const
Access.
Definition: ObsSpaces.h:61
ObsVector class used in oops; subclass of interface class interface::ObsVector.
Observations Class.
Definition: Observations.h:35
void zero()
Accumulator.
Definition: Observations.h:155
The namespace for the main oops code.