IODA
01-GroupsAndObsSpaces.py
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 
8 # This first example shows you how to make and manipulate Groups.
9 #
10 # IODA is the Interface for Observation Data Access.
11 #
12 # The objective of the IODA is to provide uniform access to
13 # observation data across the whole forecasting chain from
14 # observation pre-processing to data assimilation to diagnostics.
15 #
16 # IODA isolates the scientific code from the underlying data
17 # structures holding the data. Of course, any user needs to have
18 # a basic understanding of how the data are laid out. How are the
19 # data grouped, how can you access the data, how can you
20 # interpret variable names, and how can you read dimensions
21 # and meta data?
22 #
23 # Data in IODA are stored in a structure Groups, Variables, and
24 # Attributes. A Group is like a folder. It is a logical
25 # collection of Variables and Attributes that describes some
26 # portion of the overall data. A Variable stores bulk data.
27 # An Attribute stores smaller quantities of metadata, and can
28 # be attached to either a Group or a Variable.
29 #
30 # In ioda-engines, we separate out how the data are stored
31 # (the backend engine) from how an end user can access it (in
32 # the frontend). Different backends all provide the same general
33 # interface, but may support slightly different features and will
34 # have different performance characteristics.
35 #
36 # This example shows you how to create Groups. It creates an
37 # HDF5 file, "Example-01-python.hdf5" using the HDF5 backend. Later examples
38 # will use groups to store and read data.
39 #
40 
41 # How to view the output of these examples:
42 #
43 # These examples also double as CTest tests. So, we have a tiny
44 # bit of a setup preamble at the beginning of each example that helps
45 # us to load in the ioda Python library.
46 #
47 # If you want to run the example, try using ctest (i.e.
48 # ctest -V -R Example_ioda_engines_01_python).
49 
50 # Note: After you build and run this example, you can view the contents of this
51 # HDF5 file with either the "h5dump" or "ncdump" commands.
52 # Since we are only using a subset of HDF5's full feature set, netCDF conveniently
53 # recognizes this as a valid netCDF-4 file!
54 #
55 # If you run this example in a Jupyter kernel, you might need
56 # to shutdown the kernel to convince Python to release its resouce
57 # locks and finish writing the output file. Every Group, Attribute,
58 # and Variable returned by ioda is actually a "resource handle". Every
59 # open handle must be closed for the file to finish writing. Or, track
60 # down every ioda Python object and "delete" it
61 # (e.g. "del grpFromFile", "del g1", "del g2", ad nauseam).
62 
63 import os
64 import sys
65 
66 # Importing ioda is easy.
67 
68 import ioda
69 
70 # We want to open a new file, backed by the default engine (HDF5).
71 # We open this file as a root-level Group.
73  name = "Example-01-python.hdf5",
74  mode = ioda.Engines.BackendCreateModes.Truncate_If_Exists)
75 
76 # The only time that you need to be concerned about the
77 # backend (HDF5) is when you create or open a root-level Group.
78 # All Variables and Attributes within a Group transparently
79 # use the same backend.
80 #
81 # Groups can contain other Groups!
82 # To create a new group, use the .create() method.
83 # The new group is a child group of the object that is used to
84 # create it. It shares the same underlying backend as its parent.
85 
86 g1 = grpFromFile.create('g1')
87 g2 = grpFromFile.create("g2")
88 
89 # Groups can form a tree structure.
90 g3 = g1.create("g3")
91 g4 = g3.create("g4")
92 g5 = g4.create("g5")
93 g6 = g4.create("g6")
94 g8 = g6.create("g7/g8")
95 
96 # Your tree looks like this:
97 #
98 # / - g1 - g3 - g4 - g5
99 # | |
100 # g2 g6 - g7 - g8
101 #
102 
103 # Besides creating Groups, we can also check if a particular
104 # group exists, list them and open them.
105 
106 # Checking existance
107 if g1.exists('g3') != True:
108  raise Exception("g3 does not exist in /g1")
109 
110 # Nesting
111 # We can use '/' as a path separator.
112 if g1.exists('g3/g4') != True:
113  raise Exception("g1/g3/g4 does not exist")
114 
115 # Listing the groups contained within a group.
116 # The .list() function returns a list of strings
117 # listing all immediate (one-level) child groups.
118 # This function DOES NOT list variables and attributes. For those,
119 # you would need g3.vars.list() and g3.atts.list(), which are
120 # discussed in subsequent tutorials.
121 g3.list()
122 
123 # These lists are regular Python lists. Nothing special about them.
124 if len(g3.list()) != 1:
125  raise Exception("Wrong size for g3!")
126 
127 if len(g4.list()) != 2:
128  raise Exception("Wrong number of children in g4!")
129 
130 # C++ and Python exceptions can coexist!
131 # C++ exceptions are converted into Python exceptions.
132 
133 # Opening groups
134 # This is also really easy. Use the .open function.
135 # It also obeys nesting criteria, and throws on an error.
136 opened_g3 = g1.open("g3")
137 opened_g6 = opened_g3.open("g4/g6")
138 
139 # Groups g3 and opened_g3 are handles that point to the same object.
140 # Groups g6 and opened_g6 also point to the same object.
141 # Any changes that you make in one of these groups will
142 # be instantly visible to the other.
143 
144 # Note: we make no guarantees about concurrent access using
145 # threads. That is a detail left up to the backend, and is
146 # an area of future work.
147 
148 # What about closing groups?
149 # These Group objects can go out of scope, and they release
150 # their resource locks when they destruct. So, there is no
151 # specific close method.
152 # If you _really_ want to close an object, do this:
153 del opened_g3
154 
155 # If all references to a specific backend instance are closed,
156 # then it is released and does its cleanup tasks.
157 
158 # What about Attributes and Variables?
159 # See tutorial 2 for Attributes.
160 # Variables are covered in tutorials 3-4.
161 
162 # What are ObsGroups?
163 # These extend Groups, and they are covered in tutorial 5.
164 
165 # Thanks for reading!
166 
167 
IODA_DL Group createFile(const std::string &filename, BackendCreateModes mode, HDF5_Version_Range compat=defaultVersionRange())
Create a ioda::Group backed by an HDF5 file.
Definition: HH.cpp:120