OOPS
generic/ObsErrorBase.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
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  * In applying this licence, ECMWF does not waive the privileges and immunities
7  * granted to it by virtue of its status as an intergovernmental organisation nor
8  * does it submit to any jurisdiction.
9  */
10 
11 #ifndef OOPS_GENERIC_OBSERRORBASE_H_
12 #define OOPS_GENERIC_OBSERRORBASE_H_
13 
14 #include <map>
15 #include <memory>
16 #include <string>
17 #include <vector>
18 
19 #include <boost/noncopyable.hpp>
20 #include "eckit/config/Configuration.h"
21 
22 #include "oops/base/ObsVector.h"
24 #include "oops/util/AssociativeContainers.h"
25 #include "oops/util/Logger.h"
26 #include "oops/util/parameters/Parameter.h"
27 #include "oops/util/parameters/Parameters.h"
28 #include "oops/util/parameters/PolymorphicParameter.h"
29 #include "oops/util/Printable.h"
30 
31 namespace oops {
32 
33 // -----------------------------------------------------------------------------
34 /// \brief Base class for generic implementations of observation error covariance matrices.
35 ///
36 /// Use this class as a base class for generic implementations,
37 /// and interface::ObsErrorBase as a base class for OBS-specific implementations.
38 ///
39 /// Note: each generic implementation should typedef `Parameters_` to the name of a subclass of
40 /// ObsErrorParametersBase holding its configuration settings and provide a constructor with the
41 /// following signature:
42 ///
43 /// ObsErrorBase(const Parameters_ &, ObsSpace<OBS> &);
44 template<typename OBS>
45 class ObsErrorBase : public util::Printable,
46  private boost::noncopyable {
49 
50  public:
51  ObsErrorBase() = default;
52  virtual ~ObsErrorBase() = default;
53 
54 /// Multiply a Departure \p dy by \f$R\f$.
55  virtual void multiply(ObsVector_ & dy) const = 0;
56 /// Multiply a Departure \p dy by \f$R^{-1}\f$.
57  virtual void inverseMultiply(ObsVector_ & dy) const = 0;
58 
59 /// Generate random perturbation in \p dy.
60  virtual void randomize(ObsVector_ & dy) const = 0;
61 
62 /// Save obs errors to the group \p name.
63  virtual void save(const std::string &name) const = 0;
64 
65 /// Return a copy of obs error std. dev. If this ObsVector_ is modified (e.g. by obs filters),
66 /// it should be passed back to update() to ensure the covariance matrix stays consistent.
67  virtual ObsVector_ obserrors() const = 0;
68 
69 /// Set the diagonal of the covariance matrix to \p stddev squared.
70  virtual void update(const ObsVector_ &stddev) = 0;
71 
72 /// Return the vector of inverse obs error variances.
73  virtual ObsVector_ inverseVariance() const = 0;
74 
75 /// Get mean error for Jo table.
76  virtual double getRMSE() const = 0;
77 
78  private:
79  virtual void print(std::ostream &) const = 0;
80 };
81 
82 // =============================================================================
83 
84 template <typename OBS>
85 class ObsErrorFactory;
86 
87 // -----------------------------------------------------------------------------
88 
89 /// \brief Configuration parameters of an implementation of an observation error covariance matrix
90 /// model.
91 class ObsErrorParametersBase : public Parameters {
92  OOPS_ABSTRACT_PARAMETERS(ObsErrorParametersBase, Parameters)
93  public:
94  /// \brief Name of the covariance model.
95  Parameter<std::string> model{"covariance model", "diagonal", this};
96 };
97 
98 // -----------------------------------------------------------------------------
99 
100 /// \brief Contains a polymorphic parameter holding an instance of a subclass of
101 /// ObsErrorParametersBase.
102 template <typename OBS>
103 class ObsErrorParametersWrapper : public Parameters {
104  OOPS_CONCRETE_PARAMETERS(ObsErrorParametersWrapper, Parameters)
105  public:
106  /// After deserialization, holds an instance of a subclass of ObsErrorParametersBase controlling
107  /// the behavior of an observation error covariance matrix model. The type of the subclass is
108  /// determined by the value of the "covariance model" key in the Configuration object from which
109  /// this object is deserialized.
110  PolymorphicParameter<ObsErrorParametersBase, ObsErrorFactory<OBS>>
111  obsErrorParameters{"covariance model", "diagonal", this};
112 };
113 
114 // -----------------------------------------------------------------------------
115 
116 /// A factory creating instances of concrete subclasses of ObsErrorBase.
117 template <typename OBS>
121 
122  public:
123  /// \brief Create and return a new observation error covariance matrix model.
124  ///
125  /// The model's type is determined by the `covariance model` attribute of \p params.
126  /// \p params must be an instance of the subclass of ObsErrorParametersBase
127  /// associated with that model type, otherwise an exception will be thrown.
128  static std::unique_ptr<ObsErrorBase_> create(const ObsErrorParametersBase &params,
129  const ObsSpace_ &);
130 
131  /// \brief Create and return an instance of the subclass of ObsErrorParametersBase
132  /// storing parameters of covariance matrix models of the specified type.
133  static std::unique_ptr<ObsErrorParametersBase> createParameters(const std::string &model);
134 
135  /// \brief Return the names of all models that can be created by one of the registered makers.
136  static std::vector<std::string> getMakerNames() {
137  return keys(getMakers());
138  }
139 
140  virtual ~ObsErrorFactory() = default;
141 
142  protected:
143  /// \brief Register a maker able to create covariance matrix models of type \p model.
144  explicit ObsErrorFactory(const std::string &model);
145 
146  private:
147  virtual std::unique_ptr<ObsErrorBase_> make(const ObsErrorParametersBase &,
148  const ObsSpace_ &) = 0;
149 
150  virtual std::unique_ptr<ObsErrorParametersBase> makeParameters() const = 0;
151 
152  static std::map < std::string, ObsErrorFactory<OBS> * > & getMakers() {
153  static std::map < std::string, ObsErrorFactory<OBS> * > makers_;
154  return makers_;
155  }
156 };
157 
158 // -----------------------------------------------------------------------------
159 
160 /// \brief A subclass of ObsErrorFactory able to create instances of T (a concrete subclass of
161 /// ObsErrorBase<OBS>). Passes ObsSpace<OBS> to the constructor of T.
162 template<class OBS, class T>
163 class ObsErrorMaker : public ObsErrorFactory<OBS> {
166  typedef typename T::Parameters_ Parameters_;
167 
168  std::unique_ptr<ObsErrorBase_> make(const ObsErrorParametersBase & parameters,
169  const ObsSpace_ & obs) override {
170  const auto &stronglyTypedParameters = dynamic_cast<const Parameters_&>(parameters);
171  return std::make_unique<T>(stronglyTypedParameters, obs);
172  }
173 
174  std::unique_ptr<ObsErrorParametersBase> makeParameters() const override {
175  return std::make_unique<Parameters_>();
176  }
177 
178  public:
179  explicit ObsErrorMaker(const std::string & name) : ObsErrorFactory<OBS>(name) {}
180 };
181 
182 // =============================================================================
183 
184 template <typename OBS>
185 ObsErrorFactory<OBS>::ObsErrorFactory(const std::string & name) {
186  if (getMakers().find(name) != getMakers().end()) {
187  throw std::runtime_error(name + " already registered in obs error factory.");
188  }
189  getMakers()[name] = this;
190 }
191 
192 // -----------------------------------------------------------------------------
193 
194 template <typename OBS>
195 std::unique_ptr<ObsErrorBase<OBS>>
197  Log::trace() << "ObsErrorFactory<OBS>::create starting" << std::endl;
198  const std::string &id = params.model;
199  Log::trace() << "ObsError matrix type is: " << id << std::endl;
200  typename std::map<std::string, ObsErrorFactory<OBS>*>::iterator
201  jerr = getMakers().find(id);
202  if (jerr == getMakers().end()) {
203  throw std::runtime_error(id + " does not exist in obs error factory.");
204  }
205  std::unique_ptr<ObsErrorBase<OBS>> ptr(jerr->second->make(params, obs));
206  Log::trace() << "ObsErrorFactory<OBS>::create done" << std::endl;
207  return ptr;
208 }
209 
210 // -----------------------------------------------------------------------------
211 
212 template <typename OBS>
213 std::unique_ptr<ObsErrorParametersBase> ObsErrorFactory<OBS>::createParameters(
214  const std::string &name) {
215  Log::trace() << "ObsErrorFactory<OBS>::createParameters starting" << std::endl;
216  typename std::map<std::string, ObsErrorFactory<OBS>*>::iterator it = getMakers().find(name);
217  if (it == getMakers().end()) {
218  throw std::runtime_error(name + " does not exist in obs error factory");
219  }
220  return it->second->makeParameters();
221 }
222 
223 // -----------------------------------------------------------------------------
224 
225 } // namespace oops
226 
227 #endif // OOPS_GENERIC_OBSERRORBASE_H_
Base class for generic implementations of observation error covariance matrices.
virtual void update(const ObsVector_ &stddev)=0
Set the diagonal of the covariance matrix to stddev squared.
ObsErrorBase()=default
virtual void multiply(ObsVector_ &dy) const =0
Multiply a Departure dy by .
ObsVector< OBS > ObsVector_
virtual void print(std::ostream &) const =0
virtual double getRMSE() const =0
Get mean error for Jo table.
ObsSpace< OBS > ObsSpace_
virtual ObsVector_ obserrors() const =0
virtual ~ObsErrorBase()=default
virtual void randomize(ObsVector_ &dy) const =0
Generate random perturbation in dy.
virtual void save(const std::string &name) const =0
Save obs errors to the group name.
virtual void inverseMultiply(ObsVector_ &dy) const =0
Multiply a Departure dy by .
virtual ObsVector_ inverseVariance() const =0
Return the vector of inverse obs error variances.
A factory creating instances of concrete subclasses of ObsErrorBase.
static std::unique_ptr< ObsErrorParametersBase > createParameters(const std::string &model)
Create and return an instance of the subclass of ObsErrorParametersBase storing parameters of covaria...
virtual std::unique_ptr< ObsErrorBase_ > make(const ObsErrorParametersBase &, const ObsSpace_ &)=0
virtual std::unique_ptr< ObsErrorParametersBase > makeParameters() const =0
virtual ~ObsErrorFactory()=default
ObsErrorBase< OBS > ObsErrorBase_
static std::unique_ptr< ObsErrorBase_ > create(const ObsErrorParametersBase &params, const ObsSpace_ &)
Create and return a new observation error covariance matrix model.
static std::vector< std::string > getMakerNames()
Return the names of all models that can be created by one of the registered makers.
static std::map< std::string, ObsErrorFactory< OBS > * > & getMakers()
ObsErrorFactory(const std::string &model)
Register a maker able to create covariance matrix models of type model.
A subclass of ObsErrorFactory able to create instances of T (a concrete subclass of ObsErrorBase<OBS>...
ObsErrorBase< OBS > ObsErrorBase_
std::unique_ptr< ObsErrorParametersBase > makeParameters() const override
std::unique_ptr< ObsErrorBase_ > make(const ObsErrorParametersBase &parameters, const ObsSpace_ &obs) override
ObsErrorMaker(const std::string &name)
Configuration parameters of an implementation of an observation error covariance matrix model.
Parameter< std::string > model
Name of the covariance model.
Contains a polymorphic parameter holding an instance of a subclass of ObsErrorParametersBase.
PolymorphicParameter< ObsErrorParametersBase, ObsErrorFactory< OBS > > obsErrorParameters
ObsVector class used in oops; subclass of interface class interface::ObsVector.
The namespace for the main oops code.