HighFive 3.0.0
HighFive - Header-only C++ HDF5 interface
Loading...
Searching...
No Matches
H5Node_traits_misc.hpp
Go to the documentation of this file.
1/*
2 * Copyright (c), 2017, Adrien Devresse <adrien.devresse@epfl.ch>
3 *
4 * Distributed under the Boost Software License, Version 1.0.
5 * (See accompanying file LICENSE_1_0.txt or copy at
6 * http://www.boost.org/LICENSE_1_0.txt)
7 *
8 */
9#pragma once
10
11#include <string>
12#include <vector>
13
14#include <H5Apublic.h>
15#include <H5Fpublic.h>
16#include <H5Ppublic.h>
17#include <H5Tpublic.h>
18
19#include "../H5DataSet.hpp"
20#include "../H5Group.hpp"
21#include "../H5Selection.hpp"
22#include "../H5Utility.hpp"
23#include "H5DataSet_misc.hpp"
24#include "H5Iterables_misc.hpp"
25#include "H5Selection_misc.hpp"
27
28#include "h5l_wrapper.hpp"
29#include "h5g_wrapper.hpp"
30#include "h5o_wrapper.hpp"
31
32
33namespace HighFive {
34
35
36template <typename Derivate>
37inline DataSet NodeTraits<Derivate>::createDataSet(const std::string& dataset_name,
38 const DataSpace& space,
39 const DataType& dtype,
40 const DataSetCreateProps& createProps,
41 const DataSetAccessProps& accessProps,
42 bool parents) {
43 LinkCreateProps lcpl;
44 lcpl.add(CreateIntermediateGroup(parents));
45 return DataSet(detail::h5d_create2(static_cast<Derivate*>(this)->getId(),
46 dataset_name.c_str(),
47 dtype.getId(),
48 space.getId(),
49 lcpl.getId(),
50 createProps.getId(),
51 accessProps.getId()));
52}
53
54template <typename Derivate>
55template <typename T>
56inline DataSet NodeTraits<Derivate>::createDataSet(const std::string& dataset_name,
57 const DataSpace& space,
58 const DataSetCreateProps& createProps,
59 const DataSetAccessProps& accessProps,
60 bool parents) {
61 return createDataSet(
62 dataset_name, space, create_and_check_datatype<T>(), createProps, accessProps, parents);
63}
64
65template <typename Derivate>
66template <typename T>
67inline DataSet NodeTraits<Derivate>::createDataSet(const std::string& dataset_name,
68 const T& data,
69 const DataSetCreateProps& createProps,
70 const DataSetAccessProps& accessProps,
71 bool parents) {
72 DataSet ds =
73 createDataSet(dataset_name,
75 create_and_check_datatype<typename details::inspector<T>::base_type>(),
76 createProps,
77 accessProps,
78 parents);
79 ds.write(data);
80 return ds;
81}
82
83template <typename Derivate>
84inline DataSet NodeTraits<Derivate>::getDataSet(const std::string& dataset_name,
85 const DataSetAccessProps& accessProps) const {
86 return DataSet(detail::h5d_open2(static_cast<const Derivate*>(this)->getId(),
87 dataset_name.c_str(),
88 accessProps.getId()));
89}
90
91template <typename Derivate>
92inline Group NodeTraits<Derivate>::createGroup(const std::string& group_name, bool parents) {
93 LinkCreateProps lcpl;
95 return detail::make_group(detail::h5g_create2(static_cast<Derivate*>(this)->getId(),
96 group_name.c_str(),
97 lcpl.getId(),
98 H5P_DEFAULT,
99 H5P_DEFAULT));
100}
101
102template <typename Derivate>
103inline Group NodeTraits<Derivate>::createGroup(const std::string& group_name,
104 const GroupCreateProps& createProps,
105 bool parents) {
106 LinkCreateProps lcpl;
107 lcpl.add(CreateIntermediateGroup(parents));
108 return detail::make_group(detail::h5g_create2(static_cast<Derivate*>(this)->getId(),
109 group_name.c_str(),
110 lcpl.getId(),
111 createProps.getId(),
112 H5P_DEFAULT));
113}
114
115template <typename Derivate>
116inline Group NodeTraits<Derivate>::getGroup(const std::string& group_name) const {
117 return detail::make_group(detail::h5g_open2(static_cast<const Derivate*>(this)->getId(),
118 group_name.c_str(),
119 H5P_DEFAULT));
120}
121
122template <typename Derivate>
123inline DataType NodeTraits<Derivate>::getDataType(const std::string& type_name,
124 const DataTypeAccessProps& accessProps) const {
125 return DataType(detail::h5t_open2(static_cast<const Derivate*>(this)->getId(),
126 type_name.c_str(),
127 accessProps.getId()));
129
130template <typename Derivate>
132 hsize_t res;
133 detail::h5g_get_num_objs(static_cast<const Derivate*>(this)->getId(), &res);
134 return static_cast<size_t>(res);
135}
137template <typename Derivate>
138inline std::string NodeTraits<Derivate>::getObjectName(size_t index) const {
139 return details::get_name([&](char* buffer, size_t length) {
140 return detail::h5l_get_name_by_idx(static_cast<const Derivate*>(this)->getId(),
141 ".",
142 H5_INDEX_NAME,
143 H5_ITER_INC,
144 index,
145 buffer,
146 length,
147 H5P_DEFAULT);
148 });
149}
150
151template <typename Derivate>
152inline bool NodeTraits<Derivate>::rename(const std::string& src_path,
153 const std::string& dst_path,
154 bool parents) const {
155 LinkCreateProps lcpl;
156 lcpl.add(CreateIntermediateGroup(parents));
157 herr_t err = detail::h5l_move(static_cast<const Derivate*>(this)->getId(),
158 src_path.c_str(),
159 static_cast<const Derivate*>(this)->getId(),
160 dst_path.c_str(),
161 lcpl.getId(),
162 H5P_DEFAULT);
163
164 return err >= 0;
165}
166
167template <typename Derivate>
168inline std::vector<std::string> NodeTraits<Derivate>::listObjectNames(IndexType idx_type) const {
169 std::vector<std::string> names;
170 details::HighFiveIterateData iterateData(names);
171
172 size_t num_objs = getNumberObjects();
173 names.reserve(num_objs);
174
175 detail::h5l_iterate(static_cast<const Derivate*>(this)->getId(),
176 static_cast<H5_index_t>(idx_type),
177 H5_ITER_INC,
178 NULL,
179 &details::internal_high_five_iterate<H5L_info_t>,
180 static_cast<void*>(&iterateData));
181 return names;
182}
183
184template <typename Derivate>
185inline bool NodeTraits<Derivate>::_exist(const std::string& node_name, bool raise_errors) const {
186 SilenceHDF5 silencer{};
187 const auto val = detail::nothrow::h5l_exists(static_cast<const Derivate*>(this)->getId(),
188 node_name.c_str(),
189 H5P_DEFAULT);
190 if (val < 0) {
191 if (raise_errors) {
192 HDF5ErrMapper::ToException<GroupException>("Invalid link for exist()");
193 } else {
194 return false;
195 }
196 }
197
198 // The root path always exists, but H5Lexists return 0 or 1
199 // depending of the version of HDF5, so always return true for it
200 // We had to call H5Lexists anyway to check that there are no errors
201 return (node_name == "/") ? true : (val > 0);
202}
203
204template <typename Derivate>
205inline bool NodeTraits<Derivate>::exist(const std::string& group_path) const {
206 // When there are slashes, first check everything is fine
207 // so that subsequent errors are only due to missing intermediate groups
208 if (group_path.find('/') != std::string::npos) {
209 _exist("/"); // Shall not throw under normal circumstances
210 // Unless "/" (already checked), verify path exists (not throwing errors)
211 return (group_path == "/") ? true : _exist(group_path, false);
212 }
213 return _exist(group_path);
214}
215
216
217template <typename Derivate>
218inline void NodeTraits<Derivate>::unlink(const std::string& node_name) const {
219 detail::h5l_delete(static_cast<const Derivate*>(this)->getId(), node_name.c_str(), H5P_DEFAULT);
220}
221
222
223// convert internal link types to enum class.
224// This function is internal, so H5L_TYPE_ERROR shall be handled in the calling context
225static inline LinkType _convert_link_type(const H5L_type_t& ltype) noexcept {
226 switch (ltype) {
227 case H5L_TYPE_HARD:
228 return LinkType::Hard;
229 case H5L_TYPE_SOFT:
230 return LinkType::Soft;
231 case H5L_TYPE_EXTERNAL:
232 return LinkType::External;
233 default:
234 // Other link types are possible but are considered strange to HighFive.
235 // see https://support.hdfgroup.org/HDF5/doc/RM/H5L/H5Lregister.htm
236 return LinkType::Other;
237 }
238}
239
240template <typename Derivate>
241inline LinkType NodeTraits<Derivate>::getLinkType(const std::string& node_name) const {
242 H5L_info_t linkinfo;
243 detail::h5l_get_info(static_cast<const Derivate*>(this)->getId(),
244 node_name.c_str(),
245 &linkinfo,
246 H5P_DEFAULT);
247
248 if (linkinfo.type == H5L_TYPE_ERROR) {
249 HDF5ErrMapper::ToException<GroupException>(std::string("Link type of \"") + node_name +
250 "\" is H5L_TYPE_ERROR");
251 }
252 return _convert_link_type(linkinfo.type);
253}
254
255template <typename Derivate>
256inline ObjectType NodeTraits<Derivate>::getObjectType(const std::string& node_name) const {
257 const auto id = detail::h5o_open(static_cast<const Derivate*>(this)->getId(),
258 node_name.c_str(),
259 H5P_DEFAULT);
260 auto object_type = _convert_object_type(detail::h5i_get_type(id));
261 detail::h5o_close(id);
262 return object_type;
263}
264
265
266template <typename Derivate>
267inline void NodeTraits<Derivate>::createSoftLink(const std::string& link_name,
268 const std::string& obj_path,
269 LinkCreateProps linkCreateProps,
270 const LinkAccessProps& linkAccessProps,
271 const bool parents) {
272 if (parents) {
273 linkCreateProps.add(CreateIntermediateGroup{});
274 }
275 detail::h5l_create_soft(obj_path.c_str(),
276 static_cast<const Derivate*>(this)->getId(),
277 link_name.c_str(),
278 linkCreateProps.getId(),
279 linkAccessProps.getId());
280}
281
282
283template <typename Derivate>
284inline void NodeTraits<Derivate>::createExternalLink(const std::string& link_name,
285 const std::string& h5_file,
286 const std::string& obj_path,
287 LinkCreateProps linkCreateProps,
288 const LinkAccessProps& linkAccessProps,
289 const bool parents) {
290 if (parents) {
291 linkCreateProps.add(CreateIntermediateGroup{});
292 }
293 detail::h5l_create_external(h5_file.c_str(),
294 obj_path.c_str(),
295 static_cast<const Derivate*>(this)->getId(),
296 link_name.c_str(),
297 linkCreateProps.getId(),
298 linkAccessProps.getId());
299}
300
301template <typename Derivate>
302template <typename T, typename>
303inline void NodeTraits<Derivate>::createHardLink(const std::string& link_name,
304 const T& target_obj,
305 LinkCreateProps linkCreateProps,
306 const LinkAccessProps& linkAccessProps,
307 const bool parents) {
308 static_assert(!std::is_same<T, Attribute>::value,
309 "hdf5 doesn't support hard links to Attributes");
310 if (parents) {
311 linkCreateProps.add(CreateIntermediateGroup{});
312 }
313 detail::h5l_create_hard(target_obj.getId(),
314 ".",
315 static_cast<const Derivate*>(this)->getId(),
316 link_name.c_str(),
317 linkCreateProps.getId(),
318 linkAccessProps.getId());
319}
320
321
322} // namespace HighFive
Definition H5PropertyList.hpp:601
Class representing a dataset.
Definition H5DataSet.hpp:30
Class representing the space (dimensions) of a DataSet.
Definition H5DataSpace.hpp:39
static DataSpace From(const T &value)
Automatically deduce the DataSpace from a container/value.
Definition H5Dataspace_misc.hpp:128
HDF5 Data Type.
Definition H5DataType.hpp:61
Represents an hdf5 group.
Definition H5Group.hpp:46
NodeTraits: Base class for Group and File.
Definition H5Node_traits.hpp:28
Group getGroup(const std::string &group_name) const
open an existing group with the name group_name
Definition H5Node_traits_misc.hpp:116
std::vector< std::string > listObjectNames(IndexType idx_type=IndexType::NAME) const
list all leaf objects name of the node / group
Definition H5Node_traits_misc.hpp:168
void unlink(const std::string &node_name) const
unlink the given dataset or group
Definition H5Node_traits_misc.hpp:218
void createExternalLink(const std::string &link_name, const std::string &h5_file, const std::string &obj_path, LinkCreateProps linkCreateProps=LinkCreateProps(), const LinkAccessProps &linkAccessProps=LinkAccessProps(), const bool parents=true)
Definition H5Node_traits_misc.hpp:284
DataType getDataType(const std::string &type_name, const DataTypeAccessProps &accessProps=DataTypeAccessProps::Default()) const
open a commited datatype with the name type_name
Definition H5Node_traits_misc.hpp:123
DataSet getDataSet(const std::string &dataset_name, const DataSetAccessProps &accessProps=DataSetAccessProps::Default()) const
get an existing dataset in the current file
Definition H5Node_traits_misc.hpp:84
void createSoftLink(const std::string &linkName, const T &obj)
A shorthand to create softlink to any object which provides getPath The link will be created with def...
Definition H5Node_traits.hpp:173
void createHardLink(const std::string &link_name, const T &target_obj, LinkCreateProps linkCreateProps=LinkCreateProps(), const LinkAccessProps &linkAccessProps=LinkAccessProps(), const bool parents=true)
Creates hardlinks.
Definition H5Node_traits_misc.hpp:303
Group createGroup(const std::string &group_name, bool parents=true)
create a new group, and eventually intermediate groups
Definition H5Node_traits_misc.hpp:92
DataSet createDataSet(const std::string &dataset_name, const DataSpace &space, const DataType &type, const DataSetCreateProps &createProps=DataSetCreateProps::Default(), const DataSetAccessProps &accessProps=DataSetAccessProps::Default(), bool parents=true)
createDataSet Create a new dataset in the current file of datatype type and of size space
Definition H5Node_traits_misc.hpp:37
bool rename(const std::string &src_path, const std::string &dest_path, bool parents=true) const
moves an object and its content within an HDF5 file.
Definition H5Node_traits_misc.hpp:152
bool exist(const std::string &node_name) const
check a dataset or group exists in the current node / group
Definition H5Node_traits_misc.hpp:205
ObjectType getObjectType(const std::string &node_name) const
A shorthand to get the kind of object pointed to (group, dataset, type...)
Definition H5Node_traits_misc.hpp:256
size_t getNumberObjects() const
return the number of leaf objects of the node / group
Definition H5Node_traits_misc.hpp:131
std::string getObjectName(size_t index) const
return the name of the object with the given index
Definition H5Node_traits_misc.hpp:138
LinkType getLinkType(const std::string &node_name) const
Returns the kind of link of the given name (soft, hard...)
Definition H5Node_traits_misc.hpp:241
hid_t getId() const noexcept
getId
Definition H5Object_misc.hpp:63
HDF5 property Lists.
Definition H5PropertyList.hpp:158
void add(const P &property)
Definition H5PropertyList_misc.hpp:68
Utility class to disable HDF5 stack printing inside a scope.
Definition H5Utility.hpp:24
void write(const T &buffer, const DataTransferProps &xfer_props=DataTransferProps())
Definition H5Slice_traits_misc.hpp:404
Definition assert_compatible_spaces.hpp:15
DataType create_and_check_datatype()
Create a DataType instance representing type T and perform a sanity check on its size.
Definition H5DataType_misc.hpp:434
LinkType
The possible types of group entries (link concept)
Definition H5Node_traits.hpp:226
ObjectType
Enum of the types of objects (H5O api)
Definition H5Object.hpp:24
IndexType
Definition H5Node_traits.hpp:19