IODA
test_append.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020 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 #include <math.h>
9 
10 #include <algorithm>
11 #include <cstdlib>
12 #include <iostream>
13 #include <numeric>
14 #include <typeinfo>
15 
16 #include "Eigen/Dense"
17 #include "eckit/testing/Test.h"
18 #include "ioda/Engines/Factory.h"
19 #include "ioda/Exception.h"
20 #include "ioda/Layout.h"
21 #include "ioda/ObsGroup.h"
22 
23 #include "ioda/testconfig.h"
24 
25 void test_obsgroup_helper_funcs(std::string backendType, std::string fileName,
26  const std::string mappingFile = "") {
27  using namespace ioda;
28 
29  // Create test data
30  const int locations = 40;
31  const int channels = 30;
32  const int locationsX2 = 2 * locations;
33 
34  // Build data that holds 2 chunks (each chunk is locations in size)
35  // to see if can write the first chunk, resize the variable and write
36  // the second chunk.
37 
38  // Set nlocs (size: 2*locations) and nchans (size: channels) coordinate values
39  // nlocs set to 0..nlocs-1, and nchans set to 1..nchans
40  std::vector<int> nLocs(locationsX2);
41  std::iota(nLocs.begin(), nLocs.end(), 0);
42 
43  std::vector<int> nChans(channels);
44  std::iota(nChans.begin(), nChans.end(), 1);
45 
46  Eigen::ArrayXXf myDataExpected(locationsX2, channels);
47  std::vector<float> myLonExpected(locationsX2);
48  std::vector<float> myLatExpected(locationsX2);
49  auto mid_loc = static_cast<float>(locations);
50  auto mid_chan = static_cast<float>(channels) / 2.0f;
51  for (std::size_t i = 0; i < locationsX2; ++i) {
52  myLonExpected[i] = static_cast<float>(i % 8) * 3.0f;
53  myLatExpected[i] = static_cast<float>(i / 8) * 3.0f; // NOLINT(bugprone-integer-division)
54  for (std::size_t j = 0; j < channels; ++j) {
55  float del_i = static_cast<float>(i) - mid_loc;
56  float del_j = static_cast<float>(j) - mid_chan;
57  myDataExpected(i, j) = sqrt(del_i * del_i + del_j * del_j);
58  }
59  }
60 
61  // Split the data into two chunks
62  Eigen::ArrayXXf myDataExpected1(locations, channels);
63  Eigen::ArrayXXf myDataExpected2(locations, channels);
64  myDataExpected1 = myDataExpected.block<locations, channels>(0, 0);
65  myDataExpected2 = myDataExpected.block<locations, channels>(locations, 0);
66 
67  std::vector<float> myLatExpected1(myLatExpected.begin(), myLatExpected.begin() + locations);
68  std::vector<float> myLatExpected2(myLatExpected.begin() + locations, myLatExpected.end());
69  std::vector<float> myLonExpected1(myLonExpected.begin(), myLonExpected.begin() + locations);
70  std::vector<float> myLonExpected2(myLonExpected.begin() + locations, myLonExpected.end());
71  std::vector<int> nLocs1(nLocs.begin(), nLocs.begin() + locations);
72  std::vector<int> nLocs2(nLocs.begin() + locations, nLocs.end());
73 
74  // Create a backend
75  Engines::BackendNames backendName;
77  if (backendType == "file" || backendType == "fileRemapped") {
78  backendName = Engines::BackendNames::Hdf5File;
79 
80  backendParams.fileName = fileName;
83  } else if (backendType == "memory") {
84  backendName = Engines::BackendNames::ObsStore;
85  } else {
86  throw Exception("Unrecognized backend type", ioda_Here())
87  .add("backendType", backendType);
88  }
89  Group backend = constructBackend(backendName, backendParams);
90 
91  // Create an ObsGroup object and attach the backend
92  ObsGroup og;
93  if (backendType != "fileRemapped") {
95  backend, {
96  NewDimensionScale<int>("nlocs", locations, ioda::Unlimited, locations),
97  NewDimensionScale<int>("nchans", channels, channels, channels)
98  });
99  } else {
101  backend,
102  {
103  NewDimensionScale<int>(
104  "nlocs", locations, ioda::Unlimited, locations),
105  NewDimensionScale<int>(
106  "nchans", channels, channels, channels) },
108  mappingFile, {"nlocs", "nchans"}));
109  }
110  Variable nlocs_var = og.vars.open("nlocs");
111  nlocs_var.write(nLocs1);
112 
113  Variable nchans_var = og.vars["nchans"];
114  nchans_var.write(nChans);
115 
116  // Set up creation parameters for variables
118  float_params.chunk = true;
119  float_params.compressWithGZIP();
120  float_params.setFillValue<float>(-999);
121 
122  Variable obs_var;
123  Variable lat_var;
124  Variable lon_var;
125  if (backendType == "fileRemapped") {
126  obs_var = og.vars.createWithScales<float>("ObsValue_renamed/myObs_renamed", {nlocs_var, nchans_var}, float_params);
127 
128  og.vars.createWithScales<float>("MetaData_renamed/latitude_renamed", {nlocs_var}, float_params);
129  lat_var = og.vars.open("MetaData/latitude");
130 
131  og.vars.createWithScales<float>("MetaData_renamed/longitude_renamed", {nlocs_var}, float_params);
132  lon_var = og.vars["MetaData/longitude"];
133 
134  // Now testing that creating a variable not specified in mapping throws an exception
135  bool unspecifiedVariableThrows = false;
136  try {
137  og.vars.createWithScales<float>("Foo/bar", {nlocs_var}, float_params);
138  } catch (Exception) {
139  unspecifiedVariableThrows = true;
140  }
141  if (!unspecifiedVariableThrows) {
142  throw Exception("Foo/bar did not throw an exception");
143  }
144  } else {
145  obs_var = og.vars.createWithScales<float>("ObsValue/myObs", {nlocs_var, nchans_var}, float_params);
146 
147  og.vars.createWithScales<float>("MetaData/latitude", {nlocs_var}, float_params);
148  lat_var = og.vars.open("MetaData/latitude");
149 
150  og.vars.createWithScales<float>("MetaData/longitude", {nlocs_var}, float_params);
151  lon_var = og.vars["MetaData/longitude"];
152  }
153 
154  // Add attributes to variables
155  obs_var.atts.add<std::string>("coordinates", {"longitude latitude nchans"}, {1})
156  .add<std::string>("long_name", {"obs I made up"}, {1})
157  .add<std::string>("units", {"K"}, {1})
158  .add<float>("valid_range", {0.0, 50.0}, {2});
159  lat_var.atts.add<std::string>("long_name", {"latitude"}, {1})
160  .add<std::string>("units", {"degrees_north"}, {1})
161  .add<float>("valid_range", {-90.0, 90.0}, {2});
162  lon_var.atts.add<std::string>("long_name", {"longitude"}, {1})
163  .add<std::string>("units", {"degrees_east"}, {1})
164  .add<float>("valid_range", {-360.0, 360.0}, {2});
165 
166  // Write data into group variable structure
167  obs_var.writeWithEigenRegular(myDataExpected1);
168  lat_var.write(myLatExpected1);
169  lon_var.write(myLonExpected1);
170 
171  // Append the second data chunk
172  // resize the nlocs variable - do this before writing
173  og.resize({std::pair<ioda::Variable, ioda::Dimensions_t>(nlocs_var, locationsX2)});
174 
175  // 1D vector selection objects
176  std::vector<ioda::Dimensions_t> memStarts(1, 0);
177  std::vector<ioda::Dimensions_t> memCounts(1, locations);
178  std::vector<ioda::Dimensions_t> fileStarts(1, locations);
179  std::vector<ioda::Dimensions_t> fileCounts(1, locations);
180 
181  ioda::Selection memSelect1D;
182  ioda::Selection fileSelect1D;
183  memSelect1D.extent({locations}).select({ioda::SelectionOperator::SET, memStarts, memCounts});
184  fileSelect1D.select({ioda::SelectionOperator::SET, fileStarts, fileCounts});
185 
186  // 2D selection objects
187  memStarts.push_back(0); // borrow the arrays from above
188  memCounts.push_back(channels);
189  fileStarts.push_back(0);
190  fileCounts.push_back(channels);
191 
192  ioda::Selection memSelect2D;
193  ioda::Selection fileSelect2D;
194  memSelect2D.extent({locations, channels}).select({ioda::SelectionOperator::SET, memStarts, memCounts});
195  fileSelect2D.select({ioda::SelectionOperator::SET, fileStarts, fileCounts});
196 
197  // Write the sencond data chunk
198  nlocs_var.write(nLocs2, memSelect1D, fileSelect1D);
199  obs_var.writeWithEigenRegular(myDataExpected2, memSelect2D, fileSelect2D);
200  lat_var.write(myLatExpected2, memSelect1D, fileSelect1D);
201  lon_var.write(myLonExpected2, memSelect1D, fileSelect1D);
202 
203  // Read data back and check values
204  Eigen::ArrayXXf myData(locationsX2, channels);
205  obs_var.readWithEigenRegular(myData);
206  if (!myData.isApprox(myDataExpected)) {
207  throw Exception("Test obs data mismatch", ioda_Here());
208  }
209 
210  std::vector<float> myLats(locationsX2, 0.0);
211  lat_var.read(myLats);
212  for (std::size_t i = 0; i < locationsX2; ++i) {
213  double check = fabs((myLats[i] / myLatExpected[i]) - 1.0);
214  if (check > 1.0e-3) {
215  throw Exception("Test lats mismatch outside tolerence (1e-3)", ioda_Here())
216  .add(" i", i)
217  .add(" myLatExpected[i]", myLatExpected[i])
218  .add(" myLats[i]", myLats[i]);
219  }
220  }
221 
222  std::vector<float> myLons(locationsX2, 0.0);
223  lon_var.read(myLons);
224  for (std::size_t i = 0; i < locationsX2; ++i) {
225  double check = fabs((myLons[i] / myLonExpected[i]) - 1.0);
226  if (check > 1.0e-3) {
227  throw Exception("Test lons mismatch outside tolerence (1e-3)", ioda_Here())
228  .add(" i", i)
229  .add(" myLonExpected[i]", myLonExpected[i])
230  .add(" myLons[i]", myLons[i]);
231  }
232  }
233 
234  // Some more checks
235  Expects(og.open("ObsValue").vars["myObs"].isDimensionScaleAttached(1, og.vars["nchans"]));
236 }
237 
238 int main(int argc, char** argv) {
239  if (argc < 2) {
240  std::cerr << "Need to specify backend type (file, memory)" << std::endl;
241  return 1;
242  }
243  std::string backendType(argv[1]);
244  try {
245  if (backendType == "file") {
246  std::cout << "Testing file backend" << std::endl;
247  test_obsgroup_helper_funcs(backendType, {"ioda-engines_obsgroup_append-file.hdf5"});
248  } else if (backendType == "memory") {
249  std::cout << "Testing memory backend" << std::endl;
250  test_obsgroup_helper_funcs(backendType, {""});
251  } else if (backendType == "fileRemapped") {
252  std::cout << "Testing ODB Data Layout Policy with explicit mapping file" << std::endl;
253  std::string mappingFile = std::string(IODA_ENGINES_TEST_SOURCE_DIR)
254  + "/obsgroup/odb_default_name_map.yaml";
255  test_obsgroup_helper_funcs(backendType, {"append-remapped.hdf5"}, mappingFile);
256  std::cout << "Testing ODB Data Layout Policy with explicit mapping file" << std::endl;
257  mappingFile = std::string(IODA_ENGINES_TEST_SOURCE_DIR)
258  + "/obsgroup/odb_incomplete_name_map.yaml";
259  bool failedWhenNotAllVarsRemapped = false;
260  try {
261  test_obsgroup_helper_funcs(backendType, {"append-remapped.hdf5"}, mappingFile);
262  } catch (const std::exception &e) {
263  failedWhenNotAllVarsRemapped = true;
264  }
265  bool odbGroupFailedWithoutMapping = false;
266  // Layout_ObsGroup_ODB throws an exception if mapping yaml file not provided
267  try {
268  test_obsgroup_helper_funcs(backendType, {"ioda-engines_obsgroup_append-remapped-file.hdf5"});
269  } catch (const std::exception& e) {
270  odbGroupFailedWithoutMapping = true;
271  }
272  assert(odbGroupFailedWithoutMapping && failedWhenNotAllVarsRemapped);
273  } else {
274  throw ioda::Exception("Unrecognized backend type:", ioda_Here())
275  .add("Backend type", backendType);
276  }
277  } catch (const std::exception& e) {
279  return 1;
280  }
281  return 0;
282 }
IODA's error system.
Definitions for setting up backends with file and memory I/O.
Contains definitions for how data are arranged in ioda internally.
Interfaces for ioda::ObsGroup and related classes.
The ioda exception class.
Definition: Exception.h:54
Exception & add(const std::string &key, const T value)
Add a key-value pair to the error message.
Definition: Exception.h:75
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
An ObsGroup is a specialization of a ioda::Group. It provides convenience functions and guarantees th...
Definition: ObsGroup.h:32
static ObsGroup generate(Group &emptyGroup, const NewDimensionScales_t &fundamentalDims, std::shared_ptr< const detail::DataLayoutPolicy > layout=nullptr)
Create an empty ObsGroup and populate it with the fundamental dimensions.
Definition: ObsGroup.cpp:72
A Selection represents the bounds of the data, in ioda or in userspace, that you are reading or writi...
Definition: Selection.h:48
Variables store data!
Definition: Variable.h:680
DerivedHasAtts add(const std::string &attrname, ::gsl::span< const DataType > data, const ::std::vector< Dimensions_t > &dimensions)
Create and write an Attribute, for arbitrary dimensions.
static std::shared_ptr< const DataLayoutPolicy > generate(const std::string &polid="")
Factory generator.
Definition: Layout.cpp:28
Variable_Implementation writeWithEigenRegular(const EigenClass &d, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all)
Write an Eigen object (a Matrix, an Array, a Block, a Map).
Definition: Variable.h:361
Has_Attributes atts
Attributes.
Definition: Variable.h:71
virtual Variable read(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all) const
Read the Variable - as char array. Ordering is row-major.
Definition: Variable.cpp:330
virtual Variable write(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all)
The fundamental write function. Backends overload this function to implement all write operations.
Definition: Variable.cpp:317
BackendNames
Backend names.
Definition: Factory.h:28
IODA_DL Group constructBackend(BackendNames name, BackendCreationParameters &params)
This is a simple factory style function that will instantiate a different backend based on a given na...
Definition: Factory.cpp:124
@ Hdf5File
HDF5 file access.
@ ObsStore
ObsStore in-memory.
@ Truncate_If_Exists
If the file already exists, overwrite it.
Selection & extent(const VecDimensions_t &sz)
Provide the dimensions of the object that you are selecting from.
Definition: Selection.h:111
Selection & select(const SingleSelection &s)
Append a new selection.
Definition: Selection.h:103
constexpr int Unlimited
Specifies that a dimension is resizable to infinity.
IODA_DL void unwind_exception_stack(const std::exception &e, std::ostream &out=std::cerr, int level=0)
Convenience function for unwinding an exception stack.
Definition: Exception.cpp:48
#define ioda_Here()
Used to specify backend creation-time properties.
Definition: Factory.h:59
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57
VariableCreationParameters & setFillValue(DataType fill)
Definition: Has_Variables.h:69
bool chunk
Do we chunk this variable? Required for extendible / compressible Variables.
Definition: Has_Variables.h:84
int main(int argc, char **argv)
void test_obsgroup_helper_funcs(std::string backendType, std::string fileName, const std::string mappingFile="")
Definition: test_append.cpp:25