IODA
complex-objects/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 <iostream>
8 #include <vector>
9 
10 #include "ioda/Engines/Factory.h"
11 #include "ioda/Exception.h"
12 #include "ioda/Group.h"
13 
14 // These tests really need a better check system.
15 // Boost unit tests would have been excellent here.
16 
17 template <class T>
18 void check_equal(const std::string& name, const std::vector<T>& vals, const std::vector<T>& ref) {
19  std::ostringstream outerr;
20  outerr << "Check of " << name << " failed. Error: ";
21 
22  try {
23  if (vals.size() != ref.size()) {
24  outerr << "check_equal failed. vals.size() = " << vals.size() << ", and ref.size() = " << ref.size()
25  << ".\n";
26  throw;
27  }
28  for (size_t i = 0; i < vals.size(); ++i) {
29  if (vals[i] != ref[i]) {
30  outerr << "check_equal failed at index " << i << ". vals[" << i << "] = " << vals[i] << ", and ref["
31  << i << "] = " << ref[i] << ".\n";
32  throw;
33  }
34  }
35  } catch (...) {
36  throw std::logic_error(outerr.str().c_str());
37  }
38 }
39 
40 int main(int argc, char** argv) {
41  using namespace ioda;
42  using namespace std;
43  try {
44  // HDF5 file backend
45  auto f = Engines::constructFromCmdLine(argc, argv, "test-complex-objects.hdf5");
46 
47  // These tests try to read and write non-fundamental datatypes.
48  // These types may be enums, complex structures, strings or arrays.
49 
50  // We try to both write and read these with both attributes and variables.
51 
52  // Array types
53  // TODO(rhoneyager): Marshalling.h needs better template expansion to handle these types automatically.
54  /*
55  {
56  const int adata[2]{ 1,2 };
57  f.atts.create<int[2], Types::GetType_Wrapper<int,2>>("c_array_int_2")
58  .write<int[2], ioda::Object_Accessor<int[2]>, Types::GetType_Wrapper<int, 2>>(adata);
59  }
60  {
61  int check_adata[2]{ 0,0 };
62  auto a = f.atts.open("c_array_int_2");
63  a.read<int[2], ioda::Object_Accessor<int[2]>, Types::GetType_Wrapper<int, 2>>(check_adata);
64  check_equal("c_array_int_2", std::vector<int>{check_adata[0], check_adata[1]}, { 1,2 });
65  }
66  {
67  const array<int, 2> adata{ 3,4 };
68  f.atts.add<array<int, 2>>("std_array_int_2", adata); // 3, 4 is a single datum.
69 
70  f.vars.create<array<int, 2>>("v_std_array_int_2", { 1 }).write<array<int, 2>>(adata);
71  }
72  {
73  // Check the attribute read
74  int check_adata[2]; // Can't use std::array<int,2> because I need to expand. See a.read<int[2]...>
75  auto a = f.atts.open("std_array_int_2");
76  a.read<int[2], ioda::Object_Accessor<int[2]>, Types::GetType_Wrapper<int, 2>>(check_adata);
77  check_equal("std_array_int_2", std::vector<int>{check_adata[0], check_adata[1]}, { 3,4 });
78 
79  // Check the string read
80  auto v = f.vars["v_std_array_int_2"];
81  v.read<int[2], ioda::Object_Accessor<int[2]>, Types::GetType_Wrapper<int, 2>>(check_adata);
82  check_equal("v_std_array_int_2", std::vector<int>{check_adata[0], check_adata[1]}, { 3,4 });
83  }
84 
85  */
86 
87  // String types
88 
89  // Note the curly braces around "String 1". Needed because we are constructing a
90  // string from a character array, and we don't want to clash with the span-based
91  // function signature.
92  f.atts.add<string>("Str_1", {"String 1"});
93  f.atts.add<string>("Str_2", {"Hi Steve!", "Hi Ryan!"});
94  {
95  vector<string> v_data;
96  f.atts.read<string>("Str_1", v_data);
97  check_equal("Str_1", v_data, {"String 1"});
98 
99  f.atts.read<string>("Str_2", v_data);
100  check_equal("Str_2", v_data, {"Hi Steve!", "Hi Ryan!"});
101  }
102 
103  f.vars.create<string>("v_Str_1", {1}).write<string>(std::vector<std::string>{"var String 1"});
104  f.vars.create<string>("v_Str_2", {2})
105  .write<string>(std::vector<std::string>{"var String 2.1", "var String 2.2"});
106  f.vars.create<string>("v_Str_3", {2, 2})
107  .write<string>(std::vector<std::string>{"var String 3 [0,0]", "var String 3 [0,1]",
108  "var String 3 [1,0]", "var String 3 [1,1]"});
109  {
110  vector<string> v_data;
111  f.vars["v_Str_1"].read(v_data);
112  check_equal("v_Str_1", v_data, {"var String 1"});
113 
114  f.vars["v_Str_2"].read(v_data);
115  check_equal("v_Str_2", v_data, {"var String 2.1", "var String 2.2"});
116 
117  f.vars["v_Str_3"].read(v_data);
118  check_equal("v_Str_3", v_data,
119  {"var String 3 [0,0]", "var String 3 [0,1]", "var String 3 [1,0]", "var String 3 [1,1]"});
120  }
121 
122  // Check if string variable gets initialized to the fill value
124  string fillString("I_am_fill");
125  params.setFillValue<string>(fillString);
126  f.vars.create<string>("Str_w_fill", {2, 2}, {2, 2}, params);
127  {
128  vector<string> v_data;
129  f.vars["Str_w_fill"].read(v_data);
130  check_equal("Str_w_fill", v_data, {"I_am_fill", "I_am_fill", "I_am_fill", "I_am_fill"});
131  }
132  } catch (const std::exception& e) {
134  return 1;
135  }
136  return 0;
137 }
IODA's error system.
Definitions for setting up backends with file and memory I/O.
Interfaces for ioda::Group and related classes.
int main(int argc, char **argv)
void check_equal(const std::string &name, const std::vector< T > &vals, const std::vector< T > &ref)
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
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57