OOPS
generic/LocalizationBase.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 2020-2020 UCAR
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  * In applying this licence, ECMWF does not waive the privileges and immunities
8  * granted to it by virtue of its status as an intergovernmental organisation nor
9  * does it submit to any jurisdiction.
10  */
11 
12 #ifndef OOPS_GENERIC_LOCALIZATIONBASE_H_
13 #define OOPS_GENERIC_LOCALIZATIONBASE_H_
14 
15 #include <map>
16 #include <memory>
17 #include <string>
18 
19 #include <boost/noncopyable.hpp>
20 
21 #include "oops/base/Geometry.h"
22 #include "oops/base/Increment.h"
23 #include "oops/mpi/mpi.h"
24 #include "oops/util/Logger.h"
25 #include "oops/util/Printable.h"
26 
27 namespace eckit {
28  class Configuration;
29 }
30 
31 namespace oops {
32 
33 // -----------------------------------------------------------------------------
34 /// \brief Base class for generic implementations of model-space localization.
35 /// Use this class as a base class for generic implementations,
36 /// and interface::LocalizationBase as a base class for MODEL-specific implementations.
37 ///
38 /// Note: generic implementations need to provide a constructor with the following signature:
39 ///
40 /// LocalizationBase(const Geometry<MODEL> &,
41 /// const eckit::Configuration &);
42 template <typename MODEL>
43 class LocalizationBase : public util::Printable,
44  private boost::noncopyable {
46 
47  public:
48  LocalizationBase() = default;
49  virtual ~LocalizationBase() = default;
50 
51  /// Randomize \p dx and apply 3D localization
52  virtual void randomize(Increment_ & dx) const = 0;
53  /// Apply 3D localization to \p dx
54  virtual void multiply(Increment_ & dx) const = 0;
55 };
56 
57 // -----------------------------------------------------------------------------
58 
59 /// Localization Factory
60 template <typename MODEL>
63  public:
64  /// \brief Create and return a new Localization.
65  ///
66  /// The Localization type is determined by the "localization method" entry of
67  /// \p config.
68  static std::unique_ptr<LocalizationBase<MODEL>> create(const Geometry_ &,
69  const eckit::Configuration & config);
70  virtual ~LocalizationFactory() = default;
71  protected:
72  /// \brief Register a maker able to create Localizations of type \p name.
73  explicit LocalizationFactory(const std::string & name);
74  private:
75  virtual std::unique_ptr<LocalizationBase<MODEL>> make(const Geometry_ &,
76  const eckit::Configuration &) = 0;
77  static std::map < std::string, LocalizationFactory<MODEL> * > & getMakers() {
78  static std::map < std::string, LocalizationFactory<MODEL> * > makers_;
79  return makers_;
80  }
81 };
82 
83 // -----------------------------------------------------------------------------
84 /// \brief A subclass of LocalizationFactory able to create instances of T (a concrete
85 /// subclass of LocalizationBase<MODEL>). Passes Geometry<MODEL> to the constructor of T.
86 template<class MODEL, class T>
87 class LocalizationMaker : public LocalizationFactory<MODEL> {
89  std::unique_ptr<LocalizationBase<MODEL>> make(const Geometry_ & geometry,
90  const eckit::Configuration & conf) override
91  { return std::make_unique<T>(geometry, conf); }
92  public:
93  explicit LocalizationMaker(const std::string & name) : LocalizationFactory<MODEL>(name) {}
94 };
95 
96 // -----------------------------------------------------------------------------
97 
98 template <typename MODEL>
100  if (getMakers().find(name) != getMakers().end()) {
101  throw std::runtime_error(name + " already registered in localization factory.");
102  }
103  getMakers()[name] = this;
104 }
105 
106 // -----------------------------------------------------------------------------
107 
108 template <typename MODEL>
109 std::unique_ptr<LocalizationBase<MODEL>>
111  const eckit::Configuration & conf) {
112  Log::trace() << "LocalizationBase<MODEL>::create starting" << std::endl;
113  const std::string id = conf.getString("localization method");
114  typename std::map<std::string, LocalizationFactory<MODEL>*>::iterator
115  jloc = getMakers().find(id);
116  if (jloc == getMakers().end()) {
117  Log::error() << id << " does not exist in localization factory." << std::endl;
118  Log::error() << "Obs Localization Factory has " << getMakers().size()
119  << " elements:" << std::endl;
120  for (typename std::map<std::string, LocalizationFactory<MODEL>*>::const_iterator
121  jj = getMakers().begin(); jj != getMakers().end(); ++jj) {
122  Log::error() << "A " << jj->first << " Localization" << std::endl;
123  }
124  throw std::runtime_error(id + " does not exist in localization factory.");
125  }
126  std::unique_ptr<LocalizationBase<MODEL>> ptr(jloc->second->make(geometry, conf));
127  Log::trace() << "LocalizationBase<MODEL>::create done" << std::endl;
128  return ptr;
129 }
130 
131 // -----------------------------------------------------------------------------
132 
133 } // namespace oops
134 
135 #endif // OOPS_GENERIC_LOCALIZATIONBASE_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
Base class for generic implementations of model-space localization. Use this class as a base class fo...
virtual void randomize(Increment_ &dx) const =0
Randomize dx and apply 3D localization.
virtual void multiply(Increment_ &dx) const =0
Apply 3D localization to dx.
virtual ~LocalizationBase()=default
virtual ~LocalizationFactory()=default
LocalizationFactory(const std::string &name)
Register a maker able to create Localizations of type name.
static std::unique_ptr< LocalizationBase< MODEL > > create(const Geometry_ &, const eckit::Configuration &config)
Create and return a new Localization.
virtual std::unique_ptr< LocalizationBase< MODEL > > make(const Geometry_ &, const eckit::Configuration &)=0
static std::map< std::string, LocalizationFactory< MODEL > * > & getMakers()
A subclass of LocalizationFactory able to create instances of T (a concrete subclass of LocalizationB...
std::unique_ptr< LocalizationBase< MODEL > > make(const Geometry_ &geometry, const eckit::Configuration &conf) override
LocalizationMaker(const std::string &name)
int error
Definition: compare.py:168
Definition: FieldL95.h:22
The namespace for the main oops code.