IODA
persist/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 #include <cmath>
8 #include <iostream>
9 #include <vector>
10 
11 #include "ioda/Engines/Factory.h"
12 #include "ioda/Exception.h"
13 #include "ioda/Group.h"
14 
15 // These tests really need a better check system.
16 // Boost unit tests would have been excellent here.
17 
18 // Check dimensions against expected values
19 void check_dimensions(const std::string& name, const ioda::Dimensions& dims,
20  const std::vector<ioda::Dimensions_t>& exp_dims) {
21  std::string err_msg;
22  std::string exp_msg;
23  std::string res_msg;
24  // Check rank of dimensions
25  if (dims.dimensionality != gsl::narrow<ioda::Dimensions_t>(exp_dims.size())) {
26  err_msg = name + std::string(": dimensionality not equal to expected value");
27  exp_msg = std::string(" expected dimensionality");
28  res_msg = std::string(" ") + name + std::string(": dimensionality");
29  throw ioda::Exception(err_msg.c_str(), ioda_Here())
30  .add(exp_msg, exp_dims.size())
31  .add(res_msg, dims.dimensionality);
32  }
33 
34  // Check dimension sizes
35  for (std::size_t i = 0; i < exp_dims.size(); ++i) {
36  if (dims.dimsCur[i] != exp_dims[i]) {
37  err_msg =
38  name + std::string(": dimension ") + std::to_string(i) + std::string(" not equal to expected value");
39  exp_msg = std::string(" expected dimsCur[") + std::to_string(i) + std::string("]");
40  res_msg = std::string(" ") + name + std::string(": dimsCur[") + std::to_string(i) + std::string("]");
41  throw ioda::Exception(err_msg.c_str(), ioda_Here())
42  .add(exp_msg, exp_dims[i])
43  .add(res_msg, dims.dimsCur[i]);
44  }
45  }
46 }
47 
48 // Check double data against expected values
49 void check_data(const std::string& name, const std::vector<double>& data,
50  const std::vector<double>& exp_data) {
51  std::string err_msg;
52  std::string exp_msg;
53  std::string res_msg;
54  // Check size of data
55  if (data.size() != exp_data.size()) {
56  err_msg = name + std::string(": data size not equal to expected value");
57  exp_msg = std::string(" expected size");
58  res_msg = std::string(" ") + name + std::string(": size");
59  throw ioda::Exception(err_msg.c_str(), ioda_Here())
60  .add(exp_msg, exp_data.size())
61  .add(res_msg, data.size());
62  }
63 
64  // Check data values
65  for (std::size_t i = 0; i < exp_data.size(); ++i) {
66  double check = fabs((data[i] / exp_data[i]) - 1.0);
67  if (check > 1.0e-3) {
68  err_msg = name + std::string(": element ") + std::to_string(i) +
69  std::string(" not within tolerence (1e-3) of expected value");
70  exp_msg = std::string(" expected data[") + std::to_string(i) + std::string("]");
71  res_msg = std::string(" ") + name + std::string(": data[") + std::to_string(i) + std::string("]");
72  throw ioda::Exception(err_msg.c_str(), ioda_Here())
73  .add(exp_msg, exp_data[i])
74  .add(res_msg, data[i]);
75  }
76  }
77 }
78 
79 // Check int data against expected values
80 void check_data(const std::string& name, const std::vector<int>& data, const std::vector<int>& exp_data) {
81  std::string err_msg;
82  std::string exp_msg;
83  std::string res_msg;
84  // Check size of data
85  if (data.size() != exp_data.size()) {
86  err_msg = name + std::string(": data size not equal to expected value");
87  exp_msg = std::string(" expected size");
88  res_msg = std::string(" ") + name + std::string(": size");
89  throw ioda::Exception(err_msg.c_str(), ioda_Here())
90  .add(exp_msg, exp_data.size())
91  .add(res_msg, data.size());
92  }
93 
94  // Check data values
95  for (std::size_t i = 0; i < exp_data.size(); ++i) {
96  if (data[i] != exp_data[i]) {
97  err_msg =
98  name + std::string(": element ") + std::to_string(i) + std::string(" not equal to expected value");
99  exp_msg = std::string(" expected data[") + std::to_string(i) + std::string("]");
100  res_msg = std::string(" ") + name + std::string(": data[") + std::to_string(i) + std::string("]");
101  throw ioda::Exception(err_msg.c_str(), ioda_Here())
102  .add(exp_msg, exp_data[i])
103  .add(res_msg, data[i]);
104  }
105  }
106 }
107 
108 // Be sure to keep this routine and check_group_structure in sync.
110  // Create some sub groups
111  auto g_c1 = g.create("Child1");
112  auto g_c2 = g.create("Child2");
113 
114  // Place attributes in the sub groups
115  g_c1.atts.template add<double>("double_single", {3.14159}, {1});
116 
117  g_c2.atts.template add<int>("int_2x2", {1, 2, 3, 4}, {2, 2});
118 
119  // Place variables in the sub groups
121  params_1.setFillValue<double>(-999);
122  params_1.chunk = true;
123  params_1.compressWithGZIP();
124  auto v_double = g_c1.vars.template create<double>("double", {2, 2}, {2, 2}, params_1);
125  v_double.write<double>({10.0, 11.0, 12.0, 13.0});
126  v_double.atts.template add<int>("int_2x3", {-2, -1, 0, 1, 2, 3}, {2, 3});
127 }
128 
129 // Be sure to keep this routine and build_group_structure in sync.
131  std::vector<ioda::Dimensions_t> exp_dims;
132  std::vector<double> exp_data_d;
133  std::vector<int> exp_data_i;
134 
135  // Verify the sub groups. The open function will throw an exception
136  // if the sub group does not exist.
137  auto g_c1 = g.open("Child1");
138  auto g_c2 = g.open("Child2");
139 
140  // Check the sub group attributes
141  auto attr = g_c1.atts.open("double_single");
142  ioda::Dimensions a_dims = attr.getDimensions();
143  exp_dims = {1};
144  check_dimensions("group attribute: double_single", a_dims, exp_dims);
145  std::vector<double> a_data_d;
146  attr.read(a_data_d);
147  exp_data_d = {3.14159};
148  check_data("group attribute: double_single", a_data_d, exp_data_d);
149 
150  attr = g_c2.atts.open("int_2x2");
151  a_dims = attr.getDimensions();
152  exp_dims = {2, 2};
153  check_dimensions("group attribute: int_2x2", a_dims, exp_dims);
154  std::vector<int> a_data_i;
155  attr.read(a_data_i);
156  exp_data_i = {1, 2, 3, 4};
157  check_data("group attribute: int_2x2", a_data_i, exp_data_i);
158 
159  // Check the sub group variable
160  auto var = g_c1.vars.open("double");
161  ioda::Dimensions v_dims = var.getDimensions();
162  exp_dims = {2, 2};
163  check_dimensions("varable: double", v_dims, exp_dims);
164  std::vector<double> v_data_d;
165  var.read<double>(v_data_d);
166  exp_data_d = {10.0, 11.0, 12.0, 13.0};
167  check_data("variable: double", v_data_d, exp_data_d);
168 
169  attr = var.atts.open("int_2x3");
170  a_dims = attr.getDimensions();
171  exp_dims = {2, 3};
172  check_dimensions("variable attribute: int_2x3", a_dims, exp_dims);
173  attr.read(a_data_i);
174  exp_data_i = {-2, -1, 0, 1, 2, 3};
175  check_data("variable attribute: int_2x3", a_data_i, exp_data_i);
176 }
177 
178 int main(int argc, char** argv) {
179  using namespace ioda;
180  using namespace std;
181  try {
182  auto f = Engines::constructFromCmdLine(argc, argv, "test-persist.hdf5");
183 
184  // Build sub-groups containing variables and attributes
185  // in one function. Then check their contents in another
186  // function call. Do this to make sure that the
187  // group/attribtute/variable structure persists.
190 
191  } catch (const std::exception& e) {
193  return 1;
194  }
195  return 0;
196 }
IODA's error system.
Definitions for setting up backends with file and memory I/O.
Interfaces for ioda::Group 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
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
int main(int argc, char **argv)
void check_data(const std::string &name, const std::vector< double > &data, const std::vector< double > &exp_data)
void check_dimensions(const std::string &name, const ioda::Dimensions &dims, const std::vector< ioda::Dimensions_t > &exp_dims)
void check_group_structure(ioda::Group g)
void build_group_structure(ioda::Group g)
#define ioda_Here()
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
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