OOPS
LinearVariableChangeBase.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018 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_BASE_LINEARVARIABLECHANGEBASE_H_
9 #define OOPS_BASE_LINEARVARIABLECHANGEBASE_H_
10 
11 #include <map>
12 #include <memory>
13 #include <string>
14 #include <vector>
15 
16 #include <boost/make_unique.hpp>
17 #include <boost/noncopyable.hpp>
18 
19 #include "eckit/exception/Exceptions.h"
20 
21 #include "oops/base/Geometry.h"
22 #include "oops/base/Increment.h"
24 #include "oops/base/State.h"
25 #include "oops/base/Variables.h"
26 #include "oops/util/AssociativeContainers.h"
27 #include "oops/util/parameters/ConfigurationParameter.h"
28 #include "oops/util/parameters/HasParameters_.h"
29 #include "oops/util/parameters/Parameters.h"
30 #include "oops/util/parameters/RequiredPolymorphicParameter.h"
31 #include "oops/util/Printable.h"
32 
33 namespace eckit {
34  class Configuration;
35 }
36 
37 namespace oops {
38 
39 // -----------------------------------------------------------------------------
40 
41 /// Base class for generic variable transform
42 ///
43 /// Note: subclasses can opt to extract their settings either from a Configuration object or from a
44 /// subclass of Parameters.
45 ///
46 /// In the former case, they should provide a constructor taking a const reference to an
47 /// eckit::Configuration object. In the latter case, the implementer should first define a subclass
48 /// of Parameters holding the settings of the linear variable change in question. The latter should
49 /// then typedef `Parameters_` to the name of that subclass and provide a constructor taking a
50 /// const reference to an instance of that subclass.
51 template <typename MODEL>
52 class LinearVariableChangeBase : public util::Printable,
53  private boost::noncopyable {
57 
58  public:
60  explicit LinearVariableChangeBase(const eckit::Configuration &);
62 
63  void setInputVariables(const Variables & vars) { varin_.reset(new Variables(vars)); }
64  void setOutputVariables(const Variables & vars) { varout_.reset(new Variables(vars)); }
65 
66  virtual void multiply(const Increment_ &, Increment_ &) const = 0;
67  virtual void multiplyInverse(const Increment_ &, Increment_ &) const = 0;
68  virtual void multiplyAD(const Increment_ &, Increment_ &) const = 0;
69  virtual void multiplyInverseAD(const Increment_ &, Increment_ &) const = 0;
70 
71  Increment_ multiply(const Increment_ &) const;
72  Increment_ multiplyInverse(const Increment_ &) const;
73  Increment_ multiplyAD(const Increment_ &) const;
75 
76  private:
77  virtual void print(std::ostream &) const = 0;
78  std::unique_ptr<Variables> varin_;
79  std::unique_ptr<Variables> varout_;
80 };
81 
82 // =============================================================================
83 
84 template <typename MODEL>
86 
87 // -----------------------------------------------------------------------------
88 
89 /// \brief A subclass of LinearVariableChangeParametersBase storing the values of all options in a
90 /// single Configuration object.
91 ///
92 /// This object can be accessed by calling the value() method of the \p config member variable.
93 ///
94 /// The ConfigurationParameter class does not perform any parameter validation; models using
95 /// GenericLinearVariableChangeParameters should therefore ideally be refactored, replacing this
96 /// class with a dedicated subclass of LinearVariableChangeParametersBase storing each parameter in
97 /// a separate (Optional/Required)Parameter object.
99  OOPS_CONCRETE_PARAMETERS(GenericLinearVariableChangeParameters,
101  public:
102  ConfigurationParameter config{this};
103 };
104 
105 // -----------------------------------------------------------------------------
106 
107 /// \brief Contains a polymorphic parameter holding an instance of a subclass of
108 /// LinearVariableChangeParametersBase.
109 template <typename MODEL>
110 class LinearVariableChangeParametersWrapper : public Parameters {
111  OOPS_CONCRETE_PARAMETERS(LinearVariableChangeParametersWrapper, Parameters)
112  public:
113  /// After deserialization, holds an instance of a subclass of LinearVariableChangeParametersBase
114  /// controlling the behavior of a linear variable change. The type of the subclass is determined
115  /// by the value of the "variable change" key in the Configuration object from which this object
116  /// is deserialized.
117  RequiredPolymorphicParameter<LinearVariableChangeParametersBase,
119  variableChangeParameters{"variable change", this};
120 };
121 
122 // =============================================================================
123 
124 /// LinearVariableChange factory
125 template <typename MODEL>
129 
130  public:
131  /// \brief Create and return a new linear variable change.
132  ///
133  /// The type of the variable change is determined by the `variable change` attribute of \p
134  /// parameters. \p parameters must be an instance of the subclass of
135  /// LinearVariableChangeParametersBase associated with that variable change type, otherwise an
136  /// exception will be thrown.
138  const State_ &, const State_ &, const Geometry_ &,
139  const LinearVariableChangeParametersBase & parameters);
140 
141  /// \brief Create and return a new linear variable change.
142  ///
143  /// Deprecated overload taking a Configuration instead of a LinearVariableChangeParametersBase.
145  const State_ &, const State_ &, const Geometry_ &,
146  const eckit::Configuration & conf);
147 
148  /// \brief Create and return an instance of the subclass of LinearVariableChangeParametersBase
149  /// storing parameters of linear variable changes of the specified type.
150  static std::unique_ptr<LinearVariableChangeParametersBase> createParameters(
151  const std::string &name);
152 
153  /// \brief Return the names of all linear variable changes that can be created by one of the
154  /// registered makers.
155  static std::vector<std::string> getMakerNames() {
156  return keys(getMakers());
157  }
158 
159  virtual ~LinearVariableChangeFactory() = default;
160 
161  protected:
162  /// \brief Register a maker able to create linear variable changes of type \p name.
163  explicit LinearVariableChangeFactory(const std::string &name);
164 
165  private:
166  virtual LinearVariableChangeBase<MODEL> * make(const State_ &, const State_ &,
167  const Geometry_ &,
169 
170  virtual std::unique_ptr<LinearVariableChangeParametersBase> makeParameters() const = 0;
171 
172  static std::map < std::string, LinearVariableChangeFactory<MODEL> * > & getMakers() {
173  static std::map < std::string, LinearVariableChangeFactory<MODEL> * > makers_;
174  return makers_;
175  }
176 };
177 
178 // -----------------------------------------------------------------------------
179 
180 template<class MODEL, class T>
182  /// Defined as T::Parameters_ if T defines a Parameters_ type; otherwise as
183  /// GenericLinearVariableChangeParameters.
184  typedef TParameters_IfAvailableElseFallbackType_t<T, GenericLinearVariableChangeParameters>
186 
189 
191  const Geometry_ & geom,
192  const LinearVariableChangeParametersBase& params) override {
193  const auto &stronglyTypedParams = dynamic_cast<const Parameters_&>(params);
194  return new T(bg, fg, geom,
195  parametersOrConfiguration<HasParameters_<T>::value>(stronglyTypedParams));
196  }
197 
198  std::unique_ptr<LinearVariableChangeParametersBase> makeParameters() const override {
199  return boost::make_unique<Parameters_>();
200  }
201 
202  public:
203  explicit LinearVariableChangeMaker(const std::string & name)
204  : LinearVariableChangeFactory<MODEL>(name) {}
205 };
206 
207 // -----------------------------------------------------------------------------
208 
209 template <typename MODEL>
211  if (getMakers().find(name) != getMakers().end()) {
212  throw std::runtime_error(name + " already registered in the linear variable change factory.");
213  }
214  getMakers()[name] = this;
215 }
216 
217 // -----------------------------------------------------------------------------
218 
219 template <typename MODEL>
221  const State_ & bg, const State_ & fg,
222  const Geometry_ & geom, const LinearVariableChangeParametersBase & parameters) {
223  Log::trace() << "LinearVariableChangeBase<MODEL>::create starting" << std::endl;
224  const std::string &id = parameters.variableChange.value().value();
225  typename std::map<std::string, LinearVariableChangeFactory<MODEL>*>::iterator
226  jerr = getMakers().find(id);
227 
228  if (jerr == getMakers().end()) {
229  Log::error() << id << " does not exist in LinearVariableChangeFactory." << std::endl;
230  Log::error() << "Factory contains " << getMakers().size() << " elements:" << std::endl;
231  for (typename std::map<std::string, LinearVariableChangeFactory<MODEL>*>::const_iterator
232  jj = getMakers().begin(); jj != getMakers().end(); ++jj) {
233  Log::error() << "A " << jj->first << " variable change option" << std::endl;
234  }
235  throw std::runtime_error(id + " does not exist in the linear variable change factory.");
236  }
237 
238  LinearVariableChangeBase<MODEL> * ptr = jerr->second->make(bg, fg, geom, parameters);
239  Log::trace() << "LinearVariableChangeBase<MODEL>::create done" << std::endl;
240  return ptr;
241 }
242 
243 // -----------------------------------------------------------------------------
244 
245 template <typename MODEL>
247  const State_ & bg, const State_ & fg,
248  const Geometry_ & geom, const eckit::Configuration & conf) {
250  parameters.validateAndDeserialize(conf);
251  return create(bg, fg, geom, parameters.variableChangeParameters);
252 }
253 
254 // -----------------------------------------------------------------------------
255 
256 template <typename MODEL>
257 std::unique_ptr<LinearVariableChangeParametersBase>
259  const std::string &name) {
260  typename std::map<std::string, LinearVariableChangeFactory<MODEL>*>::iterator it =
261  getMakers().find(name);
262  if (it == getMakers().end()) {
263  throw std::runtime_error(name + " does not exist in LinearVariableChangeFactory");
264  }
265  return it->second->makeParameters();
266 }
267 
268 // =============================================================================
269 
270 template<typename MODEL>
272  const LinearVariableChangeParametersBase & parameters)
273  : varin_(), varout_()
274 {
275  if (parameters.inputVariables.value() != boost::none) {
276  varin_.reset(new Variables(*parameters.inputVariables.value()));
277  Log::trace() << "LinearVariableChangeBase<MODEL>::LinearVariableChangeBase input variables: "
278  << *varin_ << std::endl;
279  }
280  if (parameters.outputVariables.value() != boost::none) {
281  varout_.reset(new Variables(*parameters.outputVariables.value()));
282  Log::trace() << "LinearVariableChangeBase<MODEL>::LinearVariableChangeBase output variables: "
283  << *varout_ << std::endl;
284  }
285 }
286 
287 // -----------------------------------------------------------------------------
288 
289 template<typename MODEL>
291  const eckit::Configuration & conf)
292  : LinearVariableChangeBase(validateAndDeserialize<GenericLinearVariableChangeParameters>(conf))
293 {}
294 
295 // -----------------------------------------------------------------------------
296 
297 template<typename MODEL>
299  ASSERT(varin_);
300  Increment_ dxout(dxin.geometry(), *varout_, dxin.validTime());
301  this->multiply(dxin, dxout);
302  return dxout;
303 }
304 
305 // -----------------------------------------------------------------------------
306 
307 template<typename MODEL>
309  ASSERT(varout_);
310  Increment_ dxout(dxin.geometry(), *varin_, dxin.validTime());
311  this->multiplyAD(dxin, dxout);
312  return dxout;
313 }
314 
315 // -----------------------------------------------------------------------------
316 
317 template<typename MODEL>
319  ASSERT(varout_);
320  Increment_ dxout(dxin.geometry(), *varin_, dxin.validTime());
321  this->multiplyInverse(dxin, dxout);
322  return dxout;
323 }
324 
325 // -----------------------------------------------------------------------------
326 
327 template<typename MODEL>
329  ASSERT(varin_);
330  Increment_ dxout(dxin.geometry(), *varout_, dxin.validTime());
331  this->multiplyInverseAD(dxin, dxout);
332  return dxout;
333 }
334 
335 // -----------------------------------------------------------------------------
336 
337 } // namespace oops
338 
339 #endif // OOPS_BASE_LINEARVARIABLECHANGEBASE_H_
A subclass of LinearVariableChangeParametersBase storing the values of all options in a single Config...
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
virtual void multiplyInverseAD(const Increment_ &, Increment_ &) const =0
virtual void multiply(const Increment_ &, Increment_ &) const =0
void setOutputVariables(const Variables &vars)
std::unique_ptr< Variables > varout_
virtual void multiplyAD(const Increment_ &, Increment_ &) const =0
void setInputVariables(const Variables &vars)
LinearVariableChangeBase(const LinearVariableChangeParametersBase &)
std::unique_ptr< Variables > varin_
virtual void print(std::ostream &) const =0
virtual void multiplyInverse(const Increment_ &, Increment_ &) const =0
static LinearVariableChangeBase< MODEL > * create(const State_ &, const State_ &, const Geometry_ &, const LinearVariableChangeParametersBase &parameters)
Create and return a new linear variable change.
virtual std::unique_ptr< LinearVariableChangeParametersBase > makeParameters() const =0
virtual LinearVariableChangeBase< MODEL > * make(const State_ &, const State_ &, const Geometry_ &, const LinearVariableChangeParametersBase &)=0
static std::unique_ptr< LinearVariableChangeParametersBase > createParameters(const std::string &name)
Create and return an instance of the subclass of LinearVariableChangeParametersBase storing parameter...
static std::map< std::string, LinearVariableChangeFactory< MODEL > * > & getMakers()
LinearVariableChangeFactory(const std::string &name)
Register a maker able to create linear variable changes of type name.
static std::vector< std::string > getMakerNames()
Return the names of all linear variable changes that can be created by one of the registered makers.
virtual ~LinearVariableChangeFactory()=default
TParameters_IfAvailableElseFallbackType_t< T, GenericLinearVariableChangeParameters > Parameters_
LinearVariableChangeBase< MODEL > * make(const State_ &bg, const State_ &fg, const Geometry_ &geom, const LinearVariableChangeParametersBase &params) override
std::unique_ptr< LinearVariableChangeParametersBase > makeParameters() const override
LinearVariableChangeMaker(const std::string &name)
Base class of classes storing parameters controlling specific linear variable changes.
OptionalParameter< std::string > variableChange
Variable change type.
Contains a polymorphic parameter holding an instance of a subclass of LinearVariableChangeParametersB...
RequiredPolymorphicParameter< LinearVariableChangeParametersBase, LinearVariableChangeFactory< MODEL > > variableChangeParameters
State class used in oops; subclass of interface class interface::State.
Geometry_ geometry() const
Accessor to geometry associated with this Increment.
const util::DateTime validTime() const
Accessor to the time of this Increment.
int error
Definition: compare.py:168
Definition: FieldL95.h:22
The namespace for the main oops code.