IODA
test/ioda/ObsVector.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_IODA_OBSVECTOR_H_
12 #define TEST_IODA_OBSVECTOR_H_
13 
14 #include <memory>
15 #include <string>
16 #include <vector>
17 
18 #include <boost/shared_ptr.hpp>
19 
20 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
21 
22 #include "eckit/config/LocalConfiguration.h"
23 #include "eckit/testing/Test.h"
24 
25 #include "oops/mpi/mpi.h"
26 #include "oops/runs/Test.h"
27 #include "oops/test/TestEnvironment.h"
28 #include "oops/util/dot_product.h"
29 #include "oops/util/Logger.h"
30 
31 #include "ioda/IodaTrait.h"
32 #include "ioda/ObsSpace.h"
33 #include "ioda/ObsVector.h"
34 
35 namespace ioda {
36 namespace test {
37 
38 // -----------------------------------------------------------------------------
39 
40 class ObsVecTestFixture : private boost::noncopyable {
42 
43  public:
44  static std::vector<boost::shared_ptr<ObsSpace_> > & obspace() {return getInstance().ospaces_;}
45 
46  static void cleanup() {
47  auto &spaces = getInstance().ospaces_;
48  for (auto &space : spaces) {
49  space->save();
50  space.reset();
51  }
52  }
53 
54  private:
56  static ObsVecTestFixture theObsVecTestFixture;
57  return theObsVecTestFixture;
58  }
59 
61  util::DateTime bgn((::test::TestEnvironment::config().getString("window begin")));
62  util::DateTime end((::test::TestEnvironment::config().getString("window end")));
63 
64  std::vector<eckit::LocalConfiguration> conf;
65  ::test::TestEnvironment::config().get("observations", conf);
66 
67  for (std::size_t jj = 0; jj < conf.size(); ++jj) {
68  eckit::LocalConfiguration obsconf(conf[jj], "obs space");
70  obsparams.validateAndDeserialize(obsconf);
71  boost::shared_ptr<ObsSpace_> tmp(new ObsSpace_(obsparams, oops::mpi::world(),
72  bgn, end, oops::mpi::myself()));
73  ospaces_.push_back(tmp);
74  }
75  }
76 
78 
79  std::vector<boost::shared_ptr<ObsSpace_> > ospaces_;
80 };
81 
82 // -----------------------------------------------------------------------------
83 
84 void testConstructor() {
85  typedef ObsVecTestFixture Test_;
86  typedef ioda::ObsVector ObsVector_;
87 
88  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
89  std::unique_ptr<ObsVector_> ov(new ObsVector_(*Test_::obspace()[jj]));
90  EXPECT(ov.get());
91 
92  ov.reset();
93  EXPECT(!ov.get());
94  }
95 }
96 
97 // -----------------------------------------------------------------------------
98 
100  typedef ObsVecTestFixture Test_;
101  typedef ioda::ObsVector ObsVector_;
102 
103  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
104  std::unique_ptr<ObsVector_> ov(new ObsVector_(*Test_::obspace()[jj]));
105 
106  std::unique_ptr<ObsVector_> other(new ObsVector_(*ov));
107  EXPECT(other.get());
108 
109  other.reset();
110  EXPECT(!other.get());
111 
112  EXPECT(ov.get());
113  }
114 }
115 
116 // -----------------------------------------------------------------------------
117 
118 void testNotZero() {
119  typedef ObsVecTestFixture Test_;
120  typedef ioda::ObsVector ObsVector_;
121  const double zero = 0.0;
122 
123  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
124  ObsVector_ ov(*Test_::obspace()[jj]);
125 
126  ov.random();
127 
128  const double ovov2 = dot_product(ov, ov);
129  EXPECT(ovov2 > zero);
130 
131  ov.zero();
132 
133  const double zz = dot_product(ov, ov);
134  EXPECT(zz == zero);
135  }
136 }
137 
138 // -----------------------------------------------------------------------------
139 
140 void testRead() {
141  typedef ObsVecTestFixture Test_;
142  typedef ioda::ObsVector ObsVector_;
143 
144  std::vector<eckit::LocalConfiguration> conf;
145  ::test::TestEnvironment::config().get("observations", conf);
146 
147  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
148  ioda::ObsSpace * Odb = Test_::obspace()[jj].get();
149 
150  // Grab the expected RMS value and tolerance from the obsdb_ configuration.
151  // Grab the obs space and test data configurations
152  eckit::LocalConfiguration testConfig;
153  conf[jj].get("test data", testConfig);
154  double ExpectedRms = testConfig.getDouble("rms ref");
155  double Tol = testConfig.getDouble("tolerance");
156 
157  // Read in a vector and check contents with norm function.
158  std::unique_ptr<ObsVector_> ov(new ObsVector_(*Odb, "ObsValue"));
159  double Rms = ov->rms();
160 
161  EXPECT(oops::is_close(Rms, ExpectedRms, Tol));
162  }
163 }
164 
165 // -----------------------------------------------------------------------------
166 
167 void testSave() {
168  typedef ObsVecTestFixture Test_;
169  typedef ioda::ObsVector ObsVector_;
170 
171  ioda::ObsSpace * Odb;
172  double Rms;
173  double ExpectedRms;
174 
175  std::vector<eckit::LocalConfiguration> conf;
176  ::test::TestEnvironment::config().get("observations", conf);
177 
178  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
179  Odb = Test_::obspace()[jj].get();
180  bool read_obs_from_separate_file =
181  conf[jj].getBool("obs space.read obs from separate file", false);
182 
183  // Read in a vector and save the rms value. Then write the vector into a
184  // test group, read it out of the test group and compare the rms of the
185  // vector read out of the test group with that of the original.
186  std::unique_ptr<ObsVector_> ov_orig(new ObsVector_(*Odb, "ObsValue"));
187  ExpectedRms = ov_orig->rms();
188 
189  if (!read_obs_from_separate_file)
190  ov_orig->save("ObsTest");
191 
192  std::unique_ptr<ObsVector_> ov_test(new ObsVector_(*Odb, "ObsTest"));
193  Rms = ov_test->rms();
194 
195  EXPECT(oops::is_close(Rms, ExpectedRms, 1.0e-12));
196  }
197 }
198 
199 // -----------------------------------------------------------------------------
200 /// \brief tests ObsVector::axpy methods
201 /// \details Tests the following for a random vector vec1:
202 /// 1. Calling ObsVector::axpy with a single number (2.0) returns the same result
203 /// as calling it with a vector of 2.0
204 /// 2. Calling ObsVector::axpy with vectors of coefficients that differ across
205 /// variables gives reasonable result. axpy is called twice, the coefficients
206 /// between the two different calls add to 2.0.
207 void testAxpy() {
208  typedef ObsVecTestFixture Test_;
209  typedef ioda::ObsVector ObsVector_;
210 
211  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
212  ioda::ObsSpace & obspace = *Test_::obspace()[jj];
213  ObsVector_ vec1(obspace);
214  vec1.random();
215 
216  // call axpy with coefficient 2.0 two different ways
217  ObsVector_ vec2(vec1);
218  vec2.axpy(2.0, vec1);
219  ObsVector_ vec3(vec1);
220  std::vector<double> beta(obspace.obsvariables().size(), 2.0);
221  vec3.axpy(beta, vec1);
222  oops::Log::test() << "Testing ObsVector::axpy" << std::endl;
223  oops::Log::test() << "x = " << vec1 << std::endl;
224  oops::Log::test() << "x.axpy(2, x) = " << vec2 << std::endl;
225  oops::Log::test() << "x.axpy(vector of 2, x) = " << vec3 << std::endl;
226  EXPECT(oops::is_close(vec2.rms(), vec3.rms(), 1.0e-8));
227 
228  // call axpy with vector of different values
229  std::vector<double> beta1(obspace.obsvariables().size());
230  std::vector<double> beta2(obspace.obsvariables().size());
231  for (size_t jj = 0; jj < beta1.size(); ++jj) {
232  beta1[jj] = static_cast<double>(jj)/beta1.size();
233  beta2[jj] = 2.0 - beta1[jj];
234  }
235  oops::Log::test() << "beta1 = " << beta1 << ", beta2 = " << beta2 << std::endl;
236  ObsVector_ vec4(vec1);
237  vec4.axpy(beta1, vec1);
238  oops::Log::test() << "x.axpy(beta1, x) = " << vec4 << std::endl;
239  vec4.axpy(beta2, vec1);
240  oops::Log::test() << "x.axpy(beta2, x) = " << vec4 << std::endl;
241  EXPECT(oops::is_close(vec4.rms(), vec3.rms(), 1.0e-8));
242  }
243 }
244 
245 /// \brief tests ObsVector::dot_product methods
246 /// \details Tests the following for a random vector vec1:
247 /// 1. Calling ObsVector::dot_product and calling ObsVector::multivar_dot_product
248 /// are consistent.
250  typedef ObsVecTestFixture Test_;
251  typedef ioda::ObsVector ObsVector_;
252 
253  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
254  ioda::ObsSpace & obspace = *Test_::obspace()[jj];
255  ObsVector_ vec1(obspace);
256  vec1.random();
257  ObsVector_ vec2(obspace);
258  vec2.random();
259 
260  double dp1 = vec1.dot_product_with(vec2);
261  std::vector<double> dp2 = vec1.multivar_dot_product_with(vec2);
262  oops::Log::test() << "Testing ObsVector::dot_product" << std::endl;
263  oops::Log::test() << "x1 = " << vec1 << std::endl;
264  oops::Log::test() << "x2 = " << vec2 << std::endl;
265  oops::Log::test() << "x1.dot_product_with(x2) = " << dp1 << std::endl;
266  oops::Log::test() << "x1.multivar_dot_product_with(x2) = " << dp2 << std::endl;
267 
268  // test that size of vector returned by multivar dot product is correct
269  EXPECT_EQUAL(dp2.size(), vec1.nvars());
270  // test that dot products are consistent (sum of all elements in multivar one
271  // is the same as the scalar one)
272  EXPECT(oops::is_close(dp1, std::accumulate(dp2.begin(), dp2.end(), 0.0), 1.0e-12));
273  }
274 }
275 
276 // -----------------------------------------------------------------------------
277 
279  typedef ioda::ObsVector ObsVector_;
280  typedef ioda::ObsSpace ObsSpace_;
281  typedef std::vector< std::shared_ptr< ObsVector_> > ObsVectors_;
282 
283  // Some of the ObsVector math routines require global communications,
284  // and so are performed differently for different distributions. But the
285  // answers should always be the same regardless of distribution.
286 
287  // get the list of distributions to test with
288  std::vector<std::string> dist_names =
289  ::test::TestEnvironment::config().getStringVector("distributions");
290  for (std::size_t ii = 0; ii < dist_names.size(); ++ii) {
291  oops::Log::debug() << "using distribution: " << dist_names[ii] << std::endl;
292  }
293 
294  // Get some config information that is the same regardless of distribution
295  util::DateTime bgn((::test::TestEnvironment::config().getString("window begin")));
296  util::DateTime end((::test::TestEnvironment::config().getString("window end")));
297  std::vector<eckit::LocalConfiguration> conf;
298  ::test::TestEnvironment::config().get("observations", conf);
299 
300  // for each distribution, create the set of obs vectors
301  std::vector< ObsVectors_ > dist_obsvecs;
302  std::vector< std::shared_ptr<ObsSpace_> > dist_obsdbs;
303  for (std::size_t dd = 0; dd < dist_names.size(); ++dd) {
304  ObsVectors_ obsvecs;
305  for (std::size_t jj = 0; jj < conf.size(); ++jj) {
306  // We want to cycle through the set of distributions that are specified
307  // in the list with the keyword "distributions" in the YAML. The test fixture
308  // has already constructed all of the obs spaces listed in the YAML, and we
309  // are repeating that action inside this loop. In other words, we are doubling
310  // up the obs space objects that a specified in the YAML.
311  //
312  // This is okay unless the YAML has specified an output file anywhere. The issue
313  // is that the output file is written during the destructor and the HDF library
314  // (unfortunately) tends to keep file descriptors open until the process terminates.
315  // Therefore it is possible for the file writes to collide, causing the test to
316  // crash, if the obs space created here is writing to the same file as the
317  // corresponding obs space in the test fixture.
318  //
319  // The fix is to tag on the name of the distribution on the output file name here
320  // to prevent the collision. The collision avoidance is absolutely guaranteed, but
321  // we can do it in a way that is unlikely to collide with any other output file
322  // names in the YAML. Note that this also prevents clobbering any output files
323  // specfied in the YAML.
324  eckit::LocalConfiguration obsconf(conf[jj], "obs space");
325  obsconf.set("distribution", dist_names[dd]);
326  if (obsconf.has("obsdataout.obsfile")) {
327  std::string fileName = obsconf.getString("obsdataout.obsfile");
328  std::string fileTag = std::string("_Dist_") + dist_names[dd];
329  std::size_t pos = fileName.find_last_of(".");
330  if (pos != std::string::npos) {
331  // have a suffix on the file name, insert tag before the suffix
332  fileName.insert(pos, fileTag);
333  } else {
334  // do not have a suffix on the file name, append tag to end of file name
335  fileName += fileTag;
336  }
337  obsconf.set("obsdataout.obsfile", fileName);
338  }
339 
340  ioda::ObsTopLevelParameters obsparams;
341  obsparams.validateAndDeserialize(obsconf);
342 
343  // Instantiate the obs space with the distribution we are testing
344  std::shared_ptr<ObsSpace_> obsdb(new ObsSpace(obsparams, oops::mpi::world(),
345  bgn, end, oops::mpi::myself()));
346  std::shared_ptr<ObsVector_> obsvec(new ObsVector_(*obsdb, "ObsValue"));
347  oops::Log::debug() << dist_names[dd] << ": " << *obsvec << std::endl;
348  dist_obsdbs.push_back(obsdb);
349  obsvecs.push_back(obsvec);
350  }
351  dist_obsvecs.push_back(obsvecs);
352  }
353 
354  // For each ObsVector make sure the math is the same regardless of distribution.
355  // Test rms(), nobs(), dot_product_with()
356  for (std::size_t ii = 0; ii < conf.size(); ++ii) {
357  // get the values for the first distribution
358  int nobs = dist_obsvecs[0][ii]->nobs();
359  double rms = dist_obsvecs[0][ii]->rms();
360  double dot = dist_obsvecs[0][ii]->dot_product_with(*dist_obsvecs[0][ii]);
361 
362  // make sure the values are the same for all the other distributions
363  for (std::size_t jj = 1; jj < dist_obsvecs.size(); ++jj) {
364  int nobs2 = dist_obsvecs[jj][ii]->nobs();
365  double rms2 = dist_obsvecs[jj][ii]->rms();
366  double dot2 = dist_obsvecs[jj][ii]->dot_product_with(*dist_obsvecs[jj][ii]);
367 
368  EXPECT(nobs == nobs2);
369  EXPECT(oops::is_close(rms, rms2, 1.0e-12));
370  EXPECT(oops::is_close(dot, dot2, 1.0e-12));
371  }
372  }
373 }
374 
375 // -----------------------------------------------------------------------------
376 
377 void testCleanup() {
378  // This test removes the obsspaces from the test fixture and ensures that they evict
379  // their contents to disk successfully.
380  typedef ObsVecTestFixture Test_;
381 
382  Test_::cleanup();
383 }
384 
385 // -----------------------------------------------------------------------------
386 
387 class ObsVector : public oops::Test {
388  public:
390  virtual ~ObsVector() {}
391 
392  private:
393  std::string testid() const override {return "test::ObsVector<ioda::IodaTrait>";}
394 
395  void register_tests() const override {
396  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
397 
398  ts.emplace_back(CASE("ioda/ObsVector/testConstructor")
399  { testConstructor(); });
400  ts.emplace_back(CASE("ioda/ObsVector/testCopyConstructor")
401  { testCopyConstructor(); });
402  ts.emplace_back(CASE("ioda/ObsVector/testNotZero")
403  { testNotZero(); });
404  ts.emplace_back(CASE("ioda/ObsVector/testRead")
405  { testRead(); });
406  ts.emplace_back(CASE("ioda/ObsVector/testSave")
407  { testSave(); });
408  ts.emplace_back(CASE("ioda/ObsVector/testAxpy")
409  { testAxpy(); });
410  ts.emplace_back(CASE("ioda/ObsVector/testDotProduct")
411  { testDotProduct(); });
412  ts.emplace_back(CASE("ioda/ObsVector/testDistributedMath")
413  { testDistributedMath(); });
414  ts.emplace_back(CASE("ioda/ObsVector/testCleanup")
415  { testCleanup(); });
416  }
417 
418  void clear() const override {}
419 };
420 
421 // =============================================================================
422 
423 } // namespace test
424 } // namespace ioda
425 
426 #endif // TEST_IODA_OBSVECTOR_H_
ObsVector class to handle vectors in observation space for IODA.
Definition: src/ObsVector.h:34
static ObsVecTestFixture & getInstance()
std::vector< boost::shared_ptr< ObsSpace_ > > ospaces_
static std::vector< boost::shared_ptr< ObsSpace_ > > & obspace()
std::string testid() const override
void clear() const override
void register_tests() const override
CASE("Derived variable, unit conversion, and exception checking methods")
void testAxpy()
tests ObsVector::axpy methods
void testCopyConstructor()
void testDotProduct()
tests ObsVector::dot_product methods
void testDistributedMath()