IODA
ExtendedObsSpace.h
Go to the documentation of this file.
1 /*
2  * (C) Crown copyright 2021, 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_EXTENDEDOBSSPACE_H_
9 #define TEST_IODA_EXTENDEDOBSSPACE_H_
10 
11 #include <algorithm>
12 #include <sstream>
13 #include <string>
14 #include <vector>
15 
16 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
17 
18 #include "eckit/config/LocalConfiguration.h"
19 #include "eckit/testing/Test.h"
20 
21 #include "oops/mpi/mpi.h"
22 #include "oops/runs/Test.h"
23 #include "oops/test/TestEnvironment.h"
24 #include "oops/util/Expect.h"
25 
26 #include "ioda/IodaTrait.h"
27 #include "ioda/ObsSpace.h"
28 
29 namespace ioda {
30 
31 namespace test {
32 
33 void testExtendedObsSpace(const eckit::LocalConfiguration &conf) {
34  // Produce and configure ObsSpace object
35  util::DateTime bgn(conf.getString("window begin"));
36  util::DateTime end(conf.getString("window end"));
37  const eckit::LocalConfiguration obsSpaceConf(conf, "obs space");
39  obsParams.validateAndDeserialize(obsSpaceConf);
40 
41  // Instantiate ObsSpace, allowing for exceptions to be thrown.
42  const bool expectThrows = conf.getBool("expectThrows", false);
43  if (expectThrows) {
44  EXPECT_THROWS(ioda::ObsSpace obsDataThrow(obsParams,
45  oops::mpi::world(),
46  bgn, end, oops::mpi::myself()));
47  return;
48  }
49  ioda::ObsSpace obsdata(obsParams, oops::mpi::world(), bgn, end, oops::mpi::myself());
50 
51  // This test only works for grouped data.
52  if (obsdata.obs_group_vars().empty())
53  throw eckit::BadValue("Must set 'group variables' configuration option", Here());
54  // This test only works if the correct ObsSpace extension options have been supplied.
55  if (!obsSpaceConf.has("extension"))
56  throw eckit::BadValue("Must set 'extension' configuration option", Here());
57  // Number of levels per record in the extended ObsSpace.
58  const int nlevs = obsSpaceConf.getInt("extension.average profiles onto model levels", 0);
59  // The extension is not performed if the number of levels is less than or equal to zero.
60  if (nlevs <= 0) return;
61 
62  const int MPIsize = obsdata.comm().size();
63  const int MPIrank = obsdata.comm().rank();
64 
65  // Compare number of locations with expected value.
66  const int nlocs = static_cast<int>(obsdata.nlocs());
67  std::stringstream expectednumber;
68  expectednumber << "expected nlocs (" << MPIsize << " PE, rank " << MPIrank << ")";
69  const int nlocs_expected = conf.getInt(expectednumber.str());
70  EXPECT_EQUAL(nlocs, nlocs_expected);
71 
72  // Compare global number of locations with expected value.
73  const int gnlocs = static_cast<int>(obsdata.globalNumLocs());
74  expectednumber.str("");
75  expectednumber << "expected gnlocs (" << MPIsize << " PE, rank " << MPIrank << ")";
76  const int gnlocs_expected = conf.getInt(expectednumber.str());
77  EXPECT_EQUAL(gnlocs, gnlocs_expected);
78 
79  // Compare number of records with expected value.
80  const int nrecs = static_cast<int>(obsdata.nrecs());
81  expectednumber.str("");
82  expectednumber << "expected nrecs (" << MPIsize << " PE, rank " << MPIrank << ")";
83  const int nrecs_expected = conf.getInt(expectednumber.str());
84  EXPECT_EQUAL(nrecs, nrecs_expected);
85 
86  // Given the extended records have nlevs entries each,
87  // calculate the corresponding index at which extended_obs_space switches from 0 to 1.
88  std::vector <int> extended_obs_space(nlocs);
89  obsdata.get_db("MetaData", "extended_obs_space", extended_obs_space);
90  const size_t extendedObsSpaceStart =
91  std::find(extended_obs_space.begin(),
92  extended_obs_space.end(), 1) - extended_obs_space.begin();
93  // Check the index of the start of the extended ObsSpace is
94  // a multiple of nlevs from the final index.
95  EXPECT_EQUAL((nlocs - extendedObsSpaceStart) % nlevs, 0);
96  // Check the values of extended_obs_space.
97  for (int iloc = 0; iloc < extendedObsSpaceStart; ++iloc)
98  EXPECT_EQUAL(extended_obs_space[iloc], 0);
99  for (int iloc = extendedObsSpaceStart; iloc < nlocs; ++iloc)
100  EXPECT_EQUAL(extended_obs_space[iloc], 1);
101 
102  // Get all ObsValue and ObsError vectors that will be simulated.
103  // For each vector check that the values in the extended ObsSpace are all missing.
104  const float missingValueFloat = util::missingValue(missingValueFloat);
105  std::vector <float> val(nlocs);
106  std::vector <float> err(nlocs);
107  const oops::Variables& obsvars = obsdata.obsvariables();
108  for (int ivar = 0; ivar < obsvars.size(); ++ivar) {
109  const std::string varname = obsvars[ivar];
110  obsdata.get_db("ObsValue", varname, val);
111  obsdata.get_db("ObsError", varname, err);
112  for (int iloc = extendedObsSpaceStart; iloc < nlocs; ++iloc) {
113  EXPECT_EQUAL(val[iloc], missingValueFloat);
114  EXPECT_EQUAL(err[iloc], missingValueFloat);
115  }
116  }
117 
118  // Compare record numbers on this processor.
119  // There should be an even number of records; the second half should have indices shifted
120  // by a constant offset with respect to the first half. This offset should be equal to the
121  // original number of records.
122  const std::vector<std::size_t> recidx_all_recnums = obsdata.recidx_all_recnums();
123  // Determine the original record numbers by dividing the global number of records by two.
124  EXPECT_EQUAL(nrecs % 2, 0);
125  std::size_t nrecs_original = nrecs / 2;
126  const std::vector<std::size_t> original_recnums(recidx_all_recnums.begin(),
127  recidx_all_recnums.begin() + nrecs / 2);
128  const std::vector<std::size_t> extended_recnums(recidx_all_recnums.begin() + nrecs / 2,
129  recidx_all_recnums.end());
130  std::size_t gnrecs_original = original_recnums.empty() ? 0 : (original_recnums.back() + 1);
131  obsdata.distribution()->max(gnrecs_original);
132 
133  std::vector<std::size_t> extended_recnums_expected;
134  for (std::size_t i : original_recnums)
135  extended_recnums_expected.push_back(i + gnrecs_original);
136  EXPECT_EQUAL(extended_recnums, extended_recnums_expected);
137 
138  // Compare indices across all processors.
139  // Gather all indices, sort them, and produce a vector of unique indices.
140  std::vector <size_t> index_processors = obsdata.index();
141  obsdata.distribution()->allGatherv(index_processors);
142  std::sort(index_processors.begin(), index_processors.end());
143  auto it_unique = std::unique(index_processors.begin(), index_processors.end());
144  index_processors.erase(it_unique, index_processors.end());
145  // Produce expected indices.
146  std::vector <size_t> index_processors_expected(index_processors.size());
147  std::iota(index_processors_expected.begin(), index_processors_expected.end(), 0);
148  // Compare actual and expected indices.
149  EXPECT_EQUAL(index_processors, index_processors_expected);
150 
151  // Check that values in each averaged profile have been set as desired.
152 
153  // User-configured list of variables that should be filled with non-missing values.
154  const std::vector <std::string> &nonMissingExtendedVars =
155  obsSpaceConf.getStringVector("extension.variables filled with non-missing values",
156  {"latitude", "longitude", "datetime",
157  "air_pressure", "air_pressure_levels", "station_id"});
158  // List of variables to check.
159  // It is required that these are all floating-point variables in the MetaData group.
160  const std::vector <std::string> extendedVarsToCheck =
161  {"latitude", "longitude", "air_pressure"};
162  // Retrieve all station IDs in the sample.
163  std::vector <std::string> statids(obsdata.nlocs());
164  obsdata.get_db("MetaData", "station_id", statids);
165  // Unique station IDs are taken from the configuration file.
166  // The IDs are loaded in this way in order to guarantee a particular correspondence
167  // with the reference vectors.
168  const std::vector <std::string> uniqueStatids =
169  conf.getStringVector("unique statids");
170 
171  // Vector holding values of any variables to check.
172  std::vector <float> varToCheck(obsdata.nlocs());
173  // Loop over all variables to check.
174  for (const auto & extendedVar : extendedVarsToCheck) {
175  obsdata.get_db("MetaData", extendedVar, varToCheck);
176  // Check whether this variable should have been filled.
177  const bool nonMissing =
178  std::find(nonMissingExtendedVars.begin(), nonMissingExtendedVars.end(), extendedVar)
179  != nonMissingExtendedVars.end();
180  // Loop over each original profile in the sample.
181  for (std::size_t jprof = 0; jprof < nrecs_original; ++jprof) {
182  // Locations corresponding to the original profile.
183  const std::vector<std::size_t> locsOriginal =
184  obsdata.recidx_vector(recidx_all_recnums[jprof]);
185  // Locations corresponding to the averaged profile.
186  const std::vector<std::size_t> locsExtended =
187  obsdata.recidx_vector(recidx_all_recnums[jprof + nrecs_original]);
188  // Obtain comparison value from configuration file.
189  float valueToCompare = missingValueFloat;
190  if (nonMissing) {
191  const std::vector <float> expectedValues = conf.getFloatVector("expected " + extendedVar);
192  const std::string statid_prof = statids[locsOriginal.front()];
193  auto it_statid_prof = std::find(uniqueStatids.begin(), uniqueStatids.end(), statid_prof);
194  valueToCompare = expectedValues[it_statid_prof - uniqueStatids.begin()];
195  }
196  // Compare values in the averaged profile to the expected value.
197  for (const auto jloc : locsExtended)
198  EXPECT(varToCheck[jloc] == valueToCompare);
199  }
200  }
201 
202  obsdata.save();
203 }
204 
205 class ExtendedObsSpace : public oops::Test {
206  private:
207  std::string testid() const override {return "ioda::test::ExtendedObsSpace";}
208 
209  void register_tests() const override {
210  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
211 
212  const eckit::LocalConfiguration conf(::test::TestEnvironment::config());
213  for (const std::string & testCaseName : conf.keys())
214  {
215  const eckit::LocalConfiguration testCaseConf(::test::TestEnvironment::config(), testCaseName);
216  ts.emplace_back(CASE("ioda/ExtendedObsSpace/" + testCaseName, testCaseConf)
217  {
218  testExtendedObsSpace(testCaseConf);
219  });
220  }
221  }
222 
223  void clear() const override {}
224 };
225 
226 } // namespace test
227 } // namespace ioda
228 
229 #endif // TEST_IODA_EXTENDEDOBSSPACE_H_
void clear() const override
std::string testid() const override
void register_tests() const override
void testExtendedObsSpace(const eckit::LocalConfiguration &conf)
CASE("Derived variable, unit conversion, and exception checking methods")