IODA
DescendingSort.h
Go to the documentation of this file.
1 /*
2  * (C) Crown copyright 2020, Met Office
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_DESCENDINGSORT_H_
9 #define TEST_IODA_DESCENDINGSORT_H_
10 
11 #include <string>
12 #include <vector>
13 
14 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
15 
16 #include "eckit/config/LocalConfiguration.h"
17 #include "eckit/testing/Test.h"
18 
19 #include "oops/mpi/mpi.h"
20 #include "oops/runs/Test.h"
21 #include "oops/test/TestEnvironment.h"
22 #include "oops/util/Expect.h"
23 
24 #include "ioda/IodaTrait.h"
25 #include "ioda/ObsSpace.h"
26 
27 namespace ioda {
28 
29 namespace test {
30 
31 void testDescendingSort(const eckit::LocalConfiguration &conf) {
32  // Produce and configure ObsSpace object
33  util::DateTime bgn(conf.getString("window begin"));
34  util::DateTime end(conf.getString("window end"));
35  const eckit::LocalConfiguration obsSpaceConf(conf, "obs space");
37  obsParams.validateAndDeserialize(obsSpaceConf);
38  ioda::ObsSpace obsdata(obsParams, oops::mpi::world(), bgn, end, oops::mpi::myself());
39 
40  // This test only works for grouped data with descending sort order
41  if (obsdata.obs_sort_order() != "descending") {
42  throw eckit::BadValue("Must set sort_order to descending", Here());
43  }
44  if (obsdata.obs_group_vars().empty()) {
45  throw eckit::BadValue("Must set group_variable", Here());
46  }
47 
48  // Number of locations
49  const size_t nlocs = obsdata.nlocs();
50 
51  // All expected sort indices, obtained from input file
52  std::vector <int> expectedIndicesAll;
53  expectedIndicesAll.assign(nlocs, 0);
54  obsdata.get_db("MetaData", "expected_indices", expectedIndicesAll, { });
55 
56  // Record index for each location
57  const std::vector <size_t> recnums = obsdata.recnum();
58 
59  // List of unique record indices
60  const std::vector <size_t> recnumList = obsdata.recidx_all_recnums();
61 
62  for (size_t rn = 0; rn < recnumList.size(); ++rn) {
63  // Expected record indices for this recnum
64  std::vector <size_t> expectedRecordIndices;
65  for (size_t rnindex = 0; rnindex < nlocs; ++rnindex)
66  if (recnums[rnindex] == rn)
67  expectedRecordIndices.emplace_back(expectedIndicesAll[rnindex]);
68 
69  // Actual record indices for this recnum
70  const std::vector<size_t> recordIndices = obsdata.recidx_vector(rn);
71 
72  // Compare the two vectors and throw an exception if they differ
73  const bool equal = std::equal(recordIndices.begin(),
74  recordIndices.end(),
75  expectedRecordIndices.begin());
76  EXPECT(equal);
77  }
78  obsdata.save();
79 }
80 
81 class DescendingSort : public oops::Test {
82  private:
83  std::string testid() const override {return "ioda::test::DescendingSort";}
84 
85  void register_tests() const override {
86  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
87 
88  const eckit::LocalConfiguration conf(::test::TestEnvironment::config());
89  for (const std::string & testCaseName : conf.keys())
90  {
91  const eckit::LocalConfiguration testCaseConf(::test::TestEnvironment::config(), testCaseName);
92  ts.emplace_back(CASE("ioda/DescendingSort/" + testCaseName, testCaseConf)
93  {
94  testDescendingSort(testCaseConf);
95  });
96  }
97  }
98 
99  void clear() const override {}
100 };
101 
102 } // namespace test
103 } // namespace ioda
104 
105 #endif // TEST_IODA_DESCENDINGSORT_H_
void register_tests() const override
void clear() const override
std::string testid() const override
CASE("Derived variable, unit conversion, and exception checking methods")
void testDescendingSort(const eckit::LocalConfiguration &conf)