IODA
data-selections/test.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 /// This program tests that Variable read and write selections work as expected for certain engines.
8 
9 #include <Eigen/Dense>
10 #include <iostream>
11 #include <vector>
12 
13 #include "ioda/Engines/Factory.h"
14 #include "ioda/Exception.h"
15 #include "ioda/Group.h"
16 
18  // test_data1 is a 4x4 matrix that looks like:
19  // 1 2 3 4
20  // 5 6 7 8
21  // 9 10 11 12
22  // 13 14 15 16
23  Eigen::ArrayXXi test_data1(4, 4);
24  test_data1 << 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16;
25 
26  // std::cerr << test_data1 << std::endl << std::endl;
27 
28  // Make a variable in the file and write test_data1.
29  ioda::Variable file_test_data1 = g.vars.create<int>("test_data1", {4, 4}).writeWithEigenRegular(test_data1);
30 
31  // overlay_data1 is a 2x2 matrix:
32  // 17 18
33  // 19 20
34  Eigen::ArrayXXi overlay_data1(2, 2);
35  overlay_data1 << 17, 18, 19, 20;
36 
37  // std::cerr << overlay_data1 << std::endl << std::endl;
38 
39  // Write the 2x2 overlay on top of the existing data.
40 
41  // We use two selectors. The first selects the data in memory (i.e. the inputs that we
42  // provide). The second selects the range of the data in ioda.
43  // Because we are not writing the full dimensions of the variable, we need
44  // to specify selectors for both user memory and ioda.
45 
46  // The memory selector defines a hyperslab, starting at (0,0), with size (2,2).
47  // Since the 2x2 in-memory object has different dimensions than the object stored in
48  // ioda (4x4), we need to pass along the dimensions of our object to properly write
49  // rows and columns. This is accomplished by setting extent({2,2}).
50  // The ioda 'file' selector also defines a hyperslab, but it starts at (2,2) with size (2,2).
51  // So, we are writing the lower right quadrant of the matrix.
52  file_test_data1.writeWithEigenRegular(
53  overlay_data1, ioda::Selection().extent({2, 2}).select({ioda::SelectionOperator::SET, {0, 0}, {2, 2}}),
54  ioda::Selection().select({ioda::SelectionOperator::SET, {2, 2}, {2, 2}}));
55  // file_test_data1 should now look like:
56  // 1 2 3 4
57  // 5 6 7 8
58  // 9 10 17 18
59  // 13 14 19 20
60 
61  // Now, let's write this 2x2 object, again, to the same output.
62  file_test_data1.writeWithEigenRegular(
63  overlay_data1, ioda::Selection().extent({2, 2}).select({ioda::SelectionOperator::SET, {0, 0}, {2, 2}}),
64  ioda::Selection().select({ioda::SelectionOperator::SET, {1, 1}, {2, 2}}));
65  // file_test_data1 should now look like:
66  // 1 2 3 4
67  // 5 17 18 8
68  // 9 19 20 18
69  // 13 14 19 20
70 
71  // Let's use a different type of selector: the point selector.
72  // We will set (3,0) = 21, and (0,3) = 22.
73  std::vector<int> point_vals{21, 22};
74  file_test_data1.write(gsl::make_span(point_vals), // Values of the two points
75  ioda::Selection().extent({2, 1}).select(
76  {ioda::SelectionOperator::SET, {{0, 0}, {1, 0}}}), // Writing two points
78  {{3, 0}, {0, 3}}})); // Location where we write the data.
79  // file_test_data1 should now look like:
80  // 1 2 3 22
81  // 5 17 18 8
82  // 9 19 20 18
83  // 21 14 19 20
84 
85  // And check our read function.
86  Eigen::ArrayXXi reference(4, 4);
87  reference << 1, 2, 3, 22, 5, 17, 18, 8, 9, 19, 20, 18, 21, 14, 19, 20;
88 
89  Eigen::ArrayXXi check;
90  file_test_data1.readWithEigenRegular(check);
91 
92  bool r = check.isApprox(reference);
93  if (!r) throw std::logic_error("Test failure.");
94 }
95 
96 int main(int argc, char** argv) {
97  using namespace ioda;
98  using namespace std;
99  try {
100  auto f = Engines::constructFromCmdLine(argc, argv, "test-data-selections.hdf5");
102  } catch (const std::exception& e) {
104  return 1;
105  }
106  return 0;
107 }
IODA's error system.
Definitions for setting up backends with file and memory I/O.
Interfaces for ioda::Group and related classes.
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
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
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
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
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
void test_group_backend_engine(ioda::Group g)
This program tests that Variable read and write selections work as expected for certain engines.
int main(int argc, char **argv)
IODA_DL Group constructFromCmdLine(int argc, char **argv, const std::string &defaultFilename)
This is a wrapper function around the constructBackend function for creating a backend based on comma...
Definition: Factory.cpp:21
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