IODA
HH-groups.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 /*! \addtogroup ioda_internals_engines_hh
8  *
9  * @{
10  * \file HH-groups.cpp
11  * \brief HDF5 engine implementation of Group.
12  */
13 #include "./HH/HH-groups.h"
14 
15 #include <algorithm>
16 #include <iterator>
17 #include <list>
18 #include <set>
19 #include <string>
20 #include <utility>
21 
22 #include "./HH/HH-hasattributes.h"
23 #include "./HH/HH-hasvariables.h"
24 #include "./HH/Handles.h"
25 #include "ioda/Exception.h"
26 #include "ioda/Misc/Dimensions.h"
27 #include "ioda/Misc/StringFuncs.h"
28 
29 namespace ioda {
30 namespace detail {
31 namespace Engines {
32 namespace HH {
34  : backend_(grp), fileroot_(fileroot), caps_(caps) {
35  atts = Has_Attributes(std::make_shared<HH_HasAttributes>(grp));
36  vars = Has_Variables(std::make_shared<HH_HasVariables>(grp, fileroot));
37 }
38 
39 Group HH_Group::create(const std::string& name) {
40  // Group creation property list
41  HH_hid_t groupCreationProps(H5Pcreate(H5P_GROUP_CREATE),
43  if (!groupCreationProps.isValid()) throw Exception("H5Pcreate failed.", ioda_Here());
44  if (0 > H5Pset_link_creation_order(groupCreationProps(),
45  H5P_CRT_ORDER_TRACKED | H5P_CRT_ORDER_INDEXED))
46  throw Exception("H5Pset_link_creation_order failed.", ioda_Here());
47 
48  // Link creation property list
49  HH_hid_t linkCreationProps(H5Pcreate(H5P_LINK_CREATE),
51  if (!linkCreationProps.isValid()) throw Exception("H5Pcreate failed.", ioda_Here());
52  if (0 > H5Pset_create_intermediate_group(linkCreationProps(), 1))
53  throw Exception("H5Pset_create_intermediate_group failed.", ioda_Here());
54 
55  // Finally, create the group
56  hid_t res = H5Gcreate(backend_(), // hid
57  name.c_str(), // name in ascii character encoding
58  linkCreationProps(), groupCreationProps(),
59  H5P_DEFAULT // Group access property list
60  );
61  if (res < 0) throw Exception("H5Gcreate failed.", ioda_Here());
63 
64  auto backend = std::make_shared<HH_Group>(hnd, caps_, fileroot_);
65  return ::ioda::Group{backend};
66 }
67 
68 Group HH_Group::open(const std::string& name) const {
69  hid_t g = H5Gopen(backend_(), name.c_str(), H5P_DEFAULT);
70  if (g < 0) throw Exception("H5Gopen failed.", ioda_Here());
72 
73  auto res = std::make_shared<HH_Group>(grp_handle, caps_, fileroot_);
74  return ::ioda::Group{res};
75 }
76 
77 bool HH_Group::exists(const std::string& name) const {
78  auto paths = splitPaths(name);
79  for (size_t i = 0; i < paths.size(); ++i) {
80  auto p = condensePaths(paths, 0, i + 1);
81  htri_t linkExists = H5Lexists(backend_(), p.c_str(), H5P_DEFAULT);
82  if (linkExists < 0) throw Exception("H5Lexists failed.", ioda_Here());
83  if (linkExists == 0) return false;
84  }
85 
86  // Check that the object is a group
87 #if H5_VERSION_GE(1, 12, 0)
88  H5O_info1_t obj_info;
89  herr_t err = H5Oget_info_by_name1(backend_(), name.c_str(), &obj_info, H5P_DEFAULT);
90 #else
91  H5O_info_t obj_info;
92  herr_t err = H5Oget_info_by_name(backend_(), name.c_str(), &obj_info, H5P_DEFAULT);
93 #endif
94  if (err < 0) throw Exception("H5Oget_info_by_name failed.", ioda_Here());
95  return (obj_info.type == H5O_TYPE_GROUP);
96 }
97 
99 
100 /// Data to pass to/from iterator classes.
102  std::map<ObjectType, std::list<std::string>> lists; // NOLINT: Visibility
104  lists[ObjectType::Group] = std::list<std::string>();
105  lists[ObjectType::Variable] = std::list<std::string>();
106  lists[ObjectType::Unimplemented] = std::list<std::string>();
107  }
108 };
109 
110 /// Callback function for H5Lvisit / H5Literate.
111 #if H5_VERSION_GE(1, 12, 0)
112 herr_t iterate_find_by_link(hid_t g_id, const char* name, const H5L_info2_t* info, void* op_data)
113 #else
114 herr_t iterate_find_by_link(hid_t g_id, const char* name, const H5L_info_t* info, void* op_data)
115 #endif
116 {
117  Iterator_data_t* op = (Iterator_data_t*)op_data; // NOLINT: HDF5 mandates that op_data be void*.
118 
119  // HARD, SOFT, EXTERNAL are all valid link types in HDF5. We only implement hard
120  // links for now.
121  if (info->type != H5L_TYPE_HARD) {
122  op->lists[ObjectType::Unimplemented].emplace_back(name);
123  return 0;
124  }
125 
126  // Get the object and check the type
127 #if H5_VERSION_GE(1, 12, 0)
128  H5O_info1_t oinfo;
129  herr_t err
130  = H5Oget_info_by_name1(g_id, name, &oinfo, H5P_DEFAULT); // H5P_DEFAULT only, per docs.
131 #else
132  H5O_info_t oinfo;
133  herr_t err = H5Oget_info_by_name(g_id, name, &oinfo, H5P_DEFAULT); // H5P_DEFAULT only, per docs.
134 #endif
135  if (err < 0) return -1;
136 
137  if (oinfo.type == H5O_type_t::H5O_TYPE_GROUP)
138  op->lists[ObjectType::Group].emplace_back(name);
139  else if (oinfo.type == H5O_type_t::H5O_TYPE_DATASET)
140  op->lists[ObjectType::Variable].emplace_back(name);
141  else
142  op->lists[ObjectType::Unimplemented].emplace_back(name);
143 
144  return 0;
145 }
146 
147 std::map<ObjectType, std::vector<std::string>> HH_Group::listObjects(ObjectType filter,
148  bool recurse) const {
149  std::map<ObjectType, std::vector<std::string>> res;
150  Iterator_data_t iter_data;
151 
152  // Check if the group has link creation order stored and/or indexed.
153  // This is not the default, but it can speed up object lists considerably.
154  // We hope that odd cases do not occur where a parent group preserves the index
155  // but its child does not.
156  unsigned crt_order_flags = 0; // Set only for a group.
157  // We don't know yet if backend_hid refers to a file or a group. Check.
158  // This matters because link creation order is a group-specific property.
159  if (H5I_GROUP == H5Iget_type(backend_())) {
160  HH_hid_t createpl(H5Gget_create_plist(backend_()),
162  if (createpl() < 0) throw Exception("H5Gget_create_plist failed.", ioda_Here());
163  if (0 > H5Pget_link_creation_order(createpl(), &crt_order_flags))
164  throw Exception("H5Pget_link_creation_order failed.", ioda_Here());
165  }
166  // We only care if this property is tracked. Indexing is performed on the fly
167  // if it is not available (and incurs a read penalty).
168  // Oddly, the HDF5 documentation suggests that this parameter is unnecessary.
169  // The functions should fall back to indexing by name if the creation order index
170  // is unavailable. But, that's not what we observed.
171  H5_index_t idxclass = (crt_order_flags & H5P_CRT_ORDER_TRACKED) // NOLINT
172  ? H5_INDEX_CRT_ORDER
173  : H5_INDEX_NAME;
174 
175  herr_t search_res
176  = (recurse)
177  ? H5Lvisit(backend_(), idxclass, H5_ITER_NATIVE, iterate_find_by_link,
178  reinterpret_cast<void*>(&iter_data))
179  : H5Literate(backend_(), idxclass, H5_ITER_NATIVE, 0, iterate_find_by_link,
180  reinterpret_cast<void*>(&iter_data)); // NOLINT: 0 is not a nullptr here.
181 
182  if (search_res < 0) throw Exception("H5Lvisit / H5Literate failed.", ioda_Here())
183  .add("recurse", recurse);
184 
185  for (auto& cls : iter_data.lists) {
186  if (filter == ObjectType::Ignored || filter == cls.first)
187  res[cls.first] = std::vector<std::string>(std::make_move_iterator(cls.second.begin()),
188  std::make_move_iterator(cls.second.end()));
189  }
190 
191  return res;
192 }
193 } // namespace HH
194 } // namespace Engines
195 } // namespace detail
196 } // namespace ioda
197 
198 /// @}
Describe the dimensions of a ioda::Attribute or ioda::Variable.
IODA's error system.
HDF5 group interface.
HDF5 engine implementation of Has_Attributes.
HDF5 engine implementation of Has_Variables.
HDF5 resource handles in C++.
The ioda exception class.
Definition: Exception.h:54
Exception & add(const std::string &key, const T value)
Add a key-value pair to the error message.
Definition: Exception.h:75
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
This class exists inside of ioda::Group and provides the interface to manipulating Variables.
::ioda::Engines::Capabilities caps_
Definition: HH-groups.h:41
FillValuePolicy getFillValuePolicy() const final
Fill value policy in HDF5 depends on the current group and the root location.
Definition: HH-groups.cpp:98
Group create(const std::string &name) final
Create a group.
Definition: HH-groups.cpp:39
bool exists(const std::string &name) const final
Definition: HH-groups.cpp:77
Group open(const std::string &name) const final
Open a group.
Definition: HH-groups.cpp:68
std::map< ObjectType, std::vector< std::string > > listObjects(ObjectType filter=ObjectType::Ignored, bool recurse=false) const final
List all objects (groups + variables) within this group.
Definition: HH-groups.cpp:147
HH_Group(HH_hid_t grp, ::ioda::Engines::Capabilities caps, HH_hid_t fileroot)
Group constructor.
Definition: HH-groups.cpp:33
A class to wrap HDF5's hid_t resource handles.
Definition: Handles.h:92
Has_Attributes atts
Use this to access the metadata for the group / ObsSpace.
Definition: Group.h:120
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
virtual FillValuePolicy getFillValuePolicy() const
Get the fill value policy used for Variables within this Group.
IODA_DL std::string condensePaths(const std::vector< std::string > &p, size_t start=0, size_t end=std::string::npos)
The inverse of splitPaths. Concatenate strings, separating with '/'.
Definition: StringFuncs.cpp:41
IODA_DL std::vector< std::string > splitPaths(const std::string &p)
Split a string based on occurances of the '/' character.
Definition: StringFuncs.cpp:14
FillValuePolicy
This option describes the default fill values that will be used if the user does not manually specify...
Definition: FillPolicy.h:28
herr_t iterate_find_by_link(hid_t g_id, const char *name, const H5L_info_t *info, void *op_data)
Callback function for H5Lvisit / H5Literate.
Definition: HH-groups.cpp:114
#define ioda_Here()
Struct defining what an engine can/cannot do.
Definition: Capabilities.h:47
Data to pass to/from iterator classes.
Definition: HH-groups.cpp:101
std::map< ObjectType, std::list< std::string > > lists
Definition: HH-groups.cpp:102