OOPS
test/interface/Increment.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 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 TEST_INTERFACE_INCREMENT_H_
13 #define TEST_INTERFACE_INCREMENT_H_
14 
15 #include <cmath>
16 #include <iostream>
17 #include <memory>
18 #include <string>
19 #include <vector>
20 
21 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
22 
23 #include <boost/noncopyable.hpp>
24 
25 #include "atlas/field.h"
26 #include "eckit/config/LocalConfiguration.h"
27 #include "eckit/testing/Test.h"
28 #include "oops/base/Geometry.h"
29 #include "oops/base/Increment.h"
30 #include "oops/base/State.h"
31 #include "oops/base/Variables.h"
32 #include "oops/mpi/mpi.h"
33 #include "oops/runs/Test.h"
34 #include "oops/util/DateTime.h"
35 #include "oops/util/dot_product.h"
36 #include "oops/util/Logger.h"
37 #include "test/TestEnvironment.h"
38 
39 
40 namespace test {
41 
42 // =============================================================================
43 
44 template <typename MODEL> class IncrementFixture : private boost::noncopyable {
46 
47  public:
48  static const Geometry_ & resol() {return *getInstance().resol_;}
49  static const oops::Variables & ctlvars() {return *getInstance().ctlvars_;}
50  static const util::DateTime & time() {return *getInstance().time_;}
51  static const double & tolerance() {return getInstance().tolerance_;}
52  static const int & skipAtlas() {return getInstance().skipAtlas_;}
53  static const eckit::Configuration & test() {return *getInstance().test_;}
54  static void reset() {
55  getInstance().time_.reset();
56  getInstance().ctlvars_.reset();
57  getInstance().test_.reset();
58  getInstance().resol_.reset();
59  }
60 
61  private:
63  static IncrementFixture<MODEL> theIncrementFixture;
64  return theIncrementFixture;
65  }
66 
68 // Setup a geometry
69  const eckit::LocalConfiguration resolConfig(TestEnvironment::config(), "geometry");
70  resol_.reset(new Geometry_(resolConfig, oops::mpi::world()));
71 
72  ctlvars_.reset(new oops::Variables(TestEnvironment::config(), "inc variables"));
73 
74  const double tol_default = 1e-8;
75  test_.reset(new eckit::LocalConfiguration(TestEnvironment::config(), "increment test"));
76  time_.reset(new util::DateTime(test_->getString("date")));
77  tolerance_ = test_->getDouble("tolerance", tol_default);
78  if (tolerance_ > tol_default) {
79  oops::Log::warning() <<
80  "Warning: Increment norm tolerance greater than 1e-8 "
81  "may not be suitable for certain solvers." <<
82  std::endl; }
83  skipAtlas_ = test_->getInt("skip atlas", 0);
84  }
85 
86  ~IncrementFixture<MODEL>() {}
87 
88  std::unique_ptr<Geometry_> resol_;
89  std::unique_ptr<oops::Variables> ctlvars_;
90  std::unique_ptr<const eckit::LocalConfiguration> test_;
91  double tolerance_;
93  std::unique_ptr<util::DateTime> time_;
94  std::unique_ptr<bool> skipAccumTest_;
95  std::unique_ptr<bool> skipDiffTest_;
96 };
97 
98 // =============================================================================
99 /// \brief tests Increment constructor and print method
100 template <typename MODEL> void testIncrementConstructor() {
101  typedef IncrementFixture<MODEL> Test_;
102  typedef oops::Increment<MODEL> Increment_;
103 
104  Increment_ dx(Test_::resol(), Test_::ctlvars(), Test_::time());
105  oops::Log::test() << "Printing zero increment: " << dx << std::endl;
106 
107  EXPECT(dx.norm() == 0.0);
108 }
109 
110 // -----------------------------------------------------------------------------
111 
112 template <typename MODEL> void testIncrementCopyConstructor() {
113  typedef IncrementFixture<MODEL> Test_;
114  typedef oops::Increment<MODEL> Increment_;
115 
116  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
117  dx1.random();
118  oops::Log::test() << "Printing random increment: " << dx1 << std::endl;
119 
120  EXPECT(dx1.norm() > 0.0);
121 
122  Increment_ dx2(dx1);
123  EXPECT(dx2.norm() > 0.0);
124 
125 // Check that the copy is equal to the original
126  dx2 -= dx1;
127  EXPECT(dx2.norm() == 0.0);
128 }
129 
130 // -----------------------------------------------------------------------------
131 
132 template <typename MODEL> void testIncrementCopyBoolConstructor() {
133  typedef IncrementFixture<MODEL> Test_;
134  typedef oops::Increment<MODEL> Increment_;
135 
136  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
137  dx1.random();
138  EXPECT(dx1.norm() > 0.0);
139 
140  // Test with copy set to true
141  Increment_ dx2(dx1, true);
142  EXPECT(dx2.norm() == dx1.norm());
143 
144  // Test with copy set to false
145  Increment_ dx3(dx1, false);
146  EXPECT(dx3.norm() == 0.0);
147 }
148 
149 // -----------------------------------------------------------------------------
150 
151 template <typename MODEL> void testIncrementChangeResConstructor() {
152  typedef IncrementFixture<MODEL> Test_;
153  typedef oops::Increment<MODEL> Increment_;
154 
155  // For now this is just a copy and not a change res, would require config changes
156  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
157  Increment_ dx2(Test_::resol(), dx1);
158 
159  // Check they are same. Should be replaced with change res and check they are different
160  EXPECT(dx2.norm() == dx1.norm());
161 }
162 
163 // -----------------------------------------------------------------------------
164 
165 template <typename MODEL> void testIncrementTriangle() {
166  typedef IncrementFixture<MODEL> Test_;
167  typedef oops::Increment<MODEL> Increment_;
168 
169  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
170  dx1.random();
171  Increment_ dx2(Test_::resol(), Test_::ctlvars(), Test_::time());
172  dx2.random();
173 
174 // test triangle inequality
175  double dot1 = dx1.norm();
176  EXPECT(dot1 > 0.0);
177 
178  double dot2 = dx2.norm();
179  EXPECT(dot2 > 0.0);
180 
181  dx2 += dx1;
182  double dot3 = dx2.norm();
183  EXPECT(dot3 > 0.0);
184 
185  EXPECT(dot3 <= dot1 + dot2);
186 }
187 
188 // -----------------------------------------------------------------------------
189 
190 template <typename MODEL> void testIncrementOpPlusEq() {
191  typedef IncrementFixture<MODEL> Test_;
192  typedef oops::Increment<MODEL> Increment_;
193 
194  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
195  dx1.random();
196  Increment_ dx2(dx1);
197 
198 // test *= and +=
199  dx2 += dx1;
200  dx1 *= 2.0;
201 
202  dx2 -= dx1;
203  EXPECT(dx2.norm() < Test_::tolerance());
204 }
205 
206 // -----------------------------------------------------------------------------
207 
208 template <typename MODEL> void testIncrementDotProduct() {
209  typedef IncrementFixture<MODEL> Test_;
210  typedef oops::Increment<MODEL> Increment_;
211 
212  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
213  dx1.random();
214  Increment_ dx2(Test_::resol(), Test_::ctlvars(), Test_::time());
215  dx2.random();
216 
217 // test symmetry of dot product
218  double zz1 = dot_product(dx1, dx2);
219  double zz2 = dot_product(dx2, dx1);
220 
221  EXPECT(zz1 == zz2);
222 }
223 
224 // -----------------------------------------------------------------------------
225 
226 template <typename MODEL> void testIncrementZero() {
227  typedef IncrementFixture<MODEL> Test_;
228  typedef oops::Increment<MODEL> Increment_;
229 
230  Increment_ dx(Test_::resol(), Test_::ctlvars(), Test_::time());
231  dx.random();
232  EXPECT(dx.norm() > 0.0);
233 
234 // test zero
235  dx.zero();
236  EXPECT(dx.norm() == 0.0);
237  EXPECT(dx.validTime() == Test_::time());
238 
239 // Create a new time one hour in the future
240  util::Duration onehour(3600);
241  util::DateTime newTime(Test_::time().toString());
242  newTime+=onehour;
243 
244 // Confirm zero with setting new datetime works
245  dx.random();
246  dx.zero(newTime);
247  EXPECT(dx.norm() == 0.0);
248  EXPECT(dx.validTime() == newTime);
249 }
250 
251 // -----------------------------------------------------------------------------
252 
253 template <typename MODEL> void testIncrementAxpy() {
254  typedef IncrementFixture<MODEL> Test_;
255  typedef oops::Increment<MODEL> Increment_;
256 
257  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
258  dx1.random();
259 
260 // test axpy
261  Increment_ dx2(dx1);
262  dx2.axpy(2.0, dx1);
263 
264  dx2 -= dx1;
265  dx2 -= dx1;
266  dx2 -= dx1;
267 
268  EXPECT(dx2.norm() < Test_::tolerance());
269 }
270 
271 // -----------------------------------------------------------------------------
272 
273 template <typename MODEL> void testIncrementAccum() {
274  typedef IncrementFixture<MODEL> Test_;
275  typedef oops::Increment<MODEL> Increment_;
276  typedef oops::State<MODEL> State_;
277 
278  // Option to skip test
279  bool skipTest = Test_::test().getBool("skip accum test", false);
280  if (skipTest) {
281  oops::Log::warning() << "Skipping Increment.accum test";
282  return;
283  }
284 
285  // Create two different random increments
286  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
287  dx1.random();
288 
289  Increment_ dx2(Test_::resol(), Test_::ctlvars(), Test_::time());
290  dx2.random();
291 
292  Increment_ diff(dx1);
293  diff -= dx2;
294  EXPECT(diff.norm() != 0.0);
295 
296  // Create a state that is equal to dx2
297  State_ x(Test_::resol(), Test_::ctlvars(), Test_::time());
298  x.zero();
299  x += dx2;
300 
301  // Create copy of dx1 to test against axpy
302  Increment_ dx3(dx1);
303 
304  // Test accum
305  dx1.accumul(2.0, x);
306 
307  // Use axpy for reference
308  dx3.axpy(2.0, dx2);
309 
310  // Check axpy did something
311  diff = dx3;
312  diff -= dx2;
313  EXPECT(diff.norm() != 0.0);
314 
315  // Check accumul matches axpy
316  diff = dx1;
317  diff -= dx3;
318  EXPECT(diff.norm() == 0.0);
319 }
320 
321 // -----------------------------------------------------------------------------
322 
323 template <typename MODEL> void testIncrementSerialize() {
324  typedef IncrementFixture<MODEL> Test_;
325  typedef oops::Increment<MODEL> Increment_;
326 
327 // Create two random increments
328  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
329  dx1.random();
330 
331  util::DateTime tt(Test_::time() + util::Duration("PT15H"));
332  Increment_ dx2(Test_::resol(), Test_::ctlvars(), tt);
333 
334 // Test serialize-deserialize
335  std::vector<double> vect;
336  dx1.serialize(vect);
337  EXPECT(vect.size() == dx1.serialSize());
338 
339  size_t index = 0;
340  dx2.deserialize(vect, index);
341  EXPECT(index == dx1.serialSize());
342  EXPECT(index == dx2.serialSize());
343 
344  dx1.serialize(vect);
345  EXPECT(vect.size() == dx1.serialSize() * 2);
346 
347  if (dx1.serialSize() > 0) { // until all models have implemented serialize
348  EXPECT(dx1.norm() > 0.0);
349  EXPECT(dx2.norm() > 0.0);
350  EXPECT(dx2.validTime() == Test_::time());
351 
352  dx2 -= dx1;
353  EXPECT(dx2.norm() == 0.0);
354  }
355 }
356 
357 // -----------------------------------------------------------------------------
358 
359 template <typename MODEL> void testIncrementAtlas() {
360  typedef IncrementFixture<MODEL> Test_;
361  typedef oops::Increment<MODEL> Increment_;
362 
363  if (Test_::skipAtlas() == 0) {
364  // Create random increment and copy it
365  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
366  dx1.random();
367  Increment_ dx2(dx1);
368 
369  // Create ATLAS fieldset
370  std::unique_ptr<atlas::FieldSet> atlasFieldSet(new atlas::FieldSet());
371 
372  // Test setAtlas-toAtlas-fromAtlas
373  dx1.setAtlas(atlasFieldSet.get());
374  dx1.toAtlas(atlasFieldSet.get());
375  dx1.zero();
376  dx1.fromAtlas(atlasFieldSet.get());
377  dx2 -= dx1;
378  EXPECT(dx2.norm() == 0.0);
379  }
380 }
381 
382 // -----------------------------------------------------------------------------
383 
384 template <typename MODEL> void testIncrementDiff() {
385  typedef IncrementFixture<MODEL> Test_;
386  typedef oops::Increment<MODEL> Increment_;
387  typedef oops::State<MODEL> State_;
388 
389  // Option to skip test
390  bool skipTest = Test_::test().getBool("skip diff test", false);
391  if (skipTest) {
392  oops::Log::warning() << "Skipping Increment.diff test";
393  return;
394  }
395 
396  // Create two states to diff
397  State_ x1(Test_::resol(), Test_::ctlvars(), Test_::time());
398  State_ x2(Test_::resol(), Test_::ctlvars(), Test_::time());
399 
400  // Create two different random increments
401  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
402  dx1.random();
403 
404  Increment_ dx2(Test_::resol(), Test_::ctlvars(), Test_::time());
405  dx2.random();
406 
407  // Create some different increments
408  Increment_ diff1(dx1);
409  Increment_ diff2(Test_::resol(), Test_::ctlvars(), Test_::time());
410  diff1 -= dx2;
411  EXPECT(diff1.norm() != 0.0);
412 
413  // Fill states with some different random values
414  x1.zero();
415  x1+=dx1;
416  x2.zero();
417  x2+=dx2;
418 
419  // Use difference of states to compute difference
420  diff2.diff(x1, x2);
421 
422  // Compare difference with -= operator
423  EXPECT(diff1.norm() == diff2.norm());
424 }
425 
426 // -----------------------------------------------------------------------------
427 
428 template <typename MODEL> void testIncrementTime() {
429  typedef IncrementFixture<MODEL> Test_;
430  typedef oops::Increment<MODEL> Increment_;
431 
432  // Create an increment
433  Increment_ dx(Test_::resol(), Test_::ctlvars(), Test_::time());
434 
435  // Move increment time forward by one hour
436  util::Duration onehour(3600);
437  dx.updateTime(onehour);
438 
439  // Confirm new valid time and that validTime function works
440  util::DateTime newTime(Test_::time().toString());
441  newTime+=onehour;
442  EXPECT(dx.validTime() == newTime);
443 }
444 
445 // -----------------------------------------------------------------------------
446 
447 template <typename MODEL> void testIncrementSchur() {
448  typedef IncrementFixture<MODEL> Test_;
449  typedef oops::Increment<MODEL> Increment_;
450 
451  // Create two identical random increments
452  Increment_ dx1(Test_::resol(), Test_::ctlvars(), Test_::time());
453  dx1.random();
454  Increment_ dx2(dx1);
455  EXPECT(dx1.norm() == dx2.norm());
456 
457  // Compute schur product
458  dx1.schur_product_with(dx2);
459 
460  // Checks
461  EXPECT(dx1.norm() != dx2.norm());
462 
463  // Pass through zeros
464  dx1.random();
465  dx2.zero();
466  dx1.schur_product_with(dx2);
467  EXPECT(dx1.norm() == 0.0);
468 
469  // Pass through ones
470  dx1.random();
471  Increment_ dx1in(dx1);
472  dx2.ones();
473  dx1.schur_product_with(dx2);
474  EXPECT(dx1.norm() == dx1in.norm());
475 }
476 
477 // =============================================================================
478 
479 template <typename MODEL>
480 class Increment : public oops::Test {
481  public:
484 
485  private:
486  std::string testid() const override {return "test::Increment<" + MODEL::name() + ">";}
487 
488  void register_tests() const override {
489  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
490 
491  ts.emplace_back(CASE("interface/Increment/testIncrementConstructor")
492  { testIncrementConstructor<MODEL>(); });
493  ts.emplace_back(CASE("interface/Increment/testIncrementCopyConstructor")
494  { testIncrementCopyConstructor<MODEL>(); });
495  ts.emplace_back(CASE("interface/Increment/testIncrementCopyBoolConstructor")
496  { testIncrementCopyBoolConstructor<MODEL>(); });
497  ts.emplace_back(CASE("interface/Increment/testIncrementChangeResConstructor")
498  { testIncrementChangeResConstructor<MODEL>(); });
499  ts.emplace_back(CASE("interface/Increment/testIncrementTriangle")
500  { testIncrementTriangle<MODEL>(); });
501  ts.emplace_back(CASE("interface/Increment/testIncrementOpPlusEq")
502  { testIncrementOpPlusEq<MODEL>(); });
503  ts.emplace_back(CASE("interface/Increment/testIncrementDotProduct")
504  { testIncrementDotProduct<MODEL>(); });
505  ts.emplace_back(CASE("interface/Increment/testIncrementAxpy")
506  { testIncrementAxpy<MODEL>(); });
507  ts.emplace_back(CASE("interface/Increment/testIncrementAccum")
508  { testIncrementAccum<MODEL>(); });
509  ts.emplace_back(CASE("interface/Increment/testIncrementDiff")
510  { testIncrementDiff<MODEL>(); });
511  ts.emplace_back(CASE("interface/Increment/testIncrementZero")
512  { testIncrementZero<MODEL>(); });
513  ts.emplace_back(CASE("interface/Increment/testIncrementTime")
514  { testIncrementTime<MODEL>(); });
515  ts.emplace_back(CASE("interface/Increment/testIncrementSchur")
516  { testIncrementSchur<MODEL>(); });
517  ts.emplace_back(CASE("interface/Increment/testIncrementSerialize")
518  { testIncrementSerialize<MODEL>(); });
519  ts.emplace_back(CASE("interface/Increment/testIncrementAtlas")
520  { testIncrementAtlas<MODEL>(); });
521  }
522 
523  void clear() const override {}
524 };
525 
526 // =============================================================================
527 
528 } // namespace test
529 
530 #endif // TEST_INTERFACE_INCREMENT_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
State class used in oops; subclass of interface class interface::State.
std::unique_ptr< bool > skipAccumTest_
static const eckit::Configuration & test()
static const double & tolerance()
static const int & skipAtlas()
static IncrementFixture< MODEL > & getInstance()
std::unique_ptr< const eckit::LocalConfiguration > test_
static const Geometry_ & resol()
std::unique_ptr< Geometry_ > resol_
std::unique_ptr< bool > skipDiffTest_
static const oops::Variables & ctlvars()
static const util::DateTime & time()
std::unique_ptr< oops::Variables > ctlvars_
std::unique_ptr< util::DateTime > time_
oops::Geometry< MODEL > Geometry_
std::string testid() const override
void clear() const override
void register_tests() 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 testIncrementCopyBoolConstructor()
void testIncrementChangeResConstructor()
void testIncrementSchur()
void testIncrementSerialize()
void testIncrementDiff()
void testIncrementCopyConstructor()
void testIncrementZero()
void testIncrementAxpy()
void testIncrementTriangle()
void testIncrementAccum()
void testIncrementOpPlusEq()
void testIncrementDotProduct()
void testIncrementConstructor()
tests Increment constructor and print method
void testIncrementAtlas()
CASE("test_linearmodelparameterswrapper_valid_name")
void testIncrementTime()