OOPS
test/interface/State.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_STATE_H_
12 #define TEST_INTERFACE_STATE_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"
27 #include "oops/base/State.h"
28 #include "oops/base/Variables.h"
29 #include "oops/mpi/mpi.h"
30 #include "oops/runs/Test.h"
31 #include "oops/util/DateTime.h"
32 #include "oops/util/dot_product.h"
33 #include "oops/util/Logger.h"
34 #include "test/TestEnvironment.h"
35 
36 namespace test {
37 
38 // -----------------------------------------------------------------------------
39 
40 template <typename MODEL> class StateFixture : private boost::noncopyable {
43 
44  public:
45  static const eckit::Configuration & test() {return *getInstance().test_;}
46  static const Geometry_ & resol() {return *getInstance().resol_;}
47  static void reset() {
48  getInstance().resol_.reset();
49  getInstance().test_.reset();
50  }
51 
52  private:
54  static StateFixture<MODEL> theStateFixture;
55  return theStateFixture;
56  }
57 
59  test_.reset(new eckit::LocalConfiguration(TestEnvironment::config(), "state test"));
60 
61  const eckit::LocalConfiguration resolConfig(TestEnvironment::config(), "geometry");
62  resol_.reset(new Geometry_(resolConfig, oops::mpi::world()));
63  }
64 
65  ~StateFixture<MODEL>() {}
66 
67  std::unique_ptr<const eckit::LocalConfiguration> test_;
68  std::unique_ptr<Geometry_> resol_;
69 };
70 
71 // -----------------------------------------------------------------------------
72 /// \brief tests constructors and print method
73 template <typename MODEL> void testStateConstructors() {
74  typedef StateFixture<MODEL> Test_;
75  typedef oops::State<MODEL> State_;
76 
77  const double norm = Test_::test().getDouble("norm file");
78  const double tol = Test_::test().getDouble("tolerance");
79  const util::DateTime vt(Test_::test().getString("date"));
80 
81 // Test main constructor
82  const eckit::LocalConfiguration conf(Test_::test(), "statefile");
83  std::unique_ptr<State_> xx1(new State_(Test_::resol(), conf));
84 
85  EXPECT(xx1.get());
86  oops::Log::test() << "Printing State from yaml: " << *xx1 << std::endl;
87  const double norm1 = xx1->norm();
88  EXPECT(norm1 != 0);
89  EXPECT(oops::is_close(norm1, norm, tol));
90  EXPECT(xx1->validTime() == vt);
91 
92 // Test copy constructor
93  std::unique_ptr<State_> xx2(new State_(*xx1));
94  EXPECT(xx2.get());
95  EXPECT(oops::is_close(xx2->norm(), norm, tol));
96  EXPECT(xx2->validTime() == vt);
97 
98 // Destruct copy
99  xx2.reset();
100  EXPECT(!xx2.get());
101 
102 // Recompute initial norm to make sure nothing bad happened
103  const double norm2 = xx1->norm();
104  EXPECT(norm1 == norm2);
105 
106 // Test State(const Geometry_ &, const Variables &, const util::DateTime &) constructor
107  oops::Variables vars(xx1->variables());
108  State_ xx3(Test_::resol(), vars, vt);
109  oops::Log::test() << "Printing empty State: " << xx3 << std::endl;
110  EXPECT(xx3.norm() == 0);
111  EXPECT(xx3.validTime() == vt);
112  EXPECT(xx3.variables() == vars);
113 
114 // Test State(const Geometry_ &, const State &) constructor
115  State_ xx4(Test_::resol(), *xx1);
116  EXPECT(oops::is_close(xx4.norm(), norm, tol));
117  EXPECT(xx4.validTime() == vt);
118  EXPECT(xx4.variables() == xx1->variables());
119 }
120 
121 // -----------------------------------------------------------------------------
122 /*! \brief Tests State::geometry() and Geometry copy constructors
123  */
124 
125 template <typename MODEL> void testStateGeometry() {
126  typedef StateFixture<MODEL> Test_;
127  typedef oops::Geometry<MODEL> Geometry_;
128  typedef oops::State<MODEL> State_;
129 
130  const double norm = Test_::test().getDouble("norm file");
131  const double tol = Test_::test().getDouble("tolerance");
132  const util::DateTime vt(Test_::test().getString("date"));
133 
134  const eckit::LocalConfiguration conf(Test_::test(), "statefile");
135  State_ xx1(Test_::resol(), conf);
136 
137  // get geometry from xx1 and initialize xx2 (xx2 & xx1 should be the same)
138  const Geometry_ & geometry = xx1.geometry();
139  State_ xx2(geometry, conf);
140 
141  const double norm2 = xx2.norm();
142  EXPECT(oops::is_close(norm2, norm, tol));
143 }
144 
145 // -----------------------------------------------------------------------------
146 
147 /*! \brief Interpolation test
148  *
149  * \details **testStateAnalyticInitialCondition()** tests the creation of an
150  * analytic state for a given model. The conceptual steps are as follows:
151  *
152  * 1. Initialize the JEDI State object based on idealized analytic formulae
153  * 2. Compare norm to precomputed norm
154  *
155  * Relevant parameters in the **State* section of the config file include
156  *
157  * * **norm-gen** Normalization test for the generated State
158  *
159  * \date April, 2018: M. Miesch (JCSDA) adapted a preliminary version in the
160  * feature/interp branch
161  *
162  * \warning Since this model compares the interpolated state values to an exact analytic
163  * solution, it requires that the "analytic_init" option be implemented in the model and
164  * selected in the "state.state generate" section of the config file.
165  */
166 
167 template <typename MODEL> void testStateAnalyticInitialCondition() {
168  typedef StateFixture<MODEL> Test_;
169  typedef oops::State<MODEL> State_;
170 
171  // This creates a State object called xx based on information
172  // from the "geometry" and "state test.state generate" sections of
173  // the config file and checks its norm
174 
175  if (!Test_::test().has("state generate")) {
176  oops::Log::warning() << "Bypassing Analytical Initial Condition Test";
177  return;
178  }
179 
180  const eckit::LocalConfiguration confgen(Test_::test(), "state generate");
181  const State_ xx(Test_::resol(), confgen);
182 
183  const double norm = Test_::test().getDouble("norm generated state");
184  const double tol = Test_::test().getDouble("tolerance");
185 
186  oops::Log::debug() << "xx.norm(): " << std::fixed << std::setprecision(8) << xx.norm()
187  << std::endl;
188  oops::Log::debug() << "norm: " << std::fixed << std::setprecision(8) << norm << std::endl;
189 
190  EXPECT(oops::is_close(xx.norm(), norm, tol));
191 }
192 
193 // -----------------------------------------------------------------------------
194 
195 /*! \brief Tests of zero and accumul
196  *
197  * \details testStateZeroAndAccumul tests the folllowing:
198  *
199  * 1. Call State.zero() and check the resulting norm is indeed zero.
200  * 2. Call State.accumul(), passing a multiplication factor and a second State
201  * as arguments, and verify that the resulting norm is as expected.
202  * Several multiplication factors are tested.
203  */
204 
205 template <typename MODEL> void testStateZeroAndAccumul() {
206  typedef StateFixture<MODEL> Test_;
207  typedef oops::State<MODEL> State_;
208 
209  const eckit::LocalConfiguration conf(Test_::test(), "statefile");
210  State_ xx(Test_::resol(), conf);
211  const double tol = Test_::test().getDouble("tolerance");
212 
213  // Set state xx to zero
214  xx.zero();
215  EXPECT(xx.norm() == 0.0);
216 
217  // Set state xx to various multiples of yy
218  const State_ yy(Test_::resol(), conf);
219  const std::vector<double> mults {3.0, 0.0, -3.0};
220  for (const auto & mult : mults) {
221  xx.zero();
222  xx.accumul(mult, yy);
223  EXPECT(oops::is_close(xx.norm(), std::abs(mult) * yy.norm(), tol));
224  }
225 
226  // Ensure that a non-zero state, when acted on with accumul, is not equal to the result
227  State_ zz(Test_::resol(), conf);
228  zz.accumul(3.0, yy);
229  EXPECT_NOT(oops::is_close(zz.norm(), yy.norm(), tol, 0, oops::TestVerbosity::SILENT));
230 }
231 
232 /*! \brief validTime and updateTime tests
233  *
234  * \details **testStateDateTime()** tests the validTime and updateTime routines.
235  *
236  * This is performed by updating the initial state time in two ways:
237  * - two lots of one hour,
238  * - one lot of two hours.
239  *
240  * validTime is then used in the comparison of the two times obtained.
241  */
242 
243 template <typename MODEL> void testStateDateTime() {
244  typedef StateFixture<MODEL> Test_;
245  typedef oops::State<MODEL> State_;
246 
247  // Configuration to read initial state
248  const eckit::LocalConfiguration conf(Test_::test(), "statefile");
249  State_ xx(Test_::resol(), conf);
250 
251  // Update the time by two lots of one hour
252  const util::Duration onehour(3600);
253  xx.updateTime(onehour);
254  xx.updateTime(onehour);
255 
256  // Create another state
257  State_ yy(Test_::resol(), conf);
258 
259  // Update the time of the second state by two hours
260  const util::Duration twohours(7200);
261  yy.updateTime(twohours);
262 
263  EXPECT(xx.validTime() == yy.validTime());
264 
265  // Increment the first state's time again and check the times are now not equal
266  xx.updateTime(onehour);
267  EXPECT_NOT(xx.validTime() == yy.validTime());
268 }
269 
270 // -----------------------------------------------------------------------------
271 
272 /*! \brief Read and write tests
273  *
274  * \details **testStateReadWrite()** tests reading and writing model state files.
275  *
276  * The tests are as follows:
277  * 1. Read an input file and check it sets a state correctly.
278  * 2. Write an output file and read it in again, checking the states are the same.
279  */
280 
281 template <typename MODEL> void testStateReadWrite() {
282  typedef StateFixture<MODEL> Test_;
283  typedef oops::State<MODEL> State_;
284 
285  // Configuration to read initial state
286  const eckit::LocalConfiguration conf(Test_::test(), "statefile");
287  State_ xx(Test_::resol(), conf);
288  const double tol = Test_::test().getDouble("tolerance");
289 
290  // Determine initial state norm
291  const double norm = xx.norm();
292 
293  // Set state to zero
294  xx.zero();
295 
296  // Read input file
297  xx.read(conf);
298 
299  // Check norm has its initial value
300  EXPECT(xx.norm() == norm);
301 
302  if (Test_::test().has("statefileout")) {
303  // Modify state
304  const double mult = 2.0;
305  xx.accumul(mult, xx);
306 
307  // Determine modified state norm
308  const double normout = xx.norm();
309 
310  // Configuration to read and write output state
311  const eckit::LocalConfiguration confout(Test_::test(), "statefileout");
312 
313  // Write modified state to output file
314  xx.write(confout);
315 
316  // Read modifed state from output file
317  State_ yy(Test_::resol(), confout);
318 
319  // Check modified state norm has its expected value
320  EXPECT(oops::is_close(yy.norm(), normout, tol));
321 
322  // Check modified state norm is not equal to the initial norm
323  EXPECT_NOT(oops::is_close(norm, normout, tol, 0, oops::TestVerbosity::SILENT));
324  }
325 }
326 
327 // -----------------------------------------------------------------------------
328 
329 template <typename MODEL>
330 class State : public oops::Test {
331  public:
332  State() {}
334 
335  private:
336  std::string testid() const override {return "test::State<" + MODEL::name() + ">";}
337 
338  void register_tests() const override {
339  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
340 
341  ts.emplace_back(CASE("interface/State/testStateConstructors")
342  { testStateConstructors<MODEL>(); });
343  ts.emplace_back(CASE("interface/State/testStateGeometry")
344  { testStateGeometry<MODEL>(); });
345  ts.emplace_back(CASE("interface/State/testStateAnalyticInitialCondition")
346  { testStateAnalyticInitialCondition<MODEL>(); });
347  ts.emplace_back(CASE("interface/State/testStateZeroAndAccumul")
348  { testStateZeroAndAccumul<MODEL>(); });
349  ts.emplace_back(CASE("interface/State/testStateDateTime")
350  { testStateDateTime<MODEL>(); });
351  ts.emplace_back(CASE("interface/State/testStateReadWrite")
352  { testStateReadWrite<MODEL>(); });
353  }
354 
355  void clear() const override {}
356 };
357 
358 // -----------------------------------------------------------------------------
359 
360 } // namespace test
361 
362 #endif // TEST_INTERFACE_STATE_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
State class used in oops; subclass of interface class interface::State.
static StateFixture< MODEL > & getInstance()
oops::State< MODEL > State_
static const eckit::Configuration & test()
static const Geometry_ & resol()
oops::Geometry< MODEL > Geometry_
std::unique_ptr< Geometry_ > resol_
std::unique_ptr< const eckit::LocalConfiguration > test_
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
logical function has(this, var)
void testStateReadWrite()
Read and write tests.
void testStateDateTime()
validTime and updateTime tests
void testStateZeroAndAccumul()
Tests of zero and accumul.
void testStateGeometry()
Tests State::geometry() and Geometry copy constructors.
void testStateConstructors()
tests constructors and print method
CASE("test_linearmodelparameterswrapper_valid_name")
void testStateAnalyticInitialCondition()
Interpolation test.