HighFive 3.0.0
HighFive - Header-only C++ HDF5 interface
Loading...
Searching...
No Matches
H5Easy_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 "../H5Easy.hpp"
12
13namespace H5Easy {
14
15namespace detail {
16
17// Generate error-stream and return "Exception" (not yet thrown).
18inline Exception error(const File& file, const std::string& path, const std::string& message) {
19 std::ostringstream ss;
20 ss << message << std::endl
21 << "Path: " << path << std::endl
22 << "Filename: " << file.getName() << std::endl;
23 return Exception(ss.str());
24}
25
26// Generate specific dump error
27inline Exception dump_error(File& file, const std::string& path) {
28 if (file.getObjectType(path) == ObjectType::Dataset) {
29 return error(file,
30 path,
31 "H5Easy: Dataset already exists, dump with H5Easy::DumpMode::Overwrite "
32 "to overwrite (with an array of the same shape).");
33 } else {
34 return error(
35 file,
36 path,
37 "H5Easy: path exists, but does not correspond to a Dataset. Dump not possible.");
38 }
39}
40
41// get a opened DataSet: nd-array
42template <class T>
43inline DataSet initDataset(File& file,
44 const std::string& path,
45 const std::vector<size_t>& shape,
46 const DumpOptions& options) {
47 if (!file.exist(path)) {
48 if (!options.compress() && !options.isChunked()) {
49 return file.createDataSet<T>(path, DataSpace(shape), {}, {}, true);
50 } else {
51 std::vector<hsize_t> chunks(shape.begin(), shape.end());
52 if (options.isChunked()) {
53 chunks = options.getChunkSize();
54 if (chunks.size() != shape.size()) {
55 throw error(file, path, "H5Easy::dump: Incorrect rank ChunkSize");
56 }
57 }
59 props.add(Chunking(chunks));
60 if (options.compress()) {
61 props.add(Shuffle());
62 props.add(Deflate(options.getCompressionLevel()));
63 }
64 return file.createDataSet<T>(path, DataSpace(shape), props, {}, true);
65 }
66 } else if (options.overwrite() && file.getObjectType(path) == ObjectType::Dataset) {
67 DataSet dataset = file.getDataSet(path);
68 if (dataset.getDimensions() != shape) {
69 throw error(file, path, "H5Easy::dump: Inconsistent dimensions");
70 }
71 return dataset;
72 }
73 throw dump_error(file, path);
74}
75
76// get a opened DataSet: scalar
77template <class T>
78inline DataSet initScalarDataset(File& file,
79 const std::string& path,
80 const T& data,
81 const DumpOptions& options) {
82 if (!file.exist(path)) {
83 return file.createDataSet<T>(path, DataSpace::From(data), {}, {}, true);
84 } else if (options.overwrite() && file.getObjectType(path) == ObjectType::Dataset) {
85 DataSet dataset = file.getDataSet(path);
86 if (dataset.getElementCount() != 1) {
87 throw error(file, path, "H5Easy::dump: Existing field not a scalar");
88 }
89 return dataset;
90 }
91 throw dump_error(file, path);
92}
93
94template <class File, class F>
95auto apply_attr_func_impl(File& file, const std::string& path, F f) {
96 auto type = file.getObjectType(path);
97 if (type == ObjectType::Group) {
98 auto group = file.getGroup(path);
99 return f(group);
100 } else if (type == ObjectType::Dataset) {
101 auto dataset = file.getDataSet(path);
102 return f(dataset);
103 } else {
104 throw error(file, path, "path is not the root, a group or a dataset.");
105 }
106}
107
108template <class F>
109auto apply_attr_func(const H5Easy::File& file, const std::string& path, F f) {
110 return apply_attr_func_impl(file, path, f);
111}
112
113template <class F>
114auto apply_attr_func(H5Easy::File& file, const std::string& path, F f) {
115 return apply_attr_func_impl(file, path, f);
116}
117
118// get a opened Attribute: nd-array
119template <class T>
120inline Attribute initAttribute(File& file,
121 const std::string& path,
122 const std::string& key,
123 const std::vector<size_t>& shape,
124 const DumpOptions& options) {
125 auto get_attribute = [&](auto& obj) {
126 if (!obj.hasAttribute(key)) {
127 return obj.template createAttribute<T>(key, DataSpace(shape));
128 } else if (options.overwrite()) {
129 Attribute attribute = obj.getAttribute(key);
130 DataSpace dataspace = attribute.getSpace();
131 if (dataspace.getDimensions() != shape) {
132 throw error(file, path, "H5Easy::dumpAttribute: Inconsistent dimensions");
133 }
134 return attribute;
135 }
136 throw error(file,
137 path,
138 "H5Easy: Attribute exists, overwrite with H5Easy::DumpMode::Overwrite.");
139 };
140
141 if (!file.exist(path)) {
142 throw error(file, path, "H5Easy::dumpAttribute: path does not exist");
143 }
144
145 return apply_attr_func(file, path, get_attribute);
146}
147
148// get a opened Attribute: scalar
149template <class T>
150inline Attribute initScalarAttribute(File& file,
151 const std::string& path,
152 const std::string& key,
153 const T& data,
154 const DumpOptions& options) {
155 auto get_attribute = [&](auto& obj) {
156 if (!obj.hasAttribute(key)) {
157 return obj.template createAttribute<T>(key, DataSpace::From(data));
158 } else if (options.overwrite()) {
159 Attribute attribute = obj.getAttribute(key);
160 DataSpace dataspace = attribute.getSpace();
161 if (dataspace.getElementCount() != 1) {
162 throw error(file, path, "H5Easy::dumpAttribute: Existing field not a scalar");
163 }
164 return attribute;
165 }
166 throw error(file,
167 path,
168 "H5Easy: Attribute exists, overwrite with H5Easy::DumpMode::Overwrite.");
169 };
170
171 if (!file.exist(path)) {
172 throw error(file, path, "H5Easy::dumpAttribute: path does not exist");
173 }
174
175 apply_attr_func(file, path, get_attribute);
176}
177
178} // namespace detail
179} // namespace H5Easy
static DataSpace From(const T &value)
Automatically deduce the DataSpace from a container/value.
Definition H5Dataspace_misc.hpp:128
File class.
Definition H5File.hpp:25
PropertyList< PropertyType::DATASET_CREATE > DataSetCreateProps
Definition H5PropertyList.hpp:198
Read/dump DataSets or Attribute using a minimalistic syntax. To this end, the functions are templated...
Definition H5Easy.hpp:62