IODA
ObsSpacePutDbChannels.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-2021 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_OBSSPACEPUTDBCHANNELS_H_
9 #define TEST_IODA_OBSSPACEPUTDBCHANNELS_H_
10 
11 #include <memory>
12 #include <string>
13 #include <vector>
14 
15 #include <boost/make_unique.hpp>
16 #include "Eigen/Core"
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 
25 #include "ioda/Engines/HH.h"
26 #include "ioda/ObsSpace.h"
27 
28 namespace ioda {
29 namespace test {
30 
31 CASE("ioda/ObsSpace/testPutDb") {
32  const auto &topLevelConf = ::test::TestEnvironment::config();
33 
34  util::DateTime bgn(topLevelConf.getString("window begin"));
35  util::DateTime end(topLevelConf.getString("window end"));
36 
37  std::vector<eckit::LocalConfiguration> confs;
38  topLevelConf.get("observations", confs);
39 
40  for (const eckit::LocalConfiguration & conf : confs) {
41  eckit::LocalConfiguration obsconf(conf, "obs space");
43  obsparams.validateAndDeserialize(obsconf);
44 
45  std::unique_ptr<ObsSpace> obsspace = boost::make_unique<ObsSpace>(
46  obsparams, oops::mpi::world(), bgn, end, oops::mpi::myself());
47 
48  const Dimensions_t nlocs = obsspace->nlocs();
49  const Dimensions_t nchans = obsspace->nchans();
50  const bool hasChannels = nchans != 0;
51 
52  std::vector<float> testVec1(nlocs), testVec2(nlocs);
53  std::iota(testVec1.begin(), testVec1.end(), 1.0f);
54  std::iota(testVec2.begin(), testVec2.end(), 2.0f);
55  obsspace->put_db("DummyGroup", "multi_dimensional_var_2", testVec1);
56  obsspace->put_db("DummyGroup", "multi_dimensional_var_4", testVec2);
57  obsspace->put_db("MetaData", "single_dimensional_var_2", testVec1);
58  obsspace->put_db("DummyGroup", "single_dimensional_var", testVec1);
59  if (hasChannels) {
60  // Channel 1000000 does not exist
61  EXPECT_THROWS(obsspace->put_db("DummyGroup", "multi_dimensional_var_1000000", testVec1));
62  // The variable single_dimensional_var already exists, but is not associated with the
63  // nchans dimension
64  EXPECT_THROWS(obsspace->put_db("DummyGroup", "single_dimensional_var_2", testVec1));
65  }
66 
67  // Call the ObsSpace close function to force an output file to be written
68  obsspace->save();
69 
70  // Read the output file and check its contents are correct
71 
72  const std::string fileName = uniquifyFileName(obsconf.getString("obsdataout.obsfile"), 0, -1);
75 
76  if (hasChannels) {
77  const Variable nchansVar = group.vars.open("nchans");
78  const std::vector<int> channels = nchansVar.readAsVector<int>();
79  const std::size_t channel2Index =
80  std::find(channels.begin(), channels.end(), 2) - channels.begin();
81  const std::size_t channel4Index =
82  std::find(channels.begin(), channels.end(), 4) - channels.begin();
83 
84  {
85  const Variable var = group.vars.open("DummyGroup/multi_dimensional_var");
86  const Dimensions dims = var.getDimensions();
87  EXPECT_EQUAL(dims.dimensionality, 2);
88  const std::vector<Dimensions_t> expectedDimsCur{nlocs, nchans};
89  EXPECT_EQUAL(dims.dimsCur, expectedDimsCur);
90 
91  Eigen::ArrayXXf values;
92  var.readWithEigenRegular(values);
93  std::vector<float> channelValues(nlocs);
94  for (std::size_t loc = 0; loc < nlocs; ++loc)
95  channelValues[loc] = values(loc, channel2Index);
96  EXPECT_EQUAL(channelValues, testVec1);
97 
98  for (std::size_t loc = 0; loc < nlocs; ++loc)
99  channelValues[loc] = values(loc, channel4Index);
100  EXPECT_EQUAL(channelValues, testVec2);
101  }
102  } else { // has no channels
103  {
104  const Variable var = group.vars.open("DummyGroup/multi_dimensional_var_2");
105  const Dimensions dims = var.getDimensions();
106  EXPECT_EQUAL(dims.dimensionality, 1);
107  const std::vector<Dimensions_t> expectedDimsCur{nlocs};
108  EXPECT_EQUAL(dims.dimsCur, expectedDimsCur);
109 
110  const std::vector<float> values = var.readAsVector<float>();
111  EXPECT_EQUAL(values, testVec1);
112  }
113 
114  {
115  const Variable var = group.vars.open("DummyGroup/multi_dimensional_var_4");
116  const Dimensions dims = var.getDimensions();
117  EXPECT_EQUAL(dims.dimensionality, 1);
118  const std::vector<Dimensions_t> expectedDimsCur{nlocs};
119  EXPECT_EQUAL(dims.dimsCur, expectedDimsCur);
120 
121  const std::vector<float> values = var.readAsVector<float>();
122  EXPECT_EQUAL(values, testVec2);
123  }
124  }
125 
126  {
127  const Variable var = group.vars.open("DummyGroup/single_dimensional_var");
128  const Dimensions dims = var.getDimensions();
129  EXPECT_EQUAL(dims.dimensionality, 1);
130  const std::vector<Dimensions_t> expectedDimsCur{nlocs};
131  EXPECT_EQUAL(dims.dimsCur, expectedDimsCur);
132 
133  const std::vector<float> values = var.readAsVector<float>();
134  EXPECT_EQUAL(values, testVec1);
135  }
136 
137  {
138  const Variable var = group.vars.open("MetaData/single_dimensional_var_2");
139  const Dimensions dims = var.getDimensions();
140  EXPECT_EQUAL(dims.dimensionality, 1);
141  const std::vector<Dimensions_t> expectedDimsCur{nlocs};
142  EXPECT_EQUAL(dims.dimsCur, expectedDimsCur);
143 
144  const std::vector<float> values = var.readAsVector<float>();
145  EXPECT_EQUAL(values, testVec1);
146  }
147  }
148 }
149 
150 class ObsSpacePutDbChannels : public oops::Test {
151  private:
152  std::string testid() const override {return "test::ObsSpacePutDbChannels";}
153 
154  void register_tests() const override {}
155 
156  void clear() const override {}
157 };
158 
159 // -----------------------------------------------------------------------------
160 
161 } // namespace test
162 } // namespace ioda
163 
164 #endif // TEST_IODA_OBSSPACEPUTDBCHANNELS_H_
HDF5 engine.
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
Variables store data!
Definition: Variable.h:680
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
virtual Variable open(const std::string &name) const
Open a Variable by name.
virtual Dimensions getDimensions() const
Definition: Variable.cpp:160
std::vector< DataType > readAsVector(const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all) const
Read the variable into a new vector. Python convenience function.
Definition: Variable.h:502
Variable_Implementation readWithEigenRegular(EigenClass &res, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all) const
Read data into an Eigen::Array, Eigen::Matrix, Eigen::Map, etc.
Definition: Variable.h:551
std::string testid() const override
IODA_DL Group openFile(const std::string &filename, BackendOpenModes mode, HDF5_Version_Range compat=defaultVersionRange())
Open a ioda::Group backed by an HDF5 file.
Definition: HH.cpp:147
@ Read_Only
Open the file in read-only mode.
CASE("Derived variable, unit conversion, and exception checking methods")
std::string uniquifyFileName(const std::string &fileName, const std::size_t rankNum, const int timeRankNum)
uniquify the output file name
Definition: IodaUtils.cc:343
Describes the dimensions of an Attribute or Variable.
Definition: Dimensions.h:22
std::vector< Dimensions_t > dimsCur
The dimensions of the data.
Definition: Dimensions.h:23
Dimensions_t dimensionality
The dimensionality (rank) of the data.
Definition: Dimensions.h:25