OOPS
test/interface/ModelAuxIncrement.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 TEST_INTERFACE_MODELAUXINCREMENT_H_
12 #define TEST_INTERFACE_MODELAUXINCREMENT_H_
13 
14 #include <cmath>
15 #include <iostream>
16 #include <memory>
17 #include <string>
18 #include <vector>
19 
20 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
21 
22 #include <boost/noncopyable.hpp>
23 
24 #include "eckit/config/LocalConfiguration.h"
25 #include "eckit/testing/Test.h"
26 #include "oops/base/Geometry.h"
30 #include "oops/mpi/mpi.h"
31 #include "oops/runs/Test.h"
32 #include "oops/util/DateTime.h"
33 #include "oops/util/dot_product.h"
34 #include "test/TestEnvironment.h"
35 
36 namespace test {
37 
38 // =============================================================================
39 
40 template <typename MODEL> class ModelAuxIncrementFixture : private boost::noncopyable {
45 
46  public:
47  static const eckit::Configuration & config() {return *getInstance().conf_;}
48  static const Covariance_ & covariance() {return *getInstance().covar_;}
49  static const Geometry_ & resol() {return *getInstance().resol_;}
50 
51  private:
53  static ModelAuxIncrementFixture<MODEL> theModelAuxIncrementFixture;
54  return theModelAuxIncrementFixture;
55  }
56 
58 // Setup a geometry
59  const eckit::LocalConfiguration resolConfig(TestEnvironment::config(), "geometry");
60  resol_.reset(new Geometry_(resolConfig, oops::mpi::world()));
61 
62 // Setup a covariance matrix
63  conf_.reset(new eckit::LocalConfiguration(TestEnvironment::config(), "model aux error"));
64  covar_.reset(new Covariance_(*conf_, *resol_));
65  }
66 
67  ~ModelAuxIncrementFixture<MODEL>() {}
68 
69  std::unique_ptr<const eckit::LocalConfiguration> conf_;
70  std::unique_ptr<const Geometry_> resol_;
71  std::unique_ptr<const Covariance_> covar_;
72 };
73 
74 // =============================================================================
75 /// \brief tests constructor and print method
76 template <typename MODEL> void testModelAuxIncrementConstructor() {
77  typedef ModelAuxIncrementFixture<MODEL> Test_;
78  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
79 
80  AuxIncr_ dx(Test_::resol(), Test_::config());
81  oops::Log::test() << "Testing ModelAuxIncrement: " << dx << std::endl;
82  EXPECT(dx.norm() == 0.0);
83 }
84 
85 // -----------------------------------------------------------------------------
86 
87 template <typename MODEL> void testModelAuxIncrementCopyConstructor() {
88  typedef ModelAuxIncrementFixture<MODEL> Test_;
89  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
90 
91  AuxIncr_ dx1(Test_::resol(), Test_::config());
93 
94  AuxIncr_ dx2(dx1);
95  EXPECT(dx2.norm() > 0.0);
96  EXPECT(dx2.norm() == dx1.norm());
97 
98 // Check that the copy is equal to the original
99  dx2 -= dx1;
100  EXPECT(dx2.norm() == 0.0);
101 }
102 
103 // -----------------------------------------------------------------------------
104 
105 template <typename MODEL> void testModelAuxIncrementChangeRes() {
106  typedef ModelAuxIncrementFixture<MODEL> Test_;
107  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
108 
109  AuxIncr_ dx1(Test_::resol(), Test_::config());
111 
112  AuxIncr_ dx2(dx1, Test_::config());
113  EXPECT(dx2.norm() > 0.0);
114  EXPECT(dx2.norm() == dx1.norm());
115 
116 // Check that the copy is equal to the original
117  dx2 -= dx1;
118  EXPECT(dx2.norm() == 0.0);
119 }
120 
121 // -----------------------------------------------------------------------------
122 
123 template <typename MODEL> void testModelAuxIncrementTriangle() {
124  typedef ModelAuxIncrementFixture<MODEL> Test_;
125  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
126 
127  AuxIncr_ dx1(Test_::resol(), Test_::config());
129  AuxIncr_ dx2(Test_::resol(), Test_::config());
131 
132 // test triangle inequality
133  double dot1 = dx1.norm();
134  EXPECT(dot1 > 0.0);
135 
136  double dot2 = dx2.norm();
137  EXPECT(dot2 > 0.0);
138 
139  dx2 += dx1;
140  double dot3 = dx2.norm();
141  EXPECT(dot3 > 0.0);
142 
143  EXPECT(dot3 <= dot1 + dot2);
144 }
145 
146 // -----------------------------------------------------------------------------
147 
148 template <typename MODEL> void testModelAuxIncrementOpPlusEq() {
149  typedef ModelAuxIncrementFixture<MODEL> Test_;
150  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
151 
152  AuxIncr_ dx1(Test_::resol(), Test_::config());
154  AuxIncr_ dx2(dx1);
155 
156 // test *= and +=
157  dx2 += dx1;
158  dx1 *= 2.0;
159 
160  dx2 -= dx1;
161  EXPECT(dx2.norm() < 1e-8);
162 }
163 
164 // -----------------------------------------------------------------------------
165 
166 template <typename MODEL> void testModelAuxIncrementDotProduct() {
167  typedef ModelAuxIncrementFixture<MODEL> Test_;
168  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
169 
170  AuxIncr_ dx1(Test_::resol(), Test_::config());
172  AuxIncr_ dx2(Test_::resol(), Test_::config());
174 
175 // test symmetry of dot product
176  double zz1 = dot_product(dx1, dx2);
177  double zz2 = dot_product(dx2, dx1);
178 
179  EXPECT(zz1 == zz2);
180 }
181 
182 // -----------------------------------------------------------------------------
183 
184 template <typename MODEL> void testModelAuxIncrementZero() {
185  typedef ModelAuxIncrementFixture<MODEL> Test_;
186  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
187 
188  AuxIncr_ dx(Test_::resol(), Test_::config());
190  EXPECT(dx.norm() > 0.0);
191 
192 // test zero
193  dx->zero();
194  EXPECT(dx.norm() == 0.0);
195 }
196 
197 // -----------------------------------------------------------------------------
198 
199 template <typename MODEL> void testModelAuxIncrementAxpy() {
200  typedef ModelAuxIncrementFixture<MODEL> Test_;
201  typedef oops::ModelAuxIncrement<MODEL> AuxIncr_;
202 
203  AuxIncr_ dx1(Test_::resol(), Test_::config());
205 
206 // test axpy
207  AuxIncr_ dx2(dx1);
208  dx2.axpy(2.0, dx1);
209 
210  dx2 -= dx1;
211  dx2 -= dx1;
212  dx2 -= dx1;
213 
214  EXPECT(dx2.norm() < 1e-8);
215 }
216 
217 // =============================================================================
218 
219 template <typename MODEL> class ModelAuxIncrement : public oops::Test {
220  public:
222  virtual ~ModelAuxIncrement() {}
223 
224  private:
225  std::string testid() const override {return "test::ModelAuxIncrement<" + MODEL::name() + ">";}
226 
227  void register_tests() const override {
228  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
229  ts.emplace_back(CASE("interface/ModelAuxIncrement/testModelAuxIncrementConstructor")
230  { testModelAuxIncrementConstructor<MODEL>(); });
231  ts.emplace_back(CASE("interface/ModelAuxIncrement/testModelAuxIncrementCopyConstructor")
232  { testModelAuxIncrementCopyConstructor<MODEL>(); });
233  ts.emplace_back(CASE("interface/ModelAuxIncrement/testModelAuxIncrementChangeRes")
234  { testModelAuxIncrementChangeRes<MODEL>(); });
235  ts.emplace_back(CASE("interface/ModelAuxIncrement/testModelAuxIncrementTriangle")
236  { testModelAuxIncrementTriangle<MODEL>(); });
237  ts.emplace_back(CASE("interface/ModelAuxIncrement/testModelAuxIncrementOpPlusEq")
238  { testModelAuxIncrementOpPlusEq<MODEL>(); });
239  ts.emplace_back(CASE("interface/ModelAuxIncrement/testModelAuxIncrementDotProduct")
240  { testModelAuxIncrementDotProduct<MODEL>(); });
241  ts.emplace_back(CASE("interface/ModelAuxIncrement/testModelAuxIncrementAxpy")
242  { testModelAuxIncrementAxpy<MODEL>(); });
243  }
244 
245  void clear() const override {}
246 };
247 
248 // =============================================================================
249 
250 } // namespace test
251 
252 #endif // TEST_INTERFACE_MODELAUXINCREMENT_H_
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.
Auxiliary Error Covariance related to model, not used at the moment.
void randomize(ModelAuxIncrement_ &) const
randomize the values in the ModelAuxIncrement
Auxiliary Increment related to model, not used at the moment.
std::unique_ptr< const Geometry_ > resol_
std::unique_ptr< const eckit::LocalConfiguration > conf_
std::unique_ptr< const Covariance_ > covar_
static ModelAuxIncrementFixture< MODEL > & getInstance()
oops::ModelAuxIncrement< MODEL > AuxIncr_
oops::ModelAuxCovariance< MODEL > Covariance_
static const eckit::Configuration & config()
oops::ModelAuxControl< MODEL > ModelAux_
std::string testid() const override
static const eckit::Configuration & config()
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
Definition: oops/mpi/mpi.cc:84
void testModelAuxIncrementCopyConstructor()
void testModelAuxIncrementConstructor()
tests constructor and print method
void testModelAuxIncrementDotProduct()
CASE("test_linearmodelparameterswrapper_valid_name")