IODA
ObsIoWrite.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018 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 #ifndef TEST_IO_OBSIOWRITE_H_
9 #define TEST_IO_OBSIOWRITE_H_
10 
11 #include <algorithm>
12 #include <functional>
13 #include <map>
14 #include <memory>
15 #include <numeric>
16 #include <string>
17 #include <utility>
18 #include <vector>
19 
20 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
21 
22 #include "eckit/config/LocalConfiguration.h"
23 #include "eckit/testing/Test.h"
24 
25 #include "oops/mpi/mpi.h"
26 #include "oops/runs/Test.h"
27 #include "oops/test/TestEnvironment.h"
28 #include "oops/util/DateTime.h"
29 #include "oops/util/FloatCompare.h"
30 #include "oops/util/Logger.h"
31 
32 #include "ioda/core/IodaUtils.h"
33 #include "ioda/distribution/DistributionFactory.h"
34 #include "ioda/io/ObsIo.h"
35 #include "ioda/io/ObsIoFactory.h"
36 #include "ioda/ObsGroup.h"
37 #include "ioda/ObsSpaceParameters.h"
39 
40 namespace ioda {
41 namespace test {
42 
43 // -----------------------------------------------------------------------------
44 // Helper Functions
45 // -----------------------------------------------------------------------------
46 
47 // -----------------------------------------------------------------------------
48 // Test Functions
49 // -----------------------------------------------------------------------------
50 
51 // -----------------------------------------------------------------------------
52 void testWrite() {
53  const eckit::LocalConfiguration conf(::test::TestEnvironment::config());
54  std::vector<eckit::LocalConfiguration> confOspaces = conf.getSubConfigurations("observations");
55  util::DateTime bgn(::test::TestEnvironment::config().getString("window begin"));
56  util::DateTime end(::test::TestEnvironment::config().getString("window end"));
57 
58  for (std::size_t i = 0; i < confOspaces.size(); ++i) {
59  eckit::LocalConfiguration obsConfig;
60  eckit::LocalConfiguration testConfig;
61  confOspaces[i].get("obs space", obsConfig);
62  confOspaces[i].get("test data", testConfig);
63  oops::Log::trace() << "ObsIo testWrite obs space config: " << i << ": "
64  << obsConfig << std::endl;
65  oops::Log::trace() << "ObsIo testWrite test data config: " << i << ": "
66  << obsConfig << std::endl;
67 
69  topParams.validateAndDeserialize(obsConfig);
70  ioda::ObsSpaceParameters obsParams(topParams, bgn, end,
71  oops::mpi::world(), oops::mpi::myself());
72 
73  if (obsParams.top_level_.obsOutFile.value() != boost::none) {
74  std::vector<eckit::LocalConfiguration> writeVarConfigs =
75  testConfig.getSubConfigurations("write variables");
76  std::vector<eckit::LocalConfiguration> writeDimConfigs =
77  testConfig.getSubConfigurations("write dimensions");
78 
79  // Output constructor
80  setOfileParamsFromTestConfig(testConfig, obsParams);
81  std::shared_ptr<ObsIo> obsIo =
83 
84  // Write the test variables
85  for (std::size_t j = 0; j < writeVarConfigs.size(); ++j) {
86  std::string varName = writeVarConfigs[j].getString("name");
87  std::string varType = writeVarConfigs[j].getString("type");
88  std::vector<std::string> varDimNames =
89  writeVarConfigs[j].getStringVector("dims");
90 
91  // Create the variable
92  ioda::Variable var;
93  std::vector<ioda::Variable> varDims;
94  for (std::size_t idim = 0; idim < varDimNames.size(); ++idim) {
95  varDims.push_back(obsIo->vars().open(varDimNames[idim]));
96  }
97 
99  params.chunk = true;
100  params.compressWithGZIP();
101  if (varType == "int") {
102  params.setFillValue<int>(-999);
103  var = obsIo->vars()
104  .createWithScales<int>(varName, varDims, params);
105  } else if (varType == "float") {
106  params.setFillValue<float>(-999);
107  var = obsIo->vars()
108  .createWithScales<float>(varName, varDims, params);
109  } else if (varType == "string") {
110  params.setFillValue<std::string>("fill");
111  var = obsIo->vars()
112  .createWithScales<std::string>(varName, varDims, params);
113  }
114 
115  // Write the variable data
116  if (varType == "int") {
117  std::vector<int> varValues =
118  writeVarConfigs[j].getIntVector("values");
119  var.write<int>(varValues);
120  } else if (varType == "float") {
121  std::vector<float> varValues =
122  writeVarConfigs[j].getFloatVector("values");
123  var.write<float>(varValues);
124  } else if (varType == "string") {
125  std::vector<std::string> varValues =
126  writeVarConfigs[j].getStringVector("values");
127  var.write<std::string>(varValues);
128  }
129  }
130 
131  // Update the variable lists in the ObsIo object
132  obsIo->updateVarDimInfo();
133 
134  // Check if all the variables got written into the file
135  // Dimension scale variables
136  std::vector<std::string> expectedDimList;
137  for (size_t i = 0; i < writeDimConfigs.size(); ++i) {
138  expectedDimList.push_back(writeDimConfigs[i].getString("name"));
139  }
140  std::sort(expectedDimList.begin(), expectedDimList.end());
141  VarNameObjectList dimList = obsIo->dimVarList();
142  std::sort(dimList.begin(), dimList.end(), [](auto & p1, auto & p2) {
143  return (p1.first < p2.first); });
144  for (size_t i = 0; i < dimList.size(); ++i) {
145  EXPECT_EQUAL(dimList[i].first, expectedDimList[i]);
146  }
147 
148  // Regular variables
149  std::vector<std::string> expectedVarList;
150  for (size_t i = 0; i < writeVarConfigs.size(); ++i) {
151  expectedVarList.push_back(writeVarConfigs[i].getString("name"));
152  }
153  std::sort(expectedVarList.begin(), expectedVarList.end());
154  VarNameObjectList varList = obsIo->varList();
155  std::sort(varList.begin(), varList.end(), [](auto & p1, auto & p2) {
156  return (p1.first < p2.first); });
157  for (size_t i = 0; i < dimList.size(); ++i) {
158  EXPECT_EQUAL(varList[i].first, expectedVarList[i]);
159  }
160 
161  // Check if the values of the variables got written correctly
162  for (std::size_t j = 0; j < writeVarConfigs.size(); ++j) {
163  std::string varName = writeVarConfigs[j].getString("name");
164  std::string varType = writeVarConfigs[j].getString("type");
165 
166  // Read the variable data from the file and compare with
167  // the values from the YAML configuration
168  Variable var = obsIo->vars().open(varName);
169  if (varType == "int") {
170  std::vector<int> expectedVarValues =
171  writeVarConfigs[j].getIntVector("values");
172  std::vector<int> varValues;
173  var.read<int>(varValues);
174  EXPECT_EQUAL(varValues, expectedVarValues);
175  } else if (varType == "float") {
176  std::vector<float> expectedVarValues =
177  writeVarConfigs[j].getFloatVector("values");
178  std::vector<float> varValues;
179  var.read<float>(varValues);
180  EXPECT_EQUAL(varValues, expectedVarValues);
181  } else if (varType == "string") {
182  std::vector<std::string> expectedVarValues =
183  writeVarConfigs[j].getStringVector("values");
184  std::vector<std::string> varValues;
185  var.read<std::string>(varValues);
186  EXPECT_EQUAL(varValues, expectedVarValues);
187  }
188  }
189  }
190  }
191 }
192 
193 // -----------------------------------------------------------------------------
194 
195 class ObsIoWrite : public oops::Test {
196  public:
198  virtual ~ObsIoWrite() {}
199  private:
200  std::string testid() const override {return "test::ObsIoWrite";}
201 
202  void register_tests() const override {
203  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
204 
205  ts.emplace_back(CASE("ioda/ObsIoWrite/testWrite")
206  { testWrite(); });
207  }
208 
209  void clear() const override {}
210 };
211 
212 // -----------------------------------------------------------------------------
213 
214 } // namespace test
215 } // namespace ioda
216 
217 #endif // TEST_IO_OBSIOWRITE_H_
Interfaces for ioda::ObsGroup and related classes.
Interfaces for ioda::Variable and related classes.
static std::shared_ptr< ObsIo > create(ObsIoModes mode, const ObsSpaceParameters &parameters)
Create and return a new instance of an ObsIo subclass.
Definition: ObsIoFactory.cc:24
ObsTopLevelParameters top_level_
sub groups of parameters
oops::OptionalParameter< ObsFileOutParameters > obsOutFile
output specification by writing to a file
Variables store data!
Definition: Variable.h:680
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
void register_tests() const override
Definition: ObsIoWrite.h:202
void clear() const override
Definition: ObsIoWrite.h:209
std::string testid() const override
Definition: ObsIoWrite.h:200
CASE("Derived variable, unit conversion, and exception checking methods")
std::vector< std::pair< std::string, Variable > > VarNameObjectList
typedef for holding list of variable names with associated variable object
Definition: IodaUtils.h:30
void setOfileParamsFromTestConfig(const eckit::LocalConfiguration &obsConfig, ioda::ObsSpaceParameters &obsParams)
set params for output file construction from test YAML configuration
Definition: IodaUtils.cc:306
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57