IODA
ioda-test-fillvalues.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 template <typename T>
15 bool test_equal(const T& a, const T& b) {
16  return a == b;
17 }
18 
19 template <>
20 bool test_equal(const float& a, const float& b) {
21  return std::abs(a - b) < 0.00001f;
22 }
23 
24 template <>
25 bool test_equal(const double& a, const double& b) {
26  return std::abs(a - b) < 0.00001f;
27 }
28 
29 template <>
30 bool test_equal(const long double& a, const long double& b) {
31  return std::abs(a - b) < 0.00001f;
32 }
33 
34 template <class T>
35 bool testVarFill(ioda::Group& f, const std::string& varname,
36  const T& filldata) // NOLINT(google-runtime-references)
37 {
38  try {
39  std::cout << "Testing variable " << varname << std::endl;
40  std::cout << "\tCreating..." << std::endl;
41  using namespace ioda;
43  params.setFillValue(filldata);
44  f.vars.create<T>(varname, {1}, {1}, params);
45  std::cout << "\t\tSuccess.\n\tReading..." << std::endl;
46 
47  std::vector<T> checkdata;
48  f.vars[varname].read(checkdata);
49  if (checkdata.empty()) {
50  std::cout << "\t\tFailed to read." << std::endl;
51  return false;
52  }
53  std::cout << "\t\tSuccess." << std::endl;
54  std::cout << "\tChecking read value with reference..." << std::endl;
55  if (!test_equal<T>(filldata, checkdata.at(0))) {
56  std::cout << "\t\tFailed check. Ref is '" << filldata << "' and data is '" << checkdata.at(0) << "'."
57  << std::endl;
58  return false;
59  }
60  std::cout << "\t\tSuccess." << std::endl;
61 
62  std::cout << "\tCheck that the variable has a fill value." << std::endl;
63  if (f.vars[varname].hasFillValue()) {
64  std::cout << "\t\tSuccess." << std::endl;
65  } else {
66  std::cout << "\t\tFailed." << std::endl;
67  return false;
68  }
69 
70  std::cout << "\tCheck fill value read." << std::endl;
71  auto fv = f.vars[varname].getFillValue();
72  if (!fv.set_) {
73  std::cout << "\t\tfv.set_ is false. Failed." << std::endl;
74  return false;
75  }
76 
77  if (!test_equal<T>(filldata, ioda::detail::getFillValue<T>(fv))) {
78  std::cout << "\t\tFailure. Ref is " << filldata << " and fv is " << ioda::detail::getFillValue<T>(fv)
79  << "." << std::endl;
80  return false;
81  }
82 
83  std::cout << "\t\tSuccess." << std::endl;
84 
85  return true;
86  } catch (...) {
87  std::cout << "\t\tFailed with exception." << std::endl;
88  return false;
89  }
90 }
91 
92 int main(int argc, char** argv) {
93  using namespace ioda;
94  using namespace std;
95  try {
96  auto f = Engines::constructFromCmdLine(argc, argv, "test-fills.hdf5");
97 
98  // These tests try to read and write fill values.
99  // We try basic numbers, fixed-length strings and variable-length
100  // strings.
101  int good = 0, bad = 0;
102 
103  (testVarFill<std::string>(f, "varlen-string-test", "This is a test")) ? good++ : bad++;
104  (testVarFill<std::string>(f, "varlen-empty_string-test", "")) ? good++ : bad++;
105 
106  (testVarFill<int>(f, "int-test", -999)) ? good++ : bad++;
107  (testVarFill<int32_t>(f, "int32_t-test", -99)) ? good++ : bad++;
108  (testVarFill<uint32_t>(f, "uint32_t-test", 99)) ? good++ : bad++;
109  (testVarFill<int16_t>(f, "int16_t-test", -99)) ? good++ : bad++;
110  (testVarFill<uint16_t>(f, "uint16_t-test", 99)) ? good++ : bad++;
111  (testVarFill<int64_t>(f, "int64_t-test", -99)) ? good++ : bad++;
112  (testVarFill<uint64_t>(f, "uint64_t-test", 99)) ? good++ : bad++;
113  (testVarFill<char>(f, "char-test", 'a')) ? good++ : bad++;
114  (testVarFill<float>(f, "float-test", 3.14f)) ? good++ : bad++;
115  (testVarFill<double>(f, "double-test", 2.7)) ? good++ : bad++;
116  (testVarFill<long double>(f, "long-double-test", 1.428571428571429L)) ? good++ : bad++;
117 
118  // TODO(rhoneyager): Fixed-length string test.
119 
120  std::cout << "\n\nSuccesses: " << good << "\nFailures: " << bad << std::endl;
121  return (bad) ? 1 : 0;
122  } catch (const std::exception& e) {
124  return 1;
125  }
126 }
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
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
virtual Variable create(const std::string &name, const Type &in_memory_dataType, const std::vector< Dimensions_t > &dimensions={1}, const std::vector< Dimensions_t > &max_dimensions={}, const VariableCreationParameters &params=VariableCreationParameters())
Create a Variable without setting its data.
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
bool test_equal(const T &a, const T &b)
int main(int argc, char **argv)
bool testVarFill(ioda::Group &f, const std::string &varname, const T &filldata)
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