IODA
ObsSpaceInvalidNumeric.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_OBSSPACEINVALIDNUMERIC_H_
9 #define TEST_IODA_OBSSPACEINVALIDNUMERIC_H_
10 
11 #include <cmath>
12 #include <string>
13 #include <vector>
14 
15 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
16 
17 #include <boost/noncopyable.hpp>
18 #include <boost/shared_ptr.hpp>
19 
20 #include "eckit/config/LocalConfiguration.h"
21 #include "eckit/testing/Test.h"
22 
23 #include "oops/mpi/mpi.h"
24 #include "oops/runs/Test.h"
25 #include "oops/test/TestEnvironment.h"
26 
27 #include "ioda/IodaTrait.h"
28 #include "ioda/ObsSpace.h"
29 
30 namespace ioda {
31 namespace test {
32 
33 // -----------------------------------------------------------------------------
34 
35 class ObsSpaceTestFixture : private boost::noncopyable {
36  public:
37  static ioda::ObsSpace & obspace(const std::size_t ii) {
38  return *getInstance().ospaces_.at(ii);
39  }
40  static std::size_t size() {return getInstance().ospaces_.size();}
41  static void cleanup() {
42  auto &spaces = getInstance().ospaces_;
43  for (auto &space : spaces) {
44  space->save();
45  space.reset();
46  }
47  }
48 
49  private:
51  static ObsSpaceTestFixture theObsSpaceTestFixture;
52  return theObsSpaceTestFixture;
53  }
54 
56  util::DateTime bgn(::test::TestEnvironment::config().getString("window begin"));
57  util::DateTime end(::test::TestEnvironment::config().getString("window end"));
58 
59  std::vector<eckit::LocalConfiguration> conf;
60  ::test::TestEnvironment::config().get("observations", conf);
61 
62  for (std::size_t jj = 0; jj < conf.size(); ++jj) {
63  eckit::LocalConfiguration obsconf(conf[jj], "obs space");
65  obsparams.validateAndDeserialize(obsconf);
66  boost::shared_ptr<ioda::ObsSpace> tmp(new ioda::ObsSpace(obsparams, oops::mpi::world(),
67  bgn, end, oops::mpi::myself()));
68  ospaces_.push_back(tmp);
69  }
70  }
71 
73 
74  std::vector<boost::shared_ptr<ioda::ObsSpace> > ospaces_;
75 };
76 
77 // -----------------------------------------------------------------------------
78 
80  typedef ObsSpaceTestFixture Test_;
81 
82  std::vector<eckit::LocalConfiguration> conf;
83  ::test::TestEnvironment::config().get("observations", conf);
84 
85  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
86  // Grab the obs space and test data configurations
87  eckit::LocalConfiguration obsConfig;
88  eckit::LocalConfiguration testConfig;
89  conf[jj].get("obs space", obsConfig);
90  conf[jj].get("test data", testConfig);
91 
92  const ObsSpace &odb = Test_::obspace(jj);
93 
94  // Get the list of variables to check from the test configuration
95  std::vector<eckit::LocalConfiguration> testConfigVars;
96  testConfig.get("variables", testConfigVars);
97  float testTol = testConfig.getFloat("tolerance");
98 
99  // Check that that invalid numeric values got handled properly
100  for (std::size_t i = 0; i < testConfigVars.size(); ++i) {
101  std::string varName = testConfigVars[i].getString("name");
102  std::string groupName = testConfigVars[i].getString("group");
103  std::string varType = testConfigVars[i].getString("type");
104 
105  if (varType == "int") {
106  std::vector<int> expectedTestVals = testConfigVars[i].getIntVector("values");
107  std::vector<int> testVals;
108  odb.get_db(groupName, varName, testVals);
109  EXPECT_EQUAL(testVals, expectedTestVals);
110  } else if (varType == "float") {
111  std::vector<float> expectedTestVals = testConfigVars[i].getFloatVector("values");
112  std::vector<float> testVals;
113  odb.get_db(groupName, varName, testVals);
114  for (std::size_t j = 0; j < expectedTestVals.size(); ++j) {
115  EXPECT(oops::is_close(testVals[j], expectedTestVals[j], testTol));
116  }
117  } else if (varType == "string") {
118  std::vector<std::string> expectedTestVals =
119  testConfigVars[i].getStringVector("values");
120  std::vector<std::string> testVals;
121  odb.get_db(groupName, varName, testVals);
122  EXPECT_EQUAL(testVals, expectedTestVals);
123  }
124  }
125  }
126 }
127 
128 // -----------------------------------------------------------------------------
129 
130 void testCleanup() {
131  // This test removes the obsspaces and ensures that they evict their contents
132  // to disk successfully.
133  typedef ObsSpaceTestFixture Test_;
134 
135  Test_::cleanup();
136 }
137 
138 // -----------------------------------------------------------------------------
139 
140 class ObsSpaceInvalidNumeric : public oops::Test {
141  public:
144 
145  private:
146  std::string testid() const override {return "test::ObsSpaceInvalidNumeric<ioda::IodaTrait>";}
147 
148  void register_tests() const override {
149  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
150 
151  ts.emplace_back(CASE("ioda/ObsSpace/testInvalidNumeric")
152  { testInvalidNumeric(); });
153  ts.emplace_back(CASE("ioda/ObsSpace/testCleanup")
154  { testCleanup(); });
155  }
156 
157  void clear() const override {}
158 };
159 
160 // -----------------------------------------------------------------------------
161 
162 } // namespace test
163 } // namespace ioda
164 
165 #endif // TEST_IODA_OBSSPACEINVALIDNUMERIC_H_
std::string testid() const override
std::vector< boost::shared_ptr< ioda::ObsSpace > > ospaces_
static ioda::ObsSpace & obspace(const std::size_t ii)
static ObsSpaceTestFixture & getInstance()
CASE("Derived variable, unit conversion, and exception checking methods")