IODA
ObsSpaceIndexRecnum.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018 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_OBSSPACEINDEXRECNUM_H_
9 #define TEST_IODA_OBSSPACEINDEXRECNUM_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/IodaTrait.h"
30 #include "ioda/ObsSpace.h"
31 
32 namespace ioda {
33 namespace test {
34 
35 // -----------------------------------------------------------------------------
36 
37 class ObsSpaceTestFixture : private boost::noncopyable {
38  public:
39  static ioda::ObsSpace & obspace(const std::size_t ii) {
40  return *getInstance().ospaces_.at(ii);
41  }
42  static std::size_t size() {return getInstance().ospaces_.size();}
43  static void cleanup() {
44  auto &spaces = getInstance().ospaces_;
45  for (auto &space : spaces) {
46  space->save();
47  space.reset();
48  }
49  }
50 
51  private:
53  static ObsSpaceTestFixture theObsSpaceTestFixture;
54  return theObsSpaceTestFixture;
55  }
56 
58  util::DateTime bgn(::test::TestEnvironment::config().getString("window begin"));
59  util::DateTime end(::test::TestEnvironment::config().getString("window end"));
60 
61  std::vector<eckit::LocalConfiguration> conf;
62  ::test::TestEnvironment::config().get("observations", conf);
63 
64  for (std::size_t jj = 0; jj < conf.size(); ++jj) {
65  eckit::LocalConfiguration obsconf(conf[jj], "obs space");
67  obsparams.validateAndDeserialize(obsconf);
68  boost::shared_ptr<ioda::ObsSpace> tmp(new ioda::ObsSpace(obsparams, oops::mpi::world(),
69  bgn, end, oops::mpi::myself()));
70  ospaces_.push_back(tmp);
71  }
72  }
73 
75 
76  std::vector<boost::shared_ptr<ioda::ObsSpace> > ospaces_;
77 };
78 
79 // -----------------------------------------------------------------------------
80 
81 void testConstructor() {
82  typedef ObsSpaceTestFixture Test_;
83 
84  std::vector<eckit::LocalConfiguration> conf;
85  ::test::TestEnvironment::config().get("observations", conf);
86 
87  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
88  // Grab the test configuration which holds the expected data.
89  eckit::LocalConfiguration testConfig;
90  conf[jj].get("test data", testConfig);
91  oops::Log::debug() << "Test data configuration: " << testConfig << std::endl;
92 
93  const ObsSpace &odb = Test_::obspace(jj);
94 
95  // Get the number of locations (nlocs) from the ObsSpace object
96  std::size_t GlobalNlocs = odb.globalNumLocs();
97  std::size_t Nlocs = odb.nlocs();
98  std::size_t Nvars = odb.nvars();
99  bool ObsAreSorted = odb.obsAreSorted();
100 
101  // Get the expected nlocs from the obspace object's configuration
102  std::size_t ExpectedGlobalNlocs = testConfig.getUnsigned("nlocs");
103  std::size_t ExpectedNvars = testConfig.getUnsigned("nvars");
104  bool ExpectedObsAreSorted = testConfig.getBool("obs are sorted");
105 
106  oops::Log::debug() << "GlobalNlocs, ExpectedGlobalNlocs: " << GlobalNlocs << ", "
107  << ExpectedGlobalNlocs << std::endl;
108  oops::Log::debug() << "Nvars, ExpectedNvars: " << Nvars << ", "
109  << ExpectedNvars << std::endl;
110  oops::Log::debug() << "ObsAreSorted, ExpectedObsAreSorted: " << ObsAreSorted << ", "
111  << ExpectedObsAreSorted << std::endl;
112 
113  EXPECT(GlobalNlocs == ExpectedGlobalNlocs);
114  if (Nlocs > 0)
115  EXPECT(Nvars == ExpectedNvars);
116  EXPECT(ObsAreSorted == ExpectedObsAreSorted);
117 
118  // records are ambigious and not implemented for halo distribution
119  if (odb.distribution()->name() != "Halo") {
120  std::size_t Nrecs = 0;
121  std::set<std::size_t> recIndices;
122  auto accumulator = odb.distribution()->createAccumulator<std::size_t>();
123  for (std::size_t loc = 0; loc < Nlocs; ++loc) {
124  if (bool isNewRecord = recIndices.insert(odb.recnum()[loc]).second) {
125  accumulator->addTerm(loc, 1);
126  ++Nrecs;
127  }
128  }
129  const size_t ExpectedNrecs = odb.nrecs();
130  EXPECT_EQUAL(Nrecs, ExpectedNrecs);
131 
132  // Calculate the global number of unique records
133  std::size_t GlobalNrecs = accumulator->computeResult();
134  std::size_t ExpectedGlobalNrecs = testConfig.getUnsigned("nrecs");
135  EXPECT_EQUAL(GlobalNrecs, ExpectedGlobalNrecs);
136  }
137  }
138 }
139 
140 // -----------------------------------------------------------------------------
141 
143  typedef ObsSpaceTestFixture Test_;
144 
145  std::vector<eckit::LocalConfiguration> conf;
146  ::test::TestEnvironment::config().get("observations", conf);
147 
148  for (std::size_t jj = 0; jj < Test_::size(); ++jj) {
149  // Grab the test configuration which holds the expected data.
150  eckit::LocalConfiguration testConfig;
151  conf[jj].get("test data", testConfig);
152  oops::Log::debug() << "Test data configuration: " << testConfig << std::endl;
153 
154  // Get the index and recnum vectors from the obs space
155  std::vector<std::size_t> Index = Test_::obspace(jj).index();
156  std::vector<std::size_t> Recnum = Test_::obspace(jj).recnum();
157 
158  // Get the expected index and recnum vectors from the obspace object's configuration
159 
160  int MyMpiSize = Test_::obspace(jj).comm().size();
161  int MyMpiRank = Test_::obspace(jj).comm().rank();
162 
163  std::string MyPath = "mpi size" + std::to_string(MyMpiSize) +
164  ".rank" + std::to_string(MyMpiRank);
165  std::vector<std::size_t> ExpectedIndex = testConfig.getUnsignedVector(MyPath + ".index");
166  std::vector<std::size_t> ExpectedRecnum = testConfig.getUnsignedVector(MyPath + ".recnum");
167  eckit::LocalConfiguration recidxTestConfig;
168  testConfig.get(MyPath + ".recidx", recidxTestConfig);
169 
170  oops::Log::debug() << "Index, ExpectedIndex: " << Index << ", "
171  << ExpectedIndex << std::endl;
172  oops::Log::debug() << "Recnum, ExpectedRecnum: " << Recnum << ", "
173  << ExpectedRecnum << std::endl;
174 
175  EXPECT(Index == ExpectedIndex);
176  EXPECT(Recnum == ExpectedRecnum);
177 
178  // check that the recidx data structure got initialized properly
179  oops::Log::debug() << "recidxTestConfig: " << recidxTestConfig << std::endl;
180  std::vector<std::size_t> RecIdxRecNums = Test_::obspace(jj).recidx_all_recnums();
181  for (std::size_t i = 0; i < RecIdxRecNums.size(); ++i) {
182  std::size_t RecNum = RecIdxRecNums[i];
183  std::string TestConfigKey = "rec" + std::to_string(RecNum);
184  std::vector<std::size_t> ExpectedRecIdxVector =
185  recidxTestConfig.getUnsignedVector(TestConfigKey);
186  std::vector<std::size_t> RecIdxVector = Test_::obspace(jj).recidx_vector(RecNum);
187 
188  oops::Log::debug() << "RecIdxVector, ExpectedRecIdxVector: "
189  << RecIdxVector << ", " << ExpectedRecIdxVector << std::endl;
190  EXPECT(RecIdxVector == ExpectedRecIdxVector);
191  }
192  }
193 }
194 
195 // -----------------------------------------------------------------------------
196 
197 void testCleanup() {
198  // This test removes the obsspaces and ensures that they evict their contents
199  // to disk successfully.
200  typedef ObsSpaceTestFixture Test_;
201 
202  Test_::cleanup();
203 }
204 
205 // -----------------------------------------------------------------------------
206 
207 class ObsSpaceIndexRecnum : public oops::Test {
208  public:
210  virtual ~ObsSpaceIndexRecnum() {}
211  private:
212  std::string testid() const override {return "test::ObsSpaceIndexRecnum<ioda::IodaTrait>";}
213 
214  void register_tests() const override {
215  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
216 
217  ts.emplace_back(CASE("ioda/ObsSpaceIndexRecnum/testConstructor")
218  { testConstructor(); });
219  ts.emplace_back(CASE("ioda/ObsSpaceIndexRecnum/testIndexRecnum")
220  { testIndexRecnum(); });
221  ts.emplace_back(CASE("ioda/ObsSpaceIndexRecnum/testCleanup")
222  { testCleanup(); });
223  }
224 
225  void clear() const override {}
226 };
227 
228 // -----------------------------------------------------------------------------
229 
230 } // namespace test
231 } // namespace ioda
232 
233 #endif // TEST_IODA_OBSSPACEINDEXRECNUM_H_
std::string testid() const override
void register_tests() 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")