IODA
test_stitchcomplementaryvariables.cpp
Go to the documentation of this file.
1 /*
2  * (C) Crown Copyright 2021 Met Office
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 
9 
10 #include "ioda/Engines/Factory.h"
11 #include "ioda/Exception.h"
12 #include "ioda/Layout.h"
13 #include "ioda/ObsGroup.h"
14 #include "ioda/testconfig.h"
15 
16 #include "eckit/testing/Test.h"
17 
18 using namespace eckit::testing;
19 using namespace ioda;
20 
21 namespace ioda {
22 namespace test {
23 
24 const int locations = 40;
25 const int channels = 30;
26 CASE("Stitch variables, remove originals defaulted as true") {
27  typedef std::string str;
28  str mappingFile = str(IODA_ENGINES_TEST_SOURCE_DIR)
29  + "/variables/hasvariables_stitching_map.yaml";
30  Engines::BackendNames backendName = Engines::BackendNames::Hdf5File;
32  backendParams.fileName = "ioda-engines_hasvariables_stitch-file.hdf5";
33  backendParams.action = Engines::BackendFileActions::Create;
34  backendParams.createMode = Engines::BackendCreateModes::Truncate_If_Exists;
35  Group backend = ioda::Engines::constructBackend(backendName, backendParams);
36 
37  ioda::Variable completePart1 = backend.vars.create<str>("completeCombinationPart1", {3});
38  completePart1.write<str>({str("a"), str("A"), str("1")});
39  ioda::Variable completePart2 = backend.vars.create<str>("completeCombinationPart2", {3});
40  completePart2.write<str>({str("b"), str("B"), str("2")});
41  ioda::Variable completePart3 = backend.vars.create<str>("completeCombinationPart3", {3});
42  completePart3.write<str>({str("c"), str("C"), str("3")});
43  ioda::Variable incompletePart1 = backend.vars.create<str>("incompleteCombinationPart1", {3});
44  incompletePart1.write<str>({str("a"), str("A"), str("1")});
45  ioda::Variable incompletePart2 = backend.vars.create<str>("incompleteCombinationPart2", {3});
46  incompletePart2.write<str>({str("b"), str("B"), str("2")});
47  ioda::Variable oneVar = backend.vars.create<str>("oneVariableCombination", {5});
48  oneVar.write<str>({str("foo"), str("bar"), str("baz"), str("lorem"), str("ipsum")});
49 
51  newDims.push_back(NewDimensionScale<int>("nlocs", locations, ioda::Unlimited, locations));
52  newDims.push_back(NewDimensionScale<int>("nchans", channels, channels, channels));
53  ObsGroup og = ObsGroup::generate(
54  backend, newDims,
55  detail::DataLayoutPolicy::generate(detail::DataLayoutPolicy::Policies::ObsGroupODB,
56  mappingFile, {"nlocs", "nchans"}));
57 
58  EXPECT(og.vars.exists(str("completeCombinationPart1")));
59  EXPECT(og.vars.exists(str("completeCombinationPart2")));
60  EXPECT(og.vars.exists(str("completeCombinationPart3")));
61  EXPECT(og.vars.exists(str("incompleteCombinationPart1")));
62  EXPECT(og.vars.exists(str("incompleteCombinationPart2")));
63  EXPECT(og.vars.exists(str("oneVariableCombination")));
64  std::vector<str> singleVarCombPreStitch;
65  og.vars.open("oneVariableCombination").read<str>(singleVarCombPreStitch);
66  og.vars.stitchComplementaryVariables();
67 
68  EXPECT(og.vars.exists(str("completeCombination")));
69  std::vector<str> combinedVariable;
70  (og.vars.open("completeCombination")).read<str>(combinedVariable);
71  std::vector<str> expectedCombinedVariable{"abc", "ABC", "123"};
72  EXPECT(combinedVariable == expectedCombinedVariable);
73  EXPECT_NOT(og.vars.exists(str("completeCombinationPart1")));
74  EXPECT_NOT(og.vars.exists(str("completeCombinationPart2")));
75  EXPECT_NOT(og.vars.exists(str("completeCombinationPart3")));
76 
77  EXPECT_NOT(og.vars.exists(str("incompleteCombination")));
78  EXPECT(og.vars.exists(str("incompleteCombinationPart1")));
79  EXPECT(og.vars.exists(str("incompleteCombinationPart2")));
80 
81  EXPECT_NOT(og.vars.exists(str("oneVariableCombination")));
82  EXPECT(og.vars.exists("oneVariableDerivedVariable"));
83  std::vector<str> singleVarCombPostStitch;
84  (og.vars.open("oneVariableDerivedVariable")).read<str>(singleVarCombPostStitch);
85  EXPECT(singleVarCombPreStitch == singleVarCombPostStitch);
86 }
87 
88 CASE("Stitch variables, remove originals set to false") {
89  typedef std::string str;
90  str mappingFile = str(IODA_ENGINES_TEST_SOURCE_DIR)
91  + "/variables/hasvariables_stitching_map.yaml";
92  Engines::BackendNames backendName = Engines::BackendNames::Hdf5File;
94  backendParams.fileName = "ioda-engines_hasvariables_stitch-file-originals-kept.hdf5";
95  backendParams.action = Engines::BackendFileActions::Create;
96  backendParams.createMode = Engines::BackendCreateModes::Truncate_If_Exists;
97  Group backend = ioda::Engines::constructBackend(backendName, backendParams);
98 
99  ioda::Variable completePart1 = backend.vars.create<str>("completeCombinationPart1", {3});
100  completePart1.write<str>({str("a"), str("A"), str("1")});
101  ioda::Variable completePart2 = backend.vars.create<str>("completeCombinationPart2", {3});
102  completePart2.write<str>({str("b"), str("B"), str("2")});
103  ioda::Variable completePart3 = backend.vars.create<str>("completeCombinationPart3", {3});
104  completePart3.write<str>({str("c"), str("C"), str("3")});
105  ioda::Variable incompletePart1 = backend.vars.create<str>("incompleteCombinationPart1", {3});
106  incompletePart1.write<str>({str("a"), str("A"), str("1")});
107  ioda::Variable incompletePart2 = backend.vars.create<str>("incompleteCombinationPart2", {3});
108  incompletePart2.write<str>({str("b"), str("B"), str("2")});
109  ioda::Variable oneVar = backend.vars.create<str>("oneVariableCombination", {5});
110  oneVar.write<str>({str("foo"), str("bar"), str("baz"), str("lorem"), str("ipsum")});
111 
113  newDims.push_back(NewDimensionScale<int>("nlocs", locations, ioda::Unlimited, locations));
114  newDims.push_back(NewDimensionScale<int>("nchans", channels, channels, channels));
115  ObsGroup og = ObsGroup::generate(
116  backend, newDims,
117  detail::DataLayoutPolicy::generate(detail::DataLayoutPolicy::Policies::ObsGroupODB,
118  mappingFile, {"nlocs", "nchans"}));
119  EXPECT(og.vars.exists(str("completeCombinationPart1")));
120  EXPECT(og.vars.exists(str("completeCombinationPart2")));
121  EXPECT(og.vars.exists(str("completeCombinationPart3")));
122  std::vector<str> completeCombinationPart2PreStitch;
123  (og.vars.open("completeCombinationPart2")).read<str>(completeCombinationPart2PreStitch);
124 
125  og.vars.stitchComplementaryVariables(false);
126 
127  EXPECT(og.vars.exists(str("completeCombination")));
128  std::vector<str> combinedVariable =
129  (og.vars.open("completeCombination")).readAsVector<str>();
130  std::vector<str> expectedCombinedVariable{"abc", "ABC", "123"};
131  EXPECT(combinedVariable == expectedCombinedVariable);
132  EXPECT(og.vars.exists(str("completeCombinationPart1")));
133  EXPECT(og.vars.exists(str("completeCombinationPart2")));
134  EXPECT(og.vars.exists(str("completeCombinationPart3")));
135  std::vector<str> completeCombinationPart2PostStitch;
136  (og.vars.open("completeCombinationPart2")).read<str>(completeCombinationPart2PostStitch);
137  EXPECT(completeCombinationPart2PreStitch == completeCombinationPart2PostStitch);
138 }
139 
140 } // namespace test
141 } // namespace ioda
142 
143 int main(int argc, char** argv) {
144  return run_tests(argc, argv);
145 }
IODA's error system.
Definitions for setting up backends with file and memory I/O.
Interfaces for ioda::Has_Variables and related classes.
Contains definitions for how data are arranged in ioda internally.
Interfaces for ioda::ObsGroup and related classes.
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
An ObsGroup is a specialization of a ioda::Group. It provides convenience functions and guarantees th...
Definition: ObsGroup.h:32
Variables store data!
Definition: Variable.h:680
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.
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
BackendNames
Backend names.
Definition: Factory.h:28
IODA_DL Group constructBackend(BackendNames name, BackendCreationParameters &params)
This is a simple factory style function that will instantiate a different backend based on a given na...
Definition: Factory.cpp:124
list newDims
Definition: 05-ObsGroup.py:95
constexpr int Unlimited
Specifies that a dimension is resizable to infinity.
std::vector< std::shared_ptr< NewDimensionScale_Base > > NewDimensionScales_t
Used to specify backend creation-time properties.
Definition: Factory.h:59
CASE("Validation")
int main(int argc, char **argv)