OOPS
generic/ModelBase.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-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 
8 #ifndef OOPS_GENERIC_MODELBASE_H_
9 #define OOPS_GENERIC_MODELBASE_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/Duration.h"
24 #include "oops/util/Logger.h"
25 #include "oops/util/parameters/ConfigurationParameter.h"
26 #include "oops/util/parameters/HasParameters_.h"
27 #include "oops/util/parameters/OptionalParameter.h"
28 #include "oops/util/parameters/Parameters.h"
29 #include "oops/util/parameters/ParametersOrConfiguration.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 /// \brief Base class for generic implementations of the forecasting models.
42 /// Use this class as a base class for generic implementations,
43 /// and interface::ModelBase as a base class for MODEL-specific implementations.
44 ///
45 /// Note: implementations of this interface can opt to extract their settings either from
46 /// a Configuration object or from a subclass of ModelParametersBase.
47 ///
48 /// In the former case, they should provide a constructor with the following signature:
49 ///
50 /// ModelBase(const Geometry_ &, const eckit::Configuration &);
51 ///
52 /// In the latter case, the implementer should first define a subclass of ModelParametersBase
53 /// holding the settings of the model in question. The implementation of the ModelBase interface
54 /// should then typedef `Parameters_` to the name of that subclass and provide a constructor with
55 /// the following signature:
56 ///
57 /// ModelBase(const Geometry_ &, const Parameters_ &);
58 ///
59 template <typename MODEL>
60 class ModelBase : public util::Printable,
61  private boost::noncopyable {
64 
65  public:
66  static const std::string classname() {return "oops::ModelBase";}
67 
68  ModelBase() = default;
69  virtual ~ModelBase() = default;
70 
71  /// \brief Forecast initialization, called before every forecast run
72  virtual void initialize(State_ &) const = 0;
73  /// \brief Forecast "step", called during forecast run; updates state to the next time
74  virtual void step(State_ &, const ModelAux_ &) const = 0;
75  /// \brief Forecast finalization; called after each forecast run
76  virtual void finalize(State_ &) const = 0;
77 
78  /// \brief Time step for running Model's forecast in oops (frequency with which the
79  /// State will be updated)
80  virtual const util::Duration & timeResolution() const = 0;
81  /// \brief Model variables (only used in 4DVar)
82  virtual const oops::Variables & variables() const = 0;
83 
84  private:
85  /// \brief Print; used for logging
86  virtual void print(std::ostream &) const = 0;
87 };
88 
89 // =============================================================================
90 
91 template <typename MODEL>
92 class ModelFactory;
93 
94 // -----------------------------------------------------------------------------
95 
96 /// \brief Base class for classes storing model-specific parameters.
97 class ModelParametersBase : public Parameters {
98  OOPS_ABSTRACT_PARAMETERS(ModelParametersBase, Parameters)
99  public:
100  /// \brief Model name.
101  ///
102  /// \note This parameter is marked as optional because it is only required in certain
103  /// circumstances (e.g. when model parameters are deserialized into a ModelParametersWrapper
104  /// and used by ModelFactory to instantiate a model whose type is determined at runtime), but
105  /// not others (e.g. in tests written with a particular model in mind). ModelParametersWrapper
106  /// will throw an exception if this parameter is not provided.
107  OptionalParameter<std::string> name{"name", this};
108 };
109 
110 // -----------------------------------------------------------------------------
111 
112 /// \brief A subclass of ModelParametersBase storing the values of all options in a
113 /// single Configuration object.
114 ///
115 /// This object can be accessed by calling the value() method of the \p config member variable.
116 ///
117 /// The ConfigurationParameter class does not perform any parameter validation; models using
118 /// GenericModelParameters should therefore ideally be refactored, replacing this class with a
119 /// dedicated subclass of ModelParametersBase storing each parameter in a separate
120 /// (Optional/Required)Parameter object.
122  OOPS_CONCRETE_PARAMETERS(GenericModelParameters, ModelParametersBase)
123  public:
124  ConfigurationParameter config{this};
125 };
126 
127 // -----------------------------------------------------------------------------
128 
129 /// \brief Contains a polymorphic parameter holding an instance of a subclass of
130 /// ModelParametersBase.
131 template <typename MODEL>
132 class ModelParametersWrapper : public Parameters {
133  OOPS_CONCRETE_PARAMETERS(ModelParametersWrapper, Parameters)
134  public:
135  /// After deserializtion, holds an instance of a subclass of ModelParametersBase controlling
136  /// the behavior of a model. The type of the subclass is determined by the value of the "name"
137  /// key in the Configuration object from which this object is deserialized.
138  RequiredPolymorphicParameter<ModelParametersBase, ModelFactory<MODEL>>
139  modelParameters{"name", this};
140 };
141 
142 // =============================================================================
143 
144 /// Model factory
145 template <typename MODEL>
148 
149  public:
150  /// \brief Create and return a new model.
151  ///
152  /// The model's type is determined by the \c name attribute of \p parameters.
153  /// \p parameters must be an instance of the subclass of ModelParametersBase
154  /// associated with that model type, otherwise an exception will be thrown.
155  static ModelBase<MODEL> * create(const Geometry_ &,
156  const ModelParametersBase & parameters);
157 
158  /// \brief Create and return an instance of the subclass of ModelParametersBase
159  /// storing parameters of models of the specified type.
160  static std::unique_ptr<ModelParametersBase> createParameters(const std::string &name);
161 
162  /// \brief Return the names of all models that can be created by one of the registered makers.
163  static std::vector<std::string> getMakerNames() {
164  return keys(getMakers());
165  }
166 
167  virtual ~ModelFactory() = default;
168 
169  protected:
170  /// \brief Register a maker able to create models of type \p name.
171  explicit ModelFactory(const std::string & name);
172 
173  private:
174  virtual ModelBase<MODEL> * make(const Geometry_ &, const ModelParametersBase &) = 0;
175 
176  virtual std::unique_ptr<ModelParametersBase> makeParameters() const = 0;
177 
178  static std::map < std::string, ModelFactory<MODEL> * > & getMakers() {
179  static std::map < std::string, ModelFactory<MODEL> * > makers_;
180  return makers_;
181  }
182 };
183 
184 // -----------------------------------------------------------------------------
185 
186 /// \brief A subclass of ModelFactory able to create instances of T (a concrete subclass of
187 /// ModelBase<MODEL>). Passes Geometry<MODEL> to the constructor of T.
188 template<class MODEL, class T>
189 class ModelMaker : public ModelFactory<MODEL> {
190  private:
191  /// Defined as T::Parameters_ if T defines a Parameters_ type; otherwise as
192  /// GenericModelParameters.
193  typedef TParameters_IfAvailableElseFallbackType_t<T, GenericModelParameters> Parameters_;
194 
195  public:
197 
198  explicit ModelMaker(const std::string & name) : ModelFactory<MODEL>(name) {}
199 
201  const ModelParametersBase & parameters) override {
202  Log::trace() << "ModelBase<MODEL>::make starting" << std::endl;
203  const auto &stronglyTypedParameters = dynamic_cast<const Parameters_&>(parameters);
204  return new T(geom,
205  parametersOrConfiguration<HasParameters_<T>::value>(stronglyTypedParameters));
206  }
207 
208  std::unique_ptr<ModelParametersBase> makeParameters() const override {
209  return boost::make_unique<Parameters_>();
210  }
211 };
212 
213 // -----------------------------------------------------------------------------
214 
215 template <typename MODEL>
216 ModelFactory<MODEL>::ModelFactory(const std::string & name) {
217  if (getMakers().find(name) != getMakers().end()) {
218  throw std::runtime_error(name + " already registered in the model factory.");
219  }
220  getMakers()[name] = this;
221 }
222 
223 // -----------------------------------------------------------------------------
224 
225 template <typename MODEL>
227  const ModelParametersBase & parameters) {
228  Log::trace() << "ModelFactory<MODEL>::create starting" << std::endl;
229  const std::string &id = parameters.name.value().value();
230  typename std::map<std::string, ModelFactory<MODEL>*>::iterator
231  jerr = getMakers().find(id);
232  if (jerr == getMakers().end()) {
233  throw std::runtime_error(id + " does not exist in the model factory");
234  }
235  ModelBase<MODEL> * ptr = jerr->second->make(geom, parameters);
236  Log::trace() << "ModelFactory<MODEL>::create done" << std::endl;
237  return ptr;
238 }
239 
240 // -----------------------------------------------------------------------------
241 
242 template <typename MODEL>
243 std::unique_ptr<ModelParametersBase> ModelFactory<MODEL>::createParameters(
244  const std::string &name) {
245  Log::trace() << "ModelFactory<MODEL>::createParameters starting" << std::endl;
246  typename std::map<std::string, ModelFactory<MODEL>*>::iterator it = getMakers().find(name);
247  if (it == getMakers().end()) {
248  throw std::runtime_error(name + " does not exist in the model factory");
249  }
250  return it->second->makeParameters();
251 }
252 
253 // -----------------------------------------------------------------------------
254 
255 } // namespace oops
256 
257 #endif // OOPS_GENERIC_MODELBASE_H_
A subclass of ModelParametersBase storing the values of all options in a single Configuration object.
ConfigurationParameter config
Geometry class used in oops; subclass of interface class interface::Geometry.
Auxiliary state related to model (could be e.g. model bias), not used at the moment.
Base class for generic implementations of the forecasting models. Use this class as a base class for ...
State< MODEL > State_
virtual void initialize(State_ &) const =0
Forecast initialization, called before every forecast run.
virtual const util::Duration & timeResolution() const =0
Time step for running Model's forecast in oops (frequency with which the State will be updated)
ModelBase()=default
virtual const oops::Variables & variables() const =0
Model variables (only used in 4DVar)
virtual void step(State_ &, const ModelAux_ &) const =0
Forecast "step", called during forecast run; updates state to the next time.
ModelAuxControl< MODEL > ModelAux_
static const std::string classname()
virtual void finalize(State_ &) const =0
Forecast finalization; called after each forecast run.
virtual void print(std::ostream &) const =0
Print; used for logging.
virtual ~ModelBase()=default
virtual ModelBase< MODEL > * make(const Geometry_ &, const ModelParametersBase &)=0
ModelFactory(const std::string &name)
Register a maker able to create models of type name.
virtual ~ModelFactory()=default
virtual std::unique_ptr< ModelParametersBase > makeParameters() const =0
static std::vector< std::string > getMakerNames()
Return the names of all models that can be created by one of the registered makers.
static std::unique_ptr< ModelParametersBase > createParameters(const std::string &name)
Create and return an instance of the subclass of ModelParametersBase storing parameters of models of ...
static std::map< std::string, ModelFactory< MODEL > * > & getMakers()
static ModelBase< MODEL > * create(const Geometry_ &, const ModelParametersBase &parameters)
Create and return a new model.
Geometry< MODEL > Geometry_
A subclass of ModelFactory able to create instances of T (a concrete subclass of ModelBase<MODEL>)....
ModelBase< MODEL > * make(const Geometry_ &geom, const ModelParametersBase &parameters) override
Geometry< MODEL > Geometry_
std::unique_ptr< ModelParametersBase > makeParameters() const override
TParameters_IfAvailableElseFallbackType_t< T, GenericModelParameters > Parameters_
ModelMaker(const std::string &name)
Base class for classes storing model-specific parameters.
OptionalParameter< std::string > name
Model name.
Contains a polymorphic parameter holding an instance of a subclass of ModelParametersBase.
RequiredPolymorphicParameter< ModelParametersBase, ModelFactory< MODEL > > modelParameters
State class used in oops; subclass of interface class interface::State.
Definition: FieldL95.h:22
The namespace for the main oops code.