8 #ifndef TEST_IODA_EXTENDEDOBSSPACE_H_
9 #define TEST_IODA_EXTENDEDOBSSPACE_H_
16 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
18 #include "eckit/config/LocalConfiguration.h"
19 #include "eckit/testing/Test.h"
21 #include "oops/mpi/mpi.h"
22 #include "oops/runs/Test.h"
23 #include "oops/test/TestEnvironment.h"
24 #include "oops/util/Expect.h"
26 #include "ioda/IodaTrait.h"
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);
42 const bool expectThrows = conf.getBool(
"expectThrows",
false);
46 bgn, end, oops::mpi::myself()));
49 ioda::ObsSpace obsdata(obsParams, oops::mpi::world(), bgn, end, oops::mpi::myself());
52 if (obsdata.obs_group_vars().empty())
53 throw eckit::BadValue(
"Must set 'group variables' configuration option", Here());
55 if (!obsSpaceConf.has(
"extension"))
56 throw eckit::BadValue(
"Must set 'extension' configuration option", Here());
58 const int nlevs = obsSpaceConf.getInt(
"extension.average profiles onto model levels", 0);
60 if (nlevs <= 0)
return;
62 const int MPIsize = obsdata.comm().size();
63 const int MPIrank = obsdata.comm().rank();
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);
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);
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);
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();
95 EXPECT_EQUAL((nlocs - extendedObsSpaceStart) % nlevs, 0);
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);
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);
122 const std::vector<std::size_t> recidx_all_recnums = obsdata.recidx_all_recnums();
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);
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);
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());
146 std::vector <size_t> index_processors_expected(index_processors.size());
147 std::iota(index_processors_expected.begin(), index_processors_expected.end(), 0);
149 EXPECT_EQUAL(index_processors, index_processors_expected);
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"});
160 const std::vector <std::string> extendedVarsToCheck =
161 {
"latitude",
"longitude",
"air_pressure"};
163 std::vector <std::string> statids(obsdata.
nlocs());
164 obsdata.get_db(
"MetaData",
"station_id", statids);
168 const std::vector <std::string> uniqueStatids =
169 conf.getStringVector(
"unique statids");
172 std::vector <float> varToCheck(obsdata.
nlocs());
174 for (
const auto & extendedVar : extendedVarsToCheck) {
175 obsdata.get_db(
"MetaData", extendedVar, varToCheck);
177 const bool nonMissing =
178 std::find(nonMissingExtendedVars.begin(), nonMissingExtendedVars.end(), extendedVar)
179 != nonMissingExtendedVars.end();
181 for (std::size_t jprof = 0; jprof < nrecs_original; ++jprof) {
183 const std::vector<std::size_t> locsOriginal =
184 obsdata.recidx_vector(recidx_all_recnums[jprof]);
186 const std::vector<std::size_t> locsExtended =
187 obsdata.recidx_vector(recidx_all_recnums[jprof + nrecs_original]);
189 float valueToCompare = missingValueFloat;
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()];
197 for (
const auto jloc : locsExtended)
198 EXPECT(varToCheck[jloc] == valueToCompare);
207 std::string
testid()
const override {
return "ioda::test::ExtendedObsSpace";}
210 std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
212 const eckit::LocalConfiguration conf(::test::TestEnvironment::config());
213 for (
const std::string & testCaseName : conf.keys())
215 const eckit::LocalConfiguration testCaseConf(::test::TestEnvironment::config(), testCaseName);
216 ts.emplace_back(
CASE(
"ioda/ExtendedObsSpace/" + testCaseName, testCaseConf)
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")