OOPS
VariableChangeBase.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-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_BASE_VARIABLECHANGEBASE_H_
9 #define OOPS_BASE_VARIABLECHANGEBASE_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 "oops/base/Geometry.h"
20 #include "oops/base/State.h"
22 #include "oops/util/AssociativeContainers.h"
23 #include "oops/util/parameters/ConfigurationParameter.h"
24 #include "oops/util/parameters/HasParameters_.h"
25 #include "oops/util/parameters/Parameters.h"
26 #include "oops/util/parameters/PolymorphicParameter.h"
27 #include "oops/util/Printable.h"
28 
29 namespace eckit {
30  class Configuration;
31 }
32 
33 namespace oops {
34 
35 // -------------------------------------------------------------------------------------------------
36 
37 /// Base class for a variable transforms, defining the interfaces.
38 /// Use this class as a base class for generic implementations,
39 /// and VariableChangeBase as a base class for MODEL-specific implementations.
40 ///
41 /// Note: subclasses can opt to extract their settings either from a Configuration object or from a
42 /// subclass of Parameters.
43 ///
44 /// In the former case, they should provide a constructor with the following signature:
45 ///
46 /// VariableChange(const Geometry_ &, const eckit::Configuration &);
47 ///
48 /// In the latter case, the implementer should first define a subclass of
49 /// VariableChangeParametersBase holding the settings of the variable change in question.
50 /// The implementation of the VariableChange interface should then typedef `Parameters_`
51 /// to the name of that subclass and provide a constructor with the following signature:
52 ///
53 /// VariableChange(const Geometry_ &, const Parameters_ &);
54 template <typename MODEL>
55 class GenericVariableChangeBase : public util::Printable,
56  private boost::noncopyable {
58 
59  public:
61  virtual ~GenericVariableChangeBase() = default;
62 
63  /// change variables from state \p xin to \p xout
64  virtual void changeVar(const State_ & xin, State_ & xout) const = 0;
65  /// inverse of changeVar, change variables back from \p xout to \p xin
66  virtual void changeVarInverse(const State_ & xout, State_ & xin) const = 0;
67 
68  private:
69  /// Print, used for logging
70  virtual void print(std::ostream &) const = 0;
71 };
72 
73 /// \brief Base class for MODEL-specific implementations of VariableChange class.
74 /// The complete interface that needs to be implemented is described in
75 /// GenericVariableChangeBase. VariableChangeBase overrides GenericVariableChangeBase
76 /// methods to pass MODEL-specific implementations of State to the MODEL-specific
77 /// implementation of VariableChange.
78 template <typename MODEL>
80  typedef typename MODEL::State State_;
81 
82  public:
83  VariableChangeBase() = default;
84  virtual ~VariableChangeBase() = default;
85 
86  /// Overrides for VariableChangeBase classes, passing MODEL-specific classes to the
87  /// MODEL-specific implementations of VariableChange
88  void changeVar(const State<MODEL> & xin, State<MODEL> & xout) const final
89  { this->changeVar(xin.state(), xout.state()); }
90  void changeVarInverse(const State<MODEL> & xout, State<MODEL> & xin) const final
91  { this->changeVarInverse(xout.state(), xin.state()); }
92 
93  /// change variables from state \p xin to \p xout
94  virtual void changeVar(const State_ & xin, State_ & xout) const = 0;
95  /// inverse of changeVar, change variables back from \p xout to \p xin
96  virtual void changeVarInverse(const State_ & xout, State_ & xin) const = 0;
97 };
98 
99 
100 // =============================================================================
101 
102 template <typename MODEL>
104 
105 // -----------------------------------------------------------------------------
106 
107 /// \brief A subclass of VariableChangeParametersBase storing the values of all options in a
108 /// single Configuration object.
109 ///
110 /// This object can be accessed by calling the value() method of the \p config member variable.
111 ///
112 /// The ConfigurationParameter class does not perform any parameter validation; models using
113 /// GenericVariableChangeParameters should therefore ideally be refactored, replacing this
114 /// class with a dedicated subclass of VariableChangeParametersBase storing each parameter in
115 /// a separate (Optional/Required)Parameter object.
117  OOPS_CONCRETE_PARAMETERS(GenericVariableChangeParameters,
119  public:
120  ConfigurationParameter config{this};
121 };
122 
123 // -----------------------------------------------------------------------------
124 
125 /// \brief Contains a polymorphic parameter holding an instance of a subclass of
126 /// VariableChangeParametersBase.
127 template <typename MODEL>
128 class VariableChangeParametersWrapper : public Parameters {
129  OOPS_CONCRETE_PARAMETERS(VariableChangeParametersWrapper, Parameters)
130  public:
131  /// After deserialization, holds an instance of a subclass of VariableChangeParametersBase
132  /// controlling the behavior of a variable change. The type of the subclass is determined
133  /// by the value of the "variable change" key in the Configuration object from which this object
134  /// is deserialized.
135  PolymorphicParameter<VariableChangeParametersBase, VariableChangeFactory<MODEL>>
136  variableChangeParameters{"variable change", "Identity", this};
137 };
138 
139 // =============================================================================
140 
141 /// VariableChange factory
142 template <typename MODEL>
145 
146  public:
147  /// \brief Create and return a new variable change.
148  ///
149  /// The type of the variable change is determined by the `variable change` attribute of \p
150  /// parameters. \p parameters must be an instance of the subclass of
151  /// VariableChangeParametersBase associated with that variable change type, otherwise an
152  /// exception will be thrown.
155  /// \brief Create and return an instance of the subclass of VariableChangeParametersBase
156  /// storing parameters of variable changes of the specified type.
157  static std::unique_ptr<VariableChangeParametersBase> createParameters(const std::string &name);
158 
159  /// \brief Return the names of all variable changes that can be created by one of the
160  /// registered makers.
161  static std::vector<std::string> getMakerNames() {
162  return keys(getMakers());
163  }
164 
165  virtual ~VariableChangeFactory() = default;
166 
167  protected:
168  /// \brief Register a maker able to create variable changes of type \p name.
169  explicit VariableChangeFactory(const std::string &);
170 
171  private:
173  const VariableChangeParametersBase &) = 0;
174 
175  virtual std::unique_ptr<VariableChangeParametersBase> makeParameters() const = 0;
176 
177  static std::map < std::string, VariableChangeFactory<MODEL> * > & getMakers() {
178  static std::map < std::string, VariableChangeFactory<MODEL> * > makers_;
179  return makers_;
180  }
181 };
182 
183 // -------------------------------------------------------------------------------------------------
184 
185 template<class MODEL, class T>
187  /// Defined as T::Parameters_ if T defines a Parameters_ type; otherwise as
188  /// GenericVariableChangeParameters.
189  typedef TParameters_IfAvailableElseFallbackType_t<T, GenericVariableChangeParameters> Parameters_;
190 
192 
194  const VariableChangeParametersBase & params) override {
195  const auto &stronglyTypedParams = dynamic_cast<const Parameters_&>(params);
196  return new T(resol,
197  parametersOrConfiguration<HasParameters_<T>::value>(stronglyTypedParams));
198  }
199 
200  std::unique_ptr<VariableChangeParametersBase> makeParameters() const override {
201  return boost::make_unique<Parameters_>();
202  }
203 
204  public:
205  explicit GenericVariableChangeMaker(const std::string & name)
206  : VariableChangeFactory<MODEL>(name) {}
207 };
208 
209 
210 // -------------------------------------------------------------------------------------------------
211 
212 template<class MODEL, class T>
214  /// Defined as T::Parameters_ if T defines a Parameters_ type; otherwise as
215  /// GenericVariableChangeParameters.
216  typedef TParameters_IfAvailableElseFallbackType_t<T, GenericVariableChangeParameters> Parameters_;
217 
219 
221  const VariableChangeParametersBase & params) override {
222  const auto &stronglyTypedParams = dynamic_cast<const Parameters_&>(params);
223  return new T(resol.geometry(),
224  parametersOrConfiguration<HasParameters_<T>::value>(stronglyTypedParams));
225  }
226 
227  std::unique_ptr<VariableChangeParametersBase> makeParameters() const override {
228  return boost::make_unique<Parameters_>();
229  }
230 
231  public:
232  explicit VariableChangeMaker(const std::string & name)
233  : VariableChangeFactory<MODEL>(name) {}
234 };
235 
236 // -------------------------------------------------------------------------------------------------
237 
238 template <typename MODEL>
240  if (getMakers().find(name) != getMakers().end()) {
241  throw std::runtime_error(name + " already registered in the variable change factory.");
242  }
243  getMakers()[name] = this;
244 }
245 
246 // -------------------------------------------------------------------------------------------------
247 
248 template <typename MODEL>
250  const Geometry_ & resol, const VariableChangeParametersBase & params)
251 {
252  Log::trace() << "VariableChangeBase<MODEL>::create starting" << std::endl;
253 // Not good: should not create anything if no variable change required. YT
254  const std::string &id = params.variableChange.value();
255  typename std::map<std::string, VariableChangeFactory<MODEL>*>::iterator
256  jerr = getMakers().find(id);
257  if (jerr == getMakers().end()) {
258  throw std::runtime_error(id + " does not exist in variable change factory.");
259  }
260  GenericVariableChangeBase<MODEL> * ptr = jerr->second->make(resol, params);
261  Log::trace() << "VariableChangeBase<MODEL>::create done" << std::endl;
262  return ptr;
263 }
264 
265 // -----------------------------------------------------------------------------
266 
267 template <typename MODEL>
268 std::unique_ptr<VariableChangeParametersBase> VariableChangeFactory<MODEL>::createParameters(
269  const std::string &name) {
270  typename std::map<std::string, VariableChangeFactory<MODEL>*>::iterator it =
271  getMakers().find(name);
272  if (it == getMakers().end()) {
273  throw std::runtime_error(name + " does not exist in VariableChangeFactory");
274  }
275  return it->second->makeParameters();
276 }
277 
278 // -------------------------------------------------------------------------------------------------
279 
280 } // namespace oops
281 
282 #endif // OOPS_BASE_VARIABLECHANGEBASE_H_
virtual void changeVar(const State_ &xin, State_ &xout) const =0
change variables from state xin to xout
virtual ~GenericVariableChangeBase()=default
virtual void print(std::ostream &) const =0
Print, used for logging.
virtual void changeVarInverse(const State_ &xout, State_ &xin) const =0
inverse of changeVar, change variables back from xout to xin
TParameters_IfAvailableElseFallbackType_t< T, GenericVariableChangeParameters > Parameters_
GenericVariableChangeBase< MODEL > * make(const Geometry_ &resol, const VariableChangeParametersBase &params) override
GenericVariableChangeMaker(const std::string &name)
std::unique_ptr< VariableChangeParametersBase > makeParameters() const override
A subclass of VariableChangeParametersBase storing the values of all options in a single Configuratio...
Geometry class used in oops; subclass of interface class interface::Geometry.
State class used in oops; subclass of interface class interface::State.
Base class for MODEL-specific implementations of VariableChange class. The complete interface that ne...
virtual ~VariableChangeBase()=default
virtual void changeVar(const State_ &xin, State_ &xout) const =0
change variables from state xin to xout
void changeVar(const State< MODEL > &xin, State< MODEL > &xout) const final
virtual void changeVarInverse(const State_ &xout, State_ &xin) const =0
inverse of changeVar, change variables back from xout to xin
void changeVarInverse(const State< MODEL > &xout, State< MODEL > &xin) const final
inverse of changeVar, change variables back from xout to xin
VariableChange factory.
static std::unique_ptr< VariableChangeParametersBase > createParameters(const std::string &name)
Create and return an instance of the subclass of VariableChangeParametersBase storing parameters of v...
virtual ~VariableChangeFactory()=default
static std::map< std::string, VariableChangeFactory< MODEL > * > & getMakers()
static std::vector< std::string > getMakerNames()
Return the names of all variable changes that can be created by one of the registered makers.
static GenericVariableChangeBase< MODEL > * create(const Geometry_ &, const VariableChangeParametersBase &)
Create and return a new variable change.
VariableChangeFactory(const std::string &)
Register a maker able to create variable changes of type name.
virtual GenericVariableChangeBase< MODEL > * make(const Geometry_ &, const VariableChangeParametersBase &)=0
virtual std::unique_ptr< VariableChangeParametersBase > makeParameters() const =0
VariableChangeBase< MODEL > * make(const Geometry_ &resol, const VariableChangeParametersBase &params) override
TParameters_IfAvailableElseFallbackType_t< T, GenericVariableChangeParameters > Parameters_
VariableChangeMaker(const std::string &name)
std::unique_ptr< VariableChangeParametersBase > makeParameters() const override
Base class of classes storing parameters controlling specific variable changes.
Parameter< std::string > variableChange
Variable change type.
Contains a polymorphic parameter holding an instance of a subclass of VariableChangeParametersBase.
PolymorphicParameter< VariableChangeParametersBase, VariableChangeFactory< MODEL > > variableChangeParameters
const Geometry_ & geometry() const
Definition: FieldL95.h:22
The namespace for the main oops code.