IODA
test/ioda/ObsSpace.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-2021 UCAR
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  */
7 
8 #ifndef TEST_IODA_OBSSPACE_H_
9 #define TEST_IODA_OBSSPACE_H_
10 
11 #include <cmath>
12 #include <set>
13 #include <string>
14 #include <vector>
15 
16 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
17 
18 #include <boost/noncopyable.hpp>
19 #include <boost/shared_ptr.hpp>
20 
21 #include "eckit/config/LocalConfiguration.h"
22 #include "eckit/testing/Test.h"
23 
24 #include "oops/mpi/mpi.h"
25 #include "oops/runs/Test.h"
26 #include "oops/test/TestEnvironment.h"
27 
28 #include "ioda/distribution/Accumulator.h"
29 #include "ioda/distribution/DistributionUtils.h"
30 #include "ioda/IodaTrait.h"
31 #include "ioda/ObsSpace.h"
32 
33 namespace eckit
34 {
35  // Don't use contracted output for floats -- the current implementation works only for integers.
36  template <> struct VectorPrintSelector<float> { typedef VectorPrintSimple selector; };
37 } // namespace eckit
38 
39 namespace ioda {
40 namespace test {
41 
42 // -----------------------------------------------------------------------------
43 
44 class ObsSpaceTestFixture : private boost::noncopyable {
45  public:
46  static ioda::ObsSpace & obspace(const std::size_t ii) {
47  return *getInstance().ospaces_.at(ii);
48  }
49  static const eckit::LocalConfiguration & config(const std::size_t ii) {
50  return getInstance().configs_.at(ii);
51  }
52  static std::size_t size() {return getInstance().ospaces_.size();}
53  static void cleanup() {
54  auto &spaces = getInstance().ospaces_;
55  for (auto &space : spaces) {
56  space->save();
57  space.reset();
58  }
59  }
60 
61  private:
63  static ObsSpaceTestFixture theObsSpaceTestFixture;
64  return theObsSpaceTestFixture;
65  }
66 
68  util::DateTime bgn(::test::TestEnvironment::config().getString("window begin"));
69  util::DateTime end(::test::TestEnvironment::config().getString("window end"));
70 
71  ::test::TestEnvironment::config().get("observations", configs_);
72 
73  for (std::size_t jj = 0; jj < configs_.size(); ++jj) {
74  eckit::LocalConfiguration obsconf(configs_[jj], "obs space");
76  obsparams.validateAndDeserialize(obsconf);
77  boost::shared_ptr<ioda::ObsSpace> tmp(new ioda::ObsSpace(obsparams, oops::mpi::world(),
78  bgn, end, oops::mpi::myself()));
79  ospaces_.push_back(tmp);
80  }
81  }
82 
84 
85  std::vector<eckit::LocalConfiguration> configs_;
86  std::vector<boost::shared_ptr<ioda::ObsSpace> > ospaces_;
87 };
88 
89 // -----------------------------------------------------------------------------
90 
91 void testConstructor() {
92  typedef ObsSpaceTestFixture Test_;
93 
94  std::vector<eckit::LocalConfiguration> conf;
95  ::test::TestEnvironment::config().get("observations", conf);
96 
97  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
98  // Grab the obs space and test data configurations
99  eckit::LocalConfiguration obsConfig;
100  eckit::LocalConfiguration testConfig;
101  conf[jj].get("obs space", obsConfig);
102  conf[jj].get("test data", testConfig);
103 
104  std::string DistMethod = obsConfig.getString("distribution", "RoundRobin");
105 
106  const ObsSpace &odb = Test_::obspace(jj);
107 
108  // Get the numbers of locations (nlocs) from the ObsSpace object
109  std::size_t GlobalNlocs = odb.globalNumLocs();
110  std::size_t Nlocs = odb.nlocs();
111  std::size_t Nvars = odb.nvars();
112 
113  // Get the purturbation seed from the ObsSpace object
114  int obsPertSeed = odb.params().obsPertSeed();
115 
116  // Get the expected nlocs from the obspace object's configuration
117  std::size_t ExpectedGlobalNlocs = testConfig.getUnsigned("nlocs");
118  std::size_t ExpectedNvars = testConfig.getUnsigned("nvars");
119 
120  // Get the expected purturbation seed from the config object
121  int ExpectedObsPertSeed = testConfig.getUnsigned("obs perturbations seed");
122 
123  // Get the obs grouping/sorting parameters from the ObsSpace object
124  std::vector<std::string> ObsGroupVars = odb.obs_group_vars();
125  std::string ObsSortVar = odb.obs_sort_var();
126  std::string ObsSortOrder = odb.obs_sort_order();
127 
128  // Get the expected obs grouping/sorting parameters from the configuration
129  std::vector<std::string> ExpectedObsGroupVars =
130  testConfig.getStringVector("expected group variables");
131  std::string ExpectedObsSortVar = testConfig.getString("expected sort variable");
132  std::string ExpectedObsSortOrder = testConfig.getString("expected sort order");
133 
134  oops::Log::debug() << "GlobalNlocs, ExpectedGlobalNlocs: " << GlobalNlocs << ", "
135  << ExpectedGlobalNlocs << std::endl;
136  oops::Log::debug() << "Nvars, ExpectedNvars: " << Nvars << ", "
137  << ExpectedNvars << std::endl;
138  // records are ambigious for halo distribution
139  // e.g. consider airplane (a single record in round robin) flying accros the globe
140  // for Halo distr this record will be considered unique on each PE
141  if (DistMethod != "Halo") {
142  std::size_t NRecs = 0;
143  std::set<std::size_t> recIndices;
144  auto accumulator = odb.distribution()->createAccumulator<std::size_t>();
145  for (std::size_t loc = 0; loc < Nlocs; ++loc) {
146  if (bool isNewRecord = recIndices.insert(odb.recnum()[loc]).second) {
147  accumulator->addTerm(loc, 1);
148  ++NRecs;
149  }
150  }
151  std::size_t ExpectedNRecs = odb.nrecs();
152  EXPECT_EQUAL(NRecs, ExpectedNRecs);
153 
154  // Calculate the global number of unique records
155  std::size_t GlobalNRecs = accumulator->computeResult();
156  std::size_t ExpectedGlobalNrecs = testConfig.getUnsigned("nrecs");
157  EXPECT_EQUAL(GlobalNRecs, ExpectedGlobalNrecs);
158  }
159 
160  oops::Log::debug() << "ObsGroupVars, ExpectedObsGroupVars: " << ObsGroupVars << ", "
161  << ExpectedObsGroupVars << std::endl;
162  oops::Log::debug() << "ObsSortVar, ExpectedObsSortVar: " << ObsSortVar << ", "
163  << ExpectedObsSortVar << std::endl;
164  oops::Log::debug() << "ObsSortOrder, ExpectedObsSortOrder: " << ObsSortOrder << ", "
165  << ExpectedObsSortOrder << std::endl;
166 
167  // get the standard nlocs and nchans dimension names and compare with expected values
168  std::string nlocsName = odb.get_dim_name(ioda::ObsDimensionId::Nlocs);
169  std::string nchansName = odb.get_dim_name(ioda::ObsDimensionId::Nchans);
170 
171  EXPECT(GlobalNlocs == ExpectedGlobalNlocs);
172  EXPECT(Nvars == ExpectedNvars);
173 
174  EXPECT(obsPertSeed == ExpectedObsPertSeed);
175 
176  EXPECT(ObsGroupVars == ExpectedObsGroupVars);
177  EXPECT(ObsSortVar == ExpectedObsSortVar);
178  EXPECT(ObsSortOrder == ExpectedObsSortOrder);
179 
180  EXPECT(nlocsName == "nlocs");
181  EXPECT(nchansName == "nchans");
182 
183  EXPECT(odb.get_dim_id("nlocs") == ioda::ObsDimensionId::Nlocs);
184  EXPECT(odb.get_dim_id("nchans") == ioda::ObsDimensionId::Nchans);
185  }
186 }
187 
188 // -----------------------------------------------------------------------------
189 
190 void testGetDb() {
191  typedef ObsSpaceTestFixture Test_;
192 
193  std::vector<eckit::LocalConfiguration> conf;
194  ::test::TestEnvironment::config().get("observations", conf);
195 
196  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
197  // Grab the obs space and test data configurations
198  eckit::LocalConfiguration obsConfig;
199  eckit::LocalConfiguration testConfig;
200  conf[jj].get("obs space", obsConfig);
201  conf[jj].get("test data", testConfig);
202 
203  // Set up a pointer to the ObsSpace object for convenience
204  ioda::ObsSpace * Odb = &(Test_::obspace(jj));
205  std::size_t Nlocs = Odb->nlocs();
206 
207  // Get the variables section from the test data and perform checks accordingly
208  std::vector<eckit::LocalConfiguration> varconf =
209  testConfig.getSubConfigurations("variables for get test");
210  double Tol = testConfig.getDouble("tolerance");
211  for (std::size_t i = 0; i < varconf.size(); ++i) {
212  // Read in the variable group, name and expected norm values from the configuration
213  std::string VarName = varconf[i].getString("name");
214  std::string GroupName = varconf[i].getString("group");
215  std::string VarType = varconf[i].getString("type");
216  bool SkipDerived = varconf[i].getBool("skip derived", false);
217 
218  // Do different checks according to type
219  if (VarType == "float") {
220  // Check if the variable exists
221  EXPECT(Odb->has(GroupName, VarName, SkipDerived));
222 
223  // Check the type from ObsSpace
224  ObsDtype VarDataType = Odb->dtype(GroupName, VarName, SkipDerived);
225  EXPECT(VarDataType == ObsDtype::Float);
226 
227  // Check auto-conversion to double from ObsSpace float
228  // Check the norm
229  std::vector<double> TestVec(Nlocs);
230  Odb->get_db(GroupName, VarName, TestVec, {}, SkipDerived);
231 
232  // Calculate the norm of the vector
233  double ExpectedVnorm = varconf[i].getDouble("norm");
234  double Vnorm = dotProduct(*Odb->distribution(), 1, TestVec, TestVec);
235  Vnorm = sqrt(Vnorm);
236 
237  EXPECT(oops::is_close(Vnorm, ExpectedVnorm, Tol));
238  } else if (VarType == "integer") {
239  // Check if the variable exists
240  EXPECT(Odb->has(GroupName, VarName, SkipDerived));
241 
242  // Check the type from ObsSpace
243  ObsDtype VarDataType = Odb->dtype(GroupName, VarName, SkipDerived);
244  EXPECT(VarDataType == ObsDtype::Integer);
245 
246  // Check the norm
247  std::vector<int> TestVec(Nlocs);
248  Odb->get_db(GroupName, VarName, TestVec, {}, SkipDerived);
249 
250  // Calculate the norm of the vector
251  double ExpectedVnorm = varconf[i].getDouble("norm");
252  double Vnorm = dotProduct(*Odb->distribution(), 1, TestVec, TestVec);
253  Vnorm = sqrt(Vnorm);
254 
255  EXPECT(oops::is_close(Vnorm, ExpectedVnorm, Tol));
256  } else if (VarType == "string") {
257  // Check if the variable exists
258  EXPECT(Odb->has(GroupName, VarName, SkipDerived));
259 
260  // Check the type from ObsSpace
261  ObsDtype VarDataType = Odb->dtype(GroupName, VarName, SkipDerived);
262  EXPECT(VarDataType == ObsDtype::String);
263 
264  // Check the first and last values of the vector
265  std::string ExpectedFirstValue = varconf[i].getString("first value");
266  std::string ExpectedLastValue = varconf[i].getString("last value");
267  std::vector<std::string> TestVec(Nlocs);
268  Odb->get_db(GroupName, VarName, TestVec, {}, SkipDerived);
269  EXPECT(TestVec[0] == ExpectedFirstValue);
270  EXPECT(TestVec[Nlocs-1] == ExpectedLastValue);
271  } else if (VarType == "none") {
272  // Check if the variable exists
273  EXPECT_NOT(Odb->has(GroupName, VarName, SkipDerived));
274 
275  // Check the type from ObsSpace
276  ObsDtype VarDataType = Odb->dtype(GroupName, VarName, SkipDerived);
277  EXPECT(VarDataType == ObsDtype::None);
278 
279  // A call to get_db should produce an exception
280  {
281  std::vector<float> TestVec(Nlocs);
282  EXPECT_THROWS(Odb->get_db(GroupName, VarName, TestVec, {}, SkipDerived));
283  }
284  {
285  std::vector<int> TestVec(Nlocs);
286  EXPECT_THROWS(Odb->get_db(GroupName, VarName, TestVec, {}, SkipDerived));
287  }
288  {
289  std::vector<std::string> TestVec(Nlocs);
290  EXPECT_THROWS(Odb->get_db(GroupName, VarName, TestVec, {}, SkipDerived));
291  }
292  }
293  }
294  }
295 }
296 
297 // -----------------------------------------------------------------------------
298 
299 void testPutDb() {
300  typedef ObsSpaceTestFixture Test_;
301 
302  std::string VarName("DummyVar");
303 
304  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
305  // Set up a pointer to the ObsSpace object for convenience
306  ioda::ObsSpace * Odb = &(Test_::obspace(jj));
307 
308  // Create a dummy vector to put into the database
309  // Load up the vector with contrived data, put the vector then
310  // get the vector and see if the contrived data made it through.
311  std::size_t Nlocs = Odb->nlocs();
312  std::vector<double> TestVec(Nlocs);
313  std::vector<double> ExpectedVec(Nlocs);
314 
315  for (std::size_t i = 0; i < Nlocs; ++i) {
316  ExpectedVec[i] = static_cast<double>(i);
317  }
318 
319  // Put the vector into the database. Then read the vector back from the database
320  // and compare to the original
321  Odb->put_db("MetaData", VarName, ExpectedVec);
322  Odb->get_db("MetaData", VarName, TestVec);
323 
324  bool VecMatch = true;
325  for (std::size_t i = 0; i < Nlocs; ++i) {
326  VecMatch = VecMatch && (static_cast<int>(ExpectedVec[i]) == static_cast<int>(TestVec[i]));
327  }
328 
329  EXPECT(VecMatch);
330  }
331 }
332 
333 // -----------------------------------------------------------------------------
334 
336  typedef ObsSpaceTestFixture Test_;
337 
338  std::vector<eckit::LocalConfiguration> conf;
339  ::test::TestEnvironment::config().get("observations", conf);
340 
341  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
342  // Grab the obs space and test data configurations
343  eckit::LocalConfiguration obsConfig;
344  eckit::LocalConfiguration testConfig;
345  conf[jj].get("obs space", obsConfig);
346  conf[jj].get("test data", testConfig);
347 
348  // Set up a pointer to the ObsSpace object for convenience
349  ioda::ObsSpace * Odb = &(Test_::obspace(jj));
350  std::size_t Nlocs = Odb->nlocs();
351 
352  // Get the variables section from the test data and perform checks accordingly
353  std::vector<eckit::LocalConfiguration> varconf =
354  testConfig.getSubConfigurations("variables for putget test");
355  for (std::size_t i = 0; i < varconf.size(); ++i) {
356  // Read in the variables from the ObsSpace (get_db) and use put_db to
357  // place a copy in a new group. Then read the new variable back
358  // in and see if you get the same values.
359  std::string VarName = varconf[i].getString("name");
360  std::string GroupName = varconf[i].getString("group");
361  std::vector<int> Channels = varconf[i].getIntVector("channels", { });
362  oops::Log::debug() << "Reading: " << GroupName << ", " << VarName << ", "
363  << Channels << std::endl;
364 
365  // Check if the variable exists
366  EXPECT(Odb->has(GroupName, VarName));
367 
368  // Check the type from ObsSpace
369  ObsDtype VarDataType = Odb->dtype(GroupName, VarName);
370  EXPECT(VarDataType == ObsDtype::Float);
371 
372  // Read in the variable
373  std::vector<float> OrigVec(Nlocs);
374  Odb->get_db(GroupName, VarName, OrigVec, Channels);
375 
376  // Write the variable into the new group
377  std::string TestGroupName = GroupName + "_Test";
378  std::string PutDbVarName = VarName;
379  std::vector<std::string> DimList = { "nlocs" };
380  if (!Channels.empty()) {
381  PutDbVarName += "_" + std::to_string(Channels[0]);
382  DimList.push_back("nchans");
383  }
384  oops::Log::debug() << "Writing: " << TestGroupName << ", " << PutDbVarName << ", "
385  << DimList << std::endl;
386 
387  Odb->put_db(TestGroupName, PutDbVarName, OrigVec, DimList);
388 
389  // Read in what was just written and compare to original data
390  std::vector<float> TestVec(Nlocs);
391  Odb->get_db(TestGroupName, VarName, TestVec, Channels);
392 
393  EXPECT(TestVec == OrigVec);
394  }
395  }
396 }
397 
398 // -----------------------------------------------------------------------------
399 
401  typedef ObsSpaceTestFixture Test_;
402 
403  std::string VarName("DummyVar");
404 
405  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
406  // Set up a pointer to the ObsSpace object for convenience
407  ioda::ObsSpace * Odb = &(Test_::obspace(jj));
408 
409  // Create a dummy vector to put into the database
410  // All rows read from the input file should be read only.
411  // All rows added since the read of the input file should be writeable.
412  std::size_t Nlocs = Odb->nlocs();
413  std::vector<double> TestVec(Nlocs);
414  std::vector<double> ExpectedVec(Nlocs);
415 
416  for (std::size_t i = 0; i < Nlocs; ++i) {
417  ExpectedVec[i] = static_cast<double>(i);
418  }
419 
420  // Put the vector into the database. Then read the vector back from the database
421  // and compare to the original
422  Odb->put_db("TestGroup", VarName, ExpectedVec);
423  Odb->get_db("TestGroup", VarName, TestVec);
424 
425  bool VecMatch = true;
426  for (std::size_t i = 0; i < Nlocs; ++i) {
427  VecMatch = VecMatch && (static_cast<int>(ExpectedVec[i]) == static_cast<int>(TestVec[i]));
428  }
429  EXPECT(VecMatch);
430 
431  // Now update the vector with the original multiplied by 2.
432  for (std::size_t i = 0; i < Nlocs; ++i) {
433  ExpectedVec[i] = ExpectedVec[i] * 2;
434  }
435 
436  Odb->put_db("TestGroup", VarName, ExpectedVec);
437  Odb->get_db("TestGroup", VarName, TestVec);
438 
439  VecMatch = true;
440  for (std::size_t i = 0; i < Nlocs; ++i) {
441  VecMatch = VecMatch && (static_cast<int>(ExpectedVec[i]) == static_cast<int>(TestVec[i]));
442  }
443  EXPECT(VecMatch);
444  }
445 }
446 
447 // -----------------------------------------------------------------------------
448 
450  typedef ObsSpaceTestFixture Test_;
451 
452  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
453  // Set up a pointer to the ObsSpace object for convenience
454  ioda::ObsSpace * Odb = &(Test_::obspace(jj));
455 
456  // Create a dummy array to put into the database
457  // Load up the array with contrived data, put the array then
458  // get the array and see if the contrived data made it through.
459  // If Nchans comes back equal to zero, it means that this obs space does not
460  // have an nchans dimension. In this case, this test is reduced to testing
461  // a 1D vector.
462  std::size_t Nlocs = Odb->nlocs();
463  std::size_t Nchans = Odb->nchans();
464 
465  std::vector<int> TestValues;
466  std::vector<int> ExpectedValues;
467  std::vector<std::string> dimList;
468 
469  int numElements = Nlocs;
470  dimList.push_back(Odb->get_dim_name(ObsDimensionId::Nlocs));
471  if (Nchans > 0) {
472  numElements *= Nchans;
473  dimList.push_back(Odb->get_dim_name(ObsDimensionId::Nchans));
474  }
475 
476  // Load up the expected values with numbers 0..n-1.
477  TestValues.resize(numElements);
478  ExpectedValues.resize(numElements);
479  int testValue = 0;
480  for (std::size_t i = 0; i < numElements; ++i) {
481  ExpectedValues[i] = testValue;
482  testValue++;
483  }
484 
485  // Put the data into the ObsSpace, then get the data back from the ObsSpace and
486  // compare to the original.
487  Odb->put_db("MultiDimData", "DummyVar", ExpectedValues, dimList);
488  Odb->get_db("MultiDimData", "DummyVar", TestValues, {} /*select all channels*/);
489  EXPECT(TestValues == ExpectedValues);
490 
491  const int numOddChannels = Nchans/2;
492  if (numOddChannels > 0) {
493  // Test retrieval of only the odd channels.
494 
495  const std::vector<int>& channels = Odb->obsvariables().channels();
496  ASSERT(channels.size() == Nchans);
497 
498  std::vector<int> chanSelect;
499  for (int i = 0; i < numOddChannels; ++i) {
500  const std::size_t channelIndex = 1 + 2 * i;
501  chanSelect.push_back(channels[channelIndex]);
502  }
503 
504  ExpectedValues.clear();
505  for (std::size_t loc = 0; loc < Nlocs; ++loc) {
506  for (int i = 0; i < numOddChannels; ++i) {
507  const std::size_t channelIndex = 1 + 2 * i;
508  ExpectedValues.push_back(loc * Nchans + channelIndex);
509  }
510  }
511 
512  Odb->get_db("MultiDimData", "DummyVar", TestValues, chanSelect);
513  EXPECT_EQUAL(TestValues, ExpectedValues);
514  }
515 
516  if (numOddChannels > 0) {
517  // Test retrieval of a single channel using the old syntax
518  // (variable name with a channel suffix)
519 
520  const std::vector<int>& channels = Odb->obsvariables().channels();
521  const int channelIndex = 1;
522  const int channelNumber = channels[channelIndex];
523 
524  ExpectedValues.clear();
525  for (std::size_t loc = 0; loc < Nlocs; ++loc)
526  ExpectedValues.push_back(loc * Nchans + channelIndex);
527 
528  Odb->get_db("MultiDimData", "DummyVar_" + std::to_string(channelNumber), TestValues);
529  EXPECT_EQUAL(TestValues, ExpectedValues);
530  }
531  }
532 }
533 
534 // -----------------------------------------------------------------------------
535 
536 // Test the obsvariables(), initial_obsvariables() and derived_obsvariables() methods.
538  typedef ObsSpaceTestFixture Test_;
539 
540  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
541  const ioda::ObsSpace & Odb = Test_::obspace(jj);
542 
543  ObsTopLevelParameters obsparams;
544  obsparams.validateAndDeserialize(Test_::config(jj));
545 
546  const oops::Variables & allSimVars = Odb.obsvariables();
547  const oops::Variables & initialSimVars = Odb.initial_obsvariables();
548  const oops::Variables & derivedSimVars = Odb.derived_obsvariables();
549 
550  EXPECT_EQUAL(initialSimVars, obsparams.simVars);
551  EXPECT_EQUAL(derivedSimVars, obsparams.derivedSimVars);
552  EXPECT_EQUAL(allSimVars.size(), initialSimVars.size() + derivedSimVars.size());
553  }
554 }
555 
556 // -----------------------------------------------------------------------------
557 
558 // Verify that for any derived simulated variable <var> a newly created ObsSpace has a variable
559 // <var> in the ObsError group and that it is filled with missing values.
561  typedef ObsSpaceTestFixture Test_;
562 
563  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
564  const ioda::ObsSpace & Odb = Test_::obspace(jj);
565 
566  const oops::Variables & derivedSimVars = Odb.derived_obsvariables();
567  for (size_t i = 0; i < derivedSimVars.size(); ++i) {
568  EXPECT(Odb.has("ObsError", derivedSimVars[i]));
569  std::vector<float> values(Odb.nlocs());
570  Odb.get_db("ObsError", derivedSimVars[i], values);
571 
572  std::vector<float> expectedValues(Odb.nlocs(), util::missingValue(float()));
573  EXPECT_EQUAL(values, expectedValues);
574  }
575  }
576 }
577 
578 // -----------------------------------------------------------------------------
579 
580 void testCleanup() {
581  // This test removes the obsspaces and ensures that they evict their contents
582  // to disk successfully.
583  typedef ObsSpaceTestFixture Test_;
584 
585  Test_::cleanup();
586 }
587 
588 // -----------------------------------------------------------------------------
589 
590 class ObsSpace : public oops::Test {
591  public:
592  ObsSpace() {}
593  virtual ~ObsSpace() {}
594 
595  private:
596  std::string testid() const override {return "test::ObsSpace<ioda::IodaTrait>";}
597 
598  void register_tests() const override {
599  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
600 
601  ts.emplace_back(CASE("ioda/ObsSpace/testConstructor")
602  { testConstructor(); });
603  ts.emplace_back(CASE("ioda/ObsSpace/testGetDb")
604  { testGetDb(); });
605  ts.emplace_back(CASE("ioda/ObsSpace/testPutDb")
606  { testPutDb(); });
607  ts.emplace_back(CASE("ioda/ObsSpace/testPutGetChanSelect")
608  { testPutGetChanSelect(); });
609  ts.emplace_back(CASE("ioda/ObsSpace/testWriteableGroup")
610  { testWriteableGroup(); });
611  ts.emplace_back(CASE("ioda/ObsSpace/testMultiDimTransfer")
612  { testMultiDimTransfer(); });
613  ts.emplace_back(CASE("ioda/ObsSpace/testCleanup")
614  { testCleanup(); });
615  }
616 
617  void clear() const override {}
618 };
619 
620 // -----------------------------------------------------------------------------
621 
622 } // namespace test
623 } // namespace ioda
624 
625 #endif // TEST_IODA_OBSSPACE_H_
oops::RequiredParameter< oops::Variables > simVars
simulated variables
oops::Parameter< oops::Variables > derivedSimVars
std::string testid() const override
void register_tests() const override
void clear() const override
std::vector< boost::shared_ptr< ioda::ObsSpace > > ospaces_
static const eckit::LocalConfiguration & config(const std::size_t ii)
std::vector< eckit::LocalConfiguration > configs_
static ioda::ObsSpace & obspace(const std::size_t ii)
static ObsSpaceTestFixture & getInstance()
void testWriteableGroup()
CASE("Derived variable, unit conversion, and exception checking methods")
void testDerivedObsError()
void testMultiDimTransfer()
void testPutGetChanSelect()
void testObsVariables()
double dotProduct(const Distribution &dist, std::size_t numVariables, const std::vector< double > &v1, const std::vector< double > &v2)