IODA
02-Attributes.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 example shows how to manipulate Attributes and other metadata
9 # that describe Groups and Variables.
10 
11 # Attributes are metadata that help describe a Group or a Variable.
12 # Good examples of attributes include descriptive labels about
13 # the source of your data, a description or long name of a variable,
14 # and it's valid range (which is the interval where the data are valid).
15 #
16 # ioda's attributes are flexible. They can be single points, or they can
17 # be 1-D arrays. They may be integers, or floats, or doubles, strings,
18 # complex numbers, or really any type that you can think of. We will go
19 # through the attribute creation, listing, opening, reading and writing
20 # functions in this example.
21 #
22 # This example creates an HDF5 file, "Example-02.hdf5" using the HDF5 backend.
23 # This file may be viewed with the "h5dump" of "ncdump" commands. You can
24 # also use "hdfview" if you want a gui.
25 #
26 # Note also: This example introduces Attributes. Variables are very similar to
27 # this but have more features. Variables are the focus of the next few tutorials.
28 
29 import os
30 import sys
31 
32 import ioda
33 
34 # All of the attribute information for a Group or a Variable (see later
35 # tutorials) may be accessed by the ".atts" member object.
36 # i.e. g.atts;
37 
38 # ".atts" is an instance of the "Has_Attributes" class.
39 # This tutorial shows you how to *use* this object.
40 # If you want to *understand* how it is implemented, read the next paragraph.
41 # If not, then skip.
42 
43 # You can find Has_Attributes's definition
44 # in "include/ioda/Attributes/Has_Attributes.h". The class definition is not trivial.
45 # Has_Attributes is a derived class that inherits from detail::Has_Attributes_Base,
46 # which is a *template* class. If that's confusing, then just think that everything
47 # in Has_Attributes_Base is also available in Has_Attributes.
48 # We use this really weird pattern in writing the code because we share class
49 # structures between the frontend that you use (Has_Attributes) and the backends
50 # that implement the functionality (which all derive from Has_Attributes_Backend).
51 # When reading any of the ioda-engines header files, try to keep this in mind.
52 
53 # Since we just created a new, empty file, we don't have any attributes yet.
54 # Let's create some.
55 
57  name = "Example-02-python.hdf5",
58  mode = ioda.Engines.BackendCreateModes.Truncate_If_Exists)
59 
60 
61 # The fundamental creation function is .create(name, datatype, dimensions)
62 # What this does is is creates a new attribute, called "int-att-1".
63 # This attribute holds a single ({1}) integer (int32).
64 # The dtype parameter is one of several enumerated types in ioda.Types.
65 # These types match the types in C++. int32 is a 32-bit integer.
66 # Other types include float, double, int16, uint64, et cetera.
67 # The dims parameter is a list representing the dimensions of the attribute.
68 # Attributes can have a single dimension. Variables (discussed later)
69 # can be multidimensional.
70 # The function returns the new attribute. No data is yet written to this attribute.
71 intatt1 = g.atts.create(name="int-att-1", dtype=ioda.Types.int32, dims=[1])
72 # Write a single integer to the attribute using the writeDatum function.
73 # writeDatum writes just one value. If the attribute contains more than one
74 # value, then writeDatum will fail.
75 #
76 # Note the calling convention for the Python interface.
77 # [object].writeDatum.[type](). Python does not have C++'s expressive
78 # template support, so we have to define separate functions that
79 # implement a single C++ template (i.e. write<DataType>(...)).
80 # Additionally, Python types do not align perfectly with C++ types.
81 # So, we need to specify the C++ type, and the interface converts the
82 # values if necessary.
83 # write*.int32 means:
84 # first convert the data to int32_t,
85 # then write the data.
86 intatt1.writeDatum.int32(1)
87 
88 # Create another attribute. This time, it stores two values.
89 intatt2 = g.atts.create("int-att-2", ioda.Types.int32, [2])
90 # You can pass a list of values to write to the writeVector function.
91 # writeVector takes in any one-dimensional Python list and copies its
92 # data into the Attribute.
93 intatt2.writeVector.int32([1, 2])
94 
95 # Same as above.
96 intatt3 = g.atts.create("int-att-3", ioda.Types.int32, [3])
97 intatt3.writeVector.int32([1, 2, 3])
98 
99 # Create and write a C++-style float. A float has less precision than
100 # a double.
101 float1 = g.atts.create("float-1", ioda.Types.float, [2])
102 float1.writeVector.float([3.1159, 2.78])
103 
104 # Same for a double.
105 g.atts.create("double-1", ioda.Types.double, [4]).writeVector.double([1.1,2.2,3.3,4.4])
106 
107 # Write a variable-length, UTF-8 string.
108 # All strings in ioda are assumed to be this type of string.
109 g.atts.create("str-1", ioda.Types.str, [1]).writeVector.str(["This is a test."])
110 
111 # To list all attributes attached to an object (a Group or Variable),
112 # use the atts.list() function. This returns a Python list.
113 attlist = g.atts.list()
114 
115 if len(attlist) != 6:
116  raise Exception("Wrong size for g.atts.list!")
117 
118 # Opening an attribute
119 f1 = g.atts.open('float-1')
120 d1 = g.atts.open('double-1')
121 
122 # The dimensions of an attribute are set at creation time and are
123 # fixed. An attribute cannot be resized.
124 # This dimensions class is shared with variables, so not all members
125 # are relevant to attributes.
126 # The dimensions object has four members:
127 # 1) the overall dimensionality (for an Attribute, this is always one).
128 # 2) the current dimensions (a list)
129 # 3) the maximum dimensions (pertains only to resizable objects; a list)
130 # 4) the total number of elements (a scalar integer)
131 
132 f1_dims = f1.dims
133 
134 if f1_dims.dimensionality != 1:
135  raise Exception("Unexpected dimensionality")
136 
137 if f1_dims.dimsCur[0] != 2:
138  raise Exception("Unexpected size")
139 
140 if f1_dims.dimsMax[0] != 2:
141  raise Exception("Unexpected size")
142 
143 if f1_dims.numElements != 2:
144  raise Exception("Unexpected size")
145 
146 # To open an attribute:
147 a2_reopened = g.atts.open('int-att-2')
148 
149 # To check the storage type of an object:
150 # There are two very similar ways to do this, with isA and isA2.
151 # The differences are in the templating and the parameters.
152 # Prefer isA unless you are testing multiple types dynamically.
153 
154 if float1.isA.float() != True:
155  raise Exception("Unexpected type")
156 
157 if g.atts.open('int-att-1').isA2(ioda.Types.int32) != True:
158  raise Exception("Unexpected type")
159 
160 # Reading attributes:
161 # readDatum and readVector are complimentary calls to
162 # writeDatum and writeVector.
163 
164 # readDatum returns a single element.
165 # readVector returns a list.
166 
167 i2vals = a2_reopened.readVector.int32()
168 if len(i2vals) != 2:
169  raise Exception("Wrong length")
170 
171 # That's everything about attributes!
172 
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