IODA
test_convertvariableunits.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 #include "oops/util/FloatCompare.h"
19 
20 using namespace eckit::testing;
21 using namespace ioda;
22 
23 namespace ioda {
24 namespace test {
25 
26 const int locations = 40;
27 const int channels = 30;
28 CASE("Convert variables") {
29  typedef std::string str;
30  str mappingFile = str(IODA_ENGINES_TEST_SOURCE_DIR)
31  + "/variables/hasvariables_unitconversion_map.yaml";
32  Engines::BackendNames backendName = Engines::BackendNames::Hdf5File;
34  backendParams.fileName = "ioda-engines_hasvariables_unitconv-file.hdf5";
35  backendParams.action = Engines::BackendFileActions::Create;
36  backendParams.createMode = Engines::BackendCreateModes::Truncate_If_Exists;
37  Group backend = ioda::Engines::constructBackend(backendName, backendParams);
38 
39  ioda::Variable temp = backend.vars.create<double>("temp", {3});
40  temp.write<double>({0.0, 50.0, 100.0});
41  ioda::Variable windspeed = backend.vars.create<double>("windspeed", {3});
42  windspeed.write<double>({0.0, 50.0, 100.0});
43  ioda::Variable relativeHumidity = backend.vars.create<double>("rh", {3});
44  relativeHumidity.write<double>({0.0, 50.0, 100.0});
45  ioda::Variable pressure = backend.vars.create<double>("press", {3});
46  pressure.write<double>({0.0, 50.0, 100.0});
47  ioda::Variable angle = backend.vars.create<double>("angle", {3});
48  angle.write<double>({0.0, 50.0, 100.0});
49  ioda::Variable cloudCoverage = backend.vars.create<double>("cloudCov", {3});
50  cloudCoverage.write<double>({0.0, 50.0, 100.0});
51  ioda::Variable undefinedUnit = backend.vars.create<double>("bar", {3});
52  undefinedUnit.write<double>({0.0, 50.0, 100.0});
53 
54  ObsGroup og = ObsGroup::generate(
55  backend,
56  {
57  NewDimensionScale<int>("nlocs", locations, ioda::Unlimited, locations),
58  NewDimensionScale<int>("nchans", channels, channels, channels) },
59  detail::DataLayoutPolicy::generate("ObsGroupODB",
60  mappingFile));
61 
62  std::vector<double> expectedValue = {0.0, 50.0, 100.0};
63  std::vector<double> retrievedValue;
64  og.vars.open("temp").read<double>(retrievedValue);
65  EXPECT(retrievedValue == expectedValue);
66  og.vars.convertVariableUnits();
67  temp = og.vars.open("temp");
68  EXPECT(temp.atts.exists("units"));
69  std::string tempUnit;
70  temp.atts.open("units").read<std::string>(tempUnit);
71  EXPECT(tempUnit == "kelvin");
72  og.vars.open("temp").read<double>(retrievedValue);
73  expectedValue = {273.15, 323.15, 373.15};
74  EXPECT(oops::are_all_close_relative(retrievedValue, expectedValue, .05));
75  og.vars.open("windspeed").read<double>(retrievedValue);
76  expectedValue = {0.0, 25.7222, 51.4444};
77  EXPECT(oops::are_all_close_relative(retrievedValue, expectedValue, .05));
78  og.vars.open("rh").read(retrievedValue);
79  expectedValue = {0.0, 0.5, 1.0};
80  EXPECT(oops::are_all_close_relative(retrievedValue, expectedValue, .05));
81  og.vars.open("press").read<double>(retrievedValue);
82  expectedValue = {0.0, 5000.0, 10000.0};
83  EXPECT(oops::are_all_close_relative(retrievedValue, expectedValue, .05));
84  og.vars.open("angle").read<double>(retrievedValue);
85  expectedValue = {0.0, 0.872665, 1.74533};
86  EXPECT(oops::are_all_close_relative(retrievedValue, expectedValue, .05));
87  og.vars.open("cloudCov").read<double>(retrievedValue);
88  expectedValue = {0.0, 6.25, 12.5};
89  EXPECT(oops::are_all_close_relative(retrievedValue, expectedValue, .05));
90  Variable bar = og.vars.open("bar");
91  bar.read<double>(retrievedValue);
92  std::string barUnit;
93  bar.atts.open("units").read<std::string>(barUnit);
94  EXPECT(barUnit == "baz");
95  expectedValue = {0.0, 50.0, 100.0};
96  EXPECT(oops::are_all_close_relative(retrievedValue, expectedValue, .05));
97 }
98 } // namespace test
99 } // namespace ioda
100 
101 int main(int argc, char** argv) {
102  return run_tests(argc, argv);
103 }
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
virtual Attribute_Implementation read(gsl::span< char > data, const Type &in_memory_dataType) const
The fundamental read function. Backends overload this function to implement all read operations.
Definition: Attribute.cpp:75
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
virtual Attribute open(const std::string &name) const
Open an Attribute by name.
virtual bool exists(const std::string &attname) const
Does an Attribute with the specified name exist?
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.
Has_Attributes atts
Attributes.
Definition: Variable.h:71
virtual Variable read(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all) const
Read the Variable - as char array. Ordering is row-major.
Definition: Variable.cpp:330
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
constexpr int Unlimited
Specifies that a dimension is resizable to infinity.
Used to specify backend creation-time properties.
Definition: Factory.h:59
int main(int argc, char **argv)
CASE("Validation")