11 #ifndef TEST_DISTRIBUTION_DISTRIBUTION_H_
12 #define TEST_DISTRIBUTION_DISTRIBUTION_H_
22 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
24 #include <boost/noncopyable.hpp>
26 #include "eckit/config/LocalConfiguration.h"
27 #include "eckit/geometry/Point2.h"
28 #include "eckit/mpi/Comm.h"
29 #include "eckit/testing/Test.h"
31 #include "oops/mpi/mpi.h"
32 #include "oops/runs/Test.h"
33 #include "oops/test/TestEnvironment.h"
34 #include "oops/util/DateTime.h"
35 #include "oops/util/Logger.h"
37 #include "ioda/distribution/Distribution.h"
38 #include "ioda/distribution/DistributionFactory.h"
46 template <>
struct VectorPrintSelector<float> {
typedef VectorPrintSimple selector; };
47 template <>
struct VectorPrintSelector<
util::DateTime> {
typedef VectorPrintSimple selector; };
48 template <>
struct VectorPrintSelector<
util::Duration> {
typedef VectorPrintSimple selector; };
57 const eckit::LocalConfiguration conf(::test::TestEnvironment::config());
59 const eckit::mpi::Comm & MpiComm = oops::mpi::world();
60 const std::size_t MyRank = MpiComm.rank();
63 std::vector<eckit::LocalConfiguration> dist_types;
64 conf.get(
"distribution types", dist_types);
65 for (std::size_t i = 0; i < dist_types.size(); ++i) {
66 oops::Log::debug() <<
"Distribution::DistributionTypes: conf: " << dist_types[i] << std::endl;
68 const std::string MyRankCfgName =
"specs.rank" + std::to_string(MyRank) +
".config";
69 eckit::LocalConfiguration DistConfig(dist_types[i], MyRankCfgName);
70 const std::string TestDistType = DistConfig.getString(
"distribution");
71 oops::Log::debug() <<
"Distribution::DistType: " << TestDistType << std::endl;
74 EXPECT(TestDist.get());
81 const eckit::Configuration &MyRankConfig,
83 const std::vector<std::size_t> &Index,
84 const std::vector<std::size_t> &Recnums) {
86 std::size_t ExpectedNlocs = MyRankConfig.getUnsigned(
"nlocs");
87 std::size_t ExpectedNrecs = MyRankConfig.getUnsigned(
"nrecs");
88 std::size_t ExpectedNPatchLocs = MyRankConfig.getUnsigned(
"nPatchLocs");
89 std::vector<std::size_t> ExpectedIndex =
90 MyRankConfig.getUnsignedVector(
"index");
91 std::vector<std::size_t> ExpectedRecnums =
92 MyRankConfig.getUnsignedVector(
"recnums");
93 std::vector<std::size_t> ExpectedPatchIndex =
94 MyRankConfig.getUnsignedVector(
"patchIndex");
95 const std::vector<std::size_t> ExpectedAllGatherv =
96 config.getUnsignedVector(
"specs.allgatherv");
98 std::vector<bool> patchBool(Index.size());
99 std::vector<std::size_t> PatchLocsThisPE;
101 for (std::size_t j = 0; j < Index.size(); ++j) {
102 if (patchBool[j]) {PatchLocsThisPE.push_back(Index[j]);}
106 std::size_t
Nlocs = Index.size();
107 std::size_t Nrecs = std::set<std::size_t>(Recnums.begin(), Recnums.end()).size();
108 std::size_t NPatchLocs = PatchLocsThisPE.size();
110 oops::Log::debug() <<
"Location Index: " << Index << std::endl;
111 oops::Log::debug() <<
"PatchLocsThisPE: " << PatchLocsThisPE << std::endl;
112 oops::Log::debug() <<
"Nlocs: " <<
Nlocs <<
" Nrecs: " << Nrecs
113 <<
" NPatchLocs" << NPatchLocs << std::endl;
115 EXPECT_EQUAL(
Nlocs, ExpectedNlocs);
116 EXPECT_EQUAL(Nrecs, ExpectedNrecs);
117 EXPECT_EQUAL(NPatchLocs, ExpectedNPatchLocs);
120 EXPECT_EQUAL(Index, ExpectedIndex);
121 EXPECT_EQUAL(Recnums, ExpectedRecnums);
122 EXPECT_EQUAL(PatchLocsThisPE, ExpectedPatchIndex);
129 const std::vector<std::size_t> ExpectedAllGathervSizeT = ExpectedAllGatherv;
130 std::vector<size_t> AllGathervSizeT = Index;
132 EXPECT_EQUAL(AllGathervSizeT, ExpectedAllGathervSizeT);
139 std::vector<size_t> ExpectedGlobalUniqueConsecutiveLocationIndices(
Nlocs);
140 std::vector<size_t> GlobalUniqueConsecutiveLocationIndices(
Nlocs);
141 for (
size_t loc = 0; loc <
Nlocs; ++loc) {
142 const std::vector<size_t>::iterator it = std::find(
143 AllGathervSizeT.begin(), AllGathervSizeT.end(), Index[loc]);
144 ASSERT(it != AllGathervSizeT.end());
145 ExpectedGlobalUniqueConsecutiveLocationIndices[loc] = it - AllGathervSizeT.begin();
146 GlobalUniqueConsecutiveLocationIndices[loc] =
150 EXPECT_EQUAL(GlobalUniqueConsecutiveLocationIndices,
151 ExpectedGlobalUniqueConsecutiveLocationIndices);
156 std::vector<int> ExpectedAllGathervInt(ExpectedAllGatherv.begin(),
157 ExpectedAllGatherv.end());
158 std::vector<int> AllGathervInt(Index.begin(), Index.end());
160 EXPECT_EQUAL(AllGathervInt, ExpectedAllGathervInt);
165 std::vector<float> ExpectedAllGathervFloat(ExpectedAllGatherv.begin(),
166 ExpectedAllGatherv.end());
167 std::vector<float> AllGathervFloat(Index.begin(), Index.end());
169 EXPECT_EQUAL(AllGathervFloat, ExpectedAllGathervFloat);
174 std::vector<double> ExpectedAllGathervDouble(ExpectedAllGatherv.begin(),
175 ExpectedAllGatherv.end());
176 std::vector<double> AllGathervDouble(Index.begin(), Index.end());
178 EXPECT_EQUAL(AllGathervDouble, ExpectedAllGathervDouble);
183 auto numberToString = [](std::size_t x) {
return std::to_string(x); };
184 std::vector<std::string> ExpectedAllGathervString;
185 std::transform(ExpectedAllGatherv.begin(), ExpectedAllGatherv.end(),
186 std::back_inserter(ExpectedAllGathervString), numberToString);
187 std::vector<std::string> AllGathervString;
189 std::back_inserter(AllGathervString), numberToString);
191 EXPECT_EQUAL(AllGathervString, ExpectedAllGathervString);
196 auto numberToDateTime = [](std::size_t x) {
return util::DateTime(2000, 1, 1, 0, 0, x); };
197 std::vector<util::DateTime> ExpectedAllGathervDateTime;
198 std::transform(ExpectedAllGatherv.begin(), ExpectedAllGatherv.end(),
199 std::back_inserter(ExpectedAllGathervDateTime), numberToDateTime);
200 std::vector<util::DateTime> AllGathervDateTime;
202 std::back_inserter(AllGathervDateTime), numberToDateTime);
204 EXPECT_EQUAL(AllGathervDateTime, ExpectedAllGathervDateTime);
211 const eckit::LocalConfiguration conf(::test::TestEnvironment::config());
213 const eckit::mpi::Comm & MpiComm = oops::mpi::world();
214 const std::size_t MyRank = MpiComm.rank();
216 std::string TestDistType;
217 std::string DistName;
218 std::unique_ptr<ioda::Distribution> TestDist;
221 std::vector<eckit::LocalConfiguration> dist_types;
222 conf.get(
"distribution types", dist_types);
223 for (std::size_t i = 0; i < dist_types.size(); ++i) {
224 oops::Log::debug() <<
"Distribution::DistributionTypes: conf: "
225 << dist_types[i] << std::endl;
228 const std::string MyRankCfgName =
"specs.rank" + std::to_string(MyRank);
229 const eckit::LocalConfiguration MyRankConfig = dist_types[i].getSubConfiguration(MyRankCfgName);
230 oops::Log::debug() <<
"Distribution::DistributionTypes: "
231 << MyRankCfgName <<
": " << MyRankConfig << std::endl;
233 const eckit::LocalConfiguration DistConfig(MyRankConfig,
"config");
234 const std::string DistName = DistConfig.getString(
"distribution");
235 oops::Log::debug() <<
"Distribution::DistType: " << DistName << std::endl;
238 EXPECT(TestDist.get());
241 std::size_t Gnlocs = dist_types[i].getInt(
"specs.gnlocs");
242 std::vector<double> glats(Gnlocs, 0);
243 std::vector<double> glons(Gnlocs, 0);
245 dist_types[i].get(
"specs.latitude", glats);
246 dist_types[i].get(
"specs.longitude", glons);
251 std::vector<std::size_t> Groups(Gnlocs, 0);
252 if (dist_types[i].has(
"specs.obsgrouping")) {
253 Groups = dist_types[i].getUnsignedVector(
"specs.obsgrouping");
255 std::iota(Groups.begin(), Groups.end(), 0);
259 std::vector<std::size_t> Index;
260 std::vector<std::size_t> Recnums;
261 for (std::size_t j = 0; j < Gnlocs; ++j) {
262 std::size_t RecNum = Groups[j];
263 eckit::geometry::Point2 point(glons[j], glats[j]);
264 TestDist->assignRecord(RecNum, j, point);
265 if (TestDist->isMyRecord(RecNum)) {
267 Recnums.push_back(RecNum);
270 TestDist->computePatchLocs();
272 testDistribution(dist_types[i], MyRankConfig, TestDist.get(), Index, Recnums);
282 const eckit::LocalConfiguration topLevelConf(::test::TestEnvironment::config());
284 const util::DateTime winBegin(topLevelConf.getString(
"window begin"));
285 const util::DateTime winEnd(topLevelConf.getString(
"window end"));
287 const eckit::mpi::Comm & MpiComm = oops::mpi::world();
288 const std::size_t MyRank = MpiComm.rank();
290 const eckit::LocalConfiguration & obsConf = topLevelConf.getSubConfiguration(
"observations");
292 for (
const eckit::LocalConfiguration & conf : obsConf.getSubConfigurations()) {
293 eckit::LocalConfiguration obsspaceConf(conf,
"obs space");
295 obsParams.validateAndDeserialize(obsspaceConf);
300 std::string MyRankConfName =
"specs.rank" + std::to_string(MyRank);
301 eckit::LocalConfiguration MyRankConf(conf, MyRankConfName);
302 oops::Log::debug() <<
"MyRankConf: "
303 << MyRankConfName <<
": " << MyRankConf << std::endl;
317 std::string
testid()
const override {
return "test::Distribution";}
320 std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
322 ts.emplace_back(
CASE(
"distribution/Distribution/testConstructor")
324 ts.emplace_back(
CASE(
"distribution/Distribution/testDistributionConstructedManually")
326 ts.emplace_back(
CASE(
"distribution/Distribution/testDistributionConstructedByObsSpace")
static std::unique_ptr< Distribution > create(const eckit::mpi::Comm &comm, const eckit::Configuration &config)
Create a Distribution object implementing a particular method of distributing observations across mul...
class for distributing obs across multiple process elements
virtual void allGatherv(std::vector< size_t > &x) const =0
Gather observation data from all processes and deliver the combined data to all processes.
virtual void patchObs(std::vector< bool > &isPatchObs) const =0
Sets each element of the provided vector to true if the corresponding location is a "patch obs",...
virtual size_t globalUniqueConsecutiveLocationIndex(size_t loc) const =0
Map the index of a location held on the calling process to the index of the corresponding element of ...
void register_tests() const override
void clear() const override
std::string testid() const override
CASE("Derived variable, unit conversion, and exception checking methods")
void testDistribution(const eckit::Configuration &config, const eckit::Configuration &MyRankConfig, const ioda::Distribution *TestDist, const std::vector< std::size_t > &Index, const std::vector< std::size_t > &Recnums)
void testDistributionConstructedByObsSpace()
void testDistributionConstructedManually()