IODA
01-GroupsAndObsSpaces.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020-2021 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 /*! \defgroup ioda_cxx_ex Examples
8  * \brief C++ Usage Examples
9  * \ingroup ioda_cxx
10  *
11  * @{
12  * \dir Basic
13  * \brief Basic C++ ioda-engines usage tutorials
14  *
15  * \defgroup ioda_cxx_ex_1 Ex 1: Groups and ObsSpaces
16  * \brief Group manipulation using the C++ interface
17  * \see 01-GroupsAndObsSpaces.cpp for the example.
18  *
19  * @{
20  *
21  * \file 01-GroupsAndObsSpaces.cpp
22  * \brief First example showing how to make a group (or an ObsSpace).
23  *
24  * IODA is the Interface for Observation Data Access.
25  *
26  * The objective of the IODA is to provide uniform access to
27  * observation data across the whole forecasting chain from
28  * observation pre-processing to data assimilation to diagnostics.
29  *
30  * IODA isolates the scientific code from the underlying data
31  * structures holding the data. Of course, any user needs to have
32  * a basic understanding of how the data are laid out. How are the
33  * data grouped, how can you access the data, how can you
34  * interpret variable names, and how can you read dimensions
35  * and meta data?
36  *
37  * Data in IODA are stored in a structure Groups, Variables, and
38  * Attributes. A Group is like a folder. It is a logical
39  * collection of Variables and Attributes that describes some
40  * portion of the overall data. A Variable stores bulk data.
41  * An Attribute stores smaller quantities of metadata, and can
42  * be attached to either a Group or a Variable.
43  *
44  * In ioda-engines, we separate out how the data are stored
45  * (the backend engine) from how an end user can access it (in
46  * the frontend). Different backends all provide the same general
47  * interface, but may support slightly different features and will
48  * have different performance characteristics.
49  *
50  * This example shows you how to create Groups. It creates an
51  * HDF5 file, "Example-01.hdf5" using the HDF5 backend. Later examples
52  * will use groups to store and read data.
53  *
54  * \author Ryan Honeyager (honeyage@ucar.edu)
55  **/
56 
57 #include <iostream> // We want I/O.
58 #include <string> // We want strings
59 #include <vector> // We want vectors
60 
61 #include "ioda/Engines/Factory.h" // Used to kickstart the Group engine.
62 #include "ioda/Exception.h" // Exceptions and debugging
63 #include "ioda/Group.h" // We are manipulating ioda::Groups.
64 
65 int main(int argc, char** argv) {
66  using namespace ioda; // All of the ioda functions are here.
67  using std::cerr;
68  using std::endl;
69  using std::string;
70  using std::vector;
71  try {
72  // We want to open a new file, backed by the default engine (HDF5).
73  // We open this file as a root-level Group.
74  Group grpFromFile = Engines::constructFromCmdLine(argc, argv, "Example-01.hdf5");
75  // Note: After you build and run this example, you can view the contents of this
76  // HDF5 file with either the "h5dump" or "ncdump" commands.
77  // Since we are only using a subset of HDF5's full feature set, netCDF conveniently
78  // recognizes this as a valid netCDF-4 file!
79 
80  // The only time that you need to be concerned about the
81  // backend (HDF5) is when you create or open a root-level Group.
82  // All Variables and Attributes within a Group transparently
83  // use the same backend.
84 
85  // Groups can contain other Groups!
86  // To create a new group, use the .create() method.
87  // The new group is a child group of the object that is used to
88  // create it. It shares the same underlying backend as its parent.
89  Group g1 = grpFromFile.create("g1");
90  Group g2 = grpFromFile.create("g2");
91 
92  // Groups can form a tree structure.
93  Group g3 = g1.create("g3");
94  Group g4 = g3.create("g4");
95  Group g5 = g4.create("g5");
96  Group g6 = g4.create("g6");
97  Group g8 = g6.create("g7/g8");
98 
99  // Your tree looks like this:
100  //
101  // / - g1 - g3 - g4 - g5
102  // | |
103  // g2 g6 - g7 - g8
104  //
105 
106  // Besides creating Groups, we can also check if a particular
107  // group exists, list them and open them.
108 
109  // Checking existance
110  bool g3exists = g1.exists("g3");
111  if (!g3exists) throw; // jedi_throw.add("Reason", "g3 does not exist.");
112 
113  // Nesting
114  // We can use '/' as a path separator.
115  bool g4exists = g1.exists("g3/g4");
116  if (!g4exists) throw; // jedi_throw.add("Reason", "g4 does not exist.");
117 
118  // Listing the groups contained within a group.
119  // The .list() function returns a vector of strings
120  // listing all immediate (one-level) child groups.
121  vector<string> g3children = g3.list(); // Should be { "g4" }.
122  if (g3children.size() != 1)
123  throw; /* jedi_throw
124  .add("Reason", "g3 contents are unexpected.")
125  .add("size", g3children.size()); */
126  vector<string> g4children = g4.list(); // Should be { "g5", "g6" }.
127  if (g4children.size() != 2)
128  throw; /* jedi_throw
129 .add("Reason", "g4 contents are unexpected.")
130 .add("size", g4children.size()); */
131 
132  // When an error is encountered, you'll notice that
133  // we did a "throw jedi_throw" statement. This throws an error
134  // exception that is implemented in the jedi code.
135  // "jedi_throw" is a macro that tags the current source file,
136  // current function, and the line # in the current file, and
137  // creates an "xThrow" object. You can then use the
138  // .add(key, value) function to add in diagnostic information.
139  // You can add any information that you could write using an
140  // iostream like cout.
141  // So, you can pass values like strings and numbers without
142  // difficulty. Complex objects like vectors or sets would
143  // need more work.
144 
145  // Opening groups
146  // This is also really easy. Use the .open function.
147  // It also obeys nesting criteria, and throws on an error.
148  Group opened_g3 = g1.open("g3");
149  Group opened_g6 = opened_g3.open("g4/g6");
150 
151  // Groups g3 and opened_g3 point to the same object.
152  // Groups g6 and opened_g6 also point to the same object.
153  // Any changes that you make in one of these groups will
154  // be instantly visible to the other.
155 
156  // Note: we make no guarantees about concurrent access using
157  // threads. That is a detail left up to the backend, and is
158  // an area of future work.
159 
160  // What about closing groups?
161  // These Group objects can go out of scope, and they release
162  // their resource locks when they destruct. So, there is no
163  // specific close method.
164  // If you _really_ want to close an object, just reassign it.
165  opened_g3 = Group{}; // opened_g3 now points elsewhere.
166 
167  // If all references to a specific backend instance are closed,
168  // then it is released and does its cleanup tasks.
169 
170  // What about Attributes and Variables?
171  // See tutorial 2 for Attributes.
172  // Variables are covered in tutorials 3-7.
173 
174  // Thanks for reading!
175  } catch (const std::exception& e) {
177  return 1;
178  }
179  return 0;
180 }
181 
182 /// @}
183 /// @}
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
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
int main(int argc, char **argv)
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