IODA
HH-hasattributes.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2017-2020 Ryan Honeyager (ryan@honeyager.info)
3  * (C) Copyright 2020-2021 UCAR
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  */
8 /*! \addtogroup ioda_internals_engines_hh
9  *
10  * @{
11  * \file HH-hasattributes.cpp
12  * \brief HDF5 engine implementation of Has_Attributes.
13  */
14 
15 #include "./HH/HH-hasattributes.h"
16 
17 #include "./HH/HH-attributes.h"
18 #include "./HH/HH-types.h"
19 #include "./HH/HH-util.h"
20 #include "ioda/Exception.h"
21 #include "ioda/Misc/Dimensions.h"
22 
23 namespace ioda {
24 namespace detail {
25 namespace Engines {
26 namespace HH {
27 HH_HasAttributes::HH_HasAttributes() : base_(Handles::HH_hid_t::dummy()) {}
32 }
33 
34 std::vector<std::string> HH_HasAttributes::list() const {
35  std::vector<std::string> res;
36 
37 #if H5_VERSION_GE(1, 12, 0)
38  H5O_info1_t info;
39  herr_t err = H5Oget_info1(base_(), &info); // H5P_DEFAULT only, per docs.
40 #else
41  H5O_info_t info;
42  herr_t err = H5Oget_info(base_(), &info); // H5P_DEFAULT only, per docs.
43 #endif
44 
45  if (err < 0) throw; // HH_throw;
46  res.resize(gsl::narrow<size_t>(info.num_attrs));
47  for (size_t i = 0; i < res.size(); ++i) {
48  HH_Attribute a(HH_hid_t(H5Aopen_by_idx(base_(), ".", H5_INDEX_NAME, H5_ITER_NATIVE, (hsize_t)i,
49  H5P_DEFAULT, H5P_DEFAULT),
51  res[i] = a.getName();
52  }
53 
54  return res;
55 }
56 
57 bool HH_HasAttributes::exists(const std::string& attname) const {
58 #if H5_VERSION_GE(1, 12, 0)
59  H5O_info1_t info;
60  herr_t err = H5Oget_info1(base_(), &info); // H5P_DEFAULT only, per docs.
61 #else
62  H5O_info_t info;
63  herr_t err = H5Oget_info(base_(), &info); // H5P_DEFAULT only, per docs.
64 #endif
65  if (err < 0) throw Exception("H5Oget_info failed.", ioda_Here());
66  if (info.num_attrs < thresholdLinear) {
67  auto ret = iterativeAttributeSearch(base_(), attname.c_str(), getAttrCreationOrder(base_(), info.type));
68  bool success = ret.first;
69  return success;
70  } else {
71  auto ret = H5Aexists(base_(), attname.c_str());
72  if (ret < 0) throw Exception("H5Aexists failed.", ioda_Here());
73  return (ret > 0);
74  }
75 }
76 
77 void HH_HasAttributes::remove(const std::string& attname) {
78  herr_t err = H5Adelete(base_(), attname.c_str());
79  if (err < 0) throw Exception("H5Adelete failed.", ioda_Here());
80 }
81 
82 Attribute HH_HasAttributes::open(const std::string& name) const {
83 #if H5_VERSION_GE(1, 12, 0)
84  H5O_info1_t info;
85  herr_t err = H5Oget_info1(base_(), &info); // H5P_DEFAULT only, per docs.
86 #else
87  H5O_info_t info;
88  herr_t err = H5Oget_info(base_(), &info); // H5P_DEFAULT only, per docs.
89 #endif
90  if (err < 0) throw Exception("H5Oget_info failed.", ioda_Here());
91  if (info.num_attrs < thresholdLinear) {
92  HH_Attribute a = iterativeAttributeSearchAndOpen(base_(), info.type, name.c_str());
93  if (!a.get().isValid()) throw Exception("iterativeAttributeSearchAndOpen failed.",
94  ioda_Here());
95  auto b = std::make_shared<HH_Attribute>(a);
96  Attribute att{b};
97  return att;
98  } else {
99  auto ret = H5Aopen(base_(), name.c_str(), H5P_DEFAULT);
100  if (ret < 0) throw Exception("H5Aopen failed", ioda_Here());
101  auto b = std::make_shared<HH_Attribute>(ret);
102  Attribute att{b};
103  return att;
104  }
105 }
106 
107 Attribute HH_HasAttributes::create(const std::string& attrname, const Type& in_memory_dataType,
108  const std::vector<Dimensions_t>& dimensions) {
109  try {
110  auto typeBackend = std::dynamic_pointer_cast<HH_Type>(in_memory_dataType.getBackend());
111  std::vector<hsize_t> hDims;
112  hDims.reserve(dimensions.size());
113  for (const auto& d : dimensions) hDims.push_back(gsl::narrow<hsize_t>(d));
114 
115  hid_t space = (dimensions.empty())
116  ? H5Screate(H5S_SCALAR)
117  : H5Screate_simple(gsl::narrow<int>(hDims.size()), hDims.data(), nullptr);
119  auto attI = HH_hid_t(H5Acreate(base_(), attrname.c_str(), typeBackend->handle(), dspace(),
120  H5P_DEFAULT, H5P_DEFAULT),
122  if (H5Iis_valid(attI()) <= 0) throw Exception("H5Acreate failed.", ioda_Here());
123 
124  auto b = std::make_shared<HH_Attribute>(attI);
125  Attribute att{b};
126  return att;
127  } catch (std::bad_cast) {
128  std::throw_with_nested(Exception("typeBackend is the wrong type. Expected HH_Type.",
129  ioda_Here()));
130  }
131 }
132 void HH_HasAttributes::rename(const std::string& oldName, const std::string& newName) {
133  auto ret = H5Arename(base_(), oldName.c_str(), newName.c_str());
134  if (ret < 0) throw Exception("H5Arename failed.", ioda_Here());
135 }
136 
137 } // namespace HH
138 } // namespace Engines
139 } // namespace detail
140 } // namespace ioda
141 
142 /// @}
Describe the dimensions of a ioda::Attribute or ioda::Variable.
IODA's error system.
HDF5 engine implementation of Attribute.
HDF5 engine implementation of Has_Attributes.
HDF5 engine implementation of ioda::detail::Type_Provider.
Utility functions for HDF5.
This class represents attributes, which may be attached to both Variables and Groups.
Definition: Attribute.h:493
The ioda exception class.
Definition: Exception.h:54
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
This is the implementation of Attributes using HDF5.
Definition: HH-attributes.h:30
bool exists(const std::string &attname) const final
Check if an attribute exists.
std::vector< std::string > list() const final
Attribute create(const std::string &attrname, const Type &in_memory_dataType, const std::vector< Dimensions_t > &dimensions={1}) final
Create an Attribute without setting its data.
detail::Type_Provider * getTypeProvider() const final
Query the backend and get the type provider.
void rename(const std::string &oldName, const std::string &newName) final
Rename an Attribute.
void remove(const std::string &attname) final
Delete an Attribute with the specified name.
Attribute open(const std::string &name) const final
Open an attribute.
static HH_Type_Provider * instance()
Definition: HH-types.cpp:36
A class to wrap HDF5's hid_t resource handles.
Definition: Handles.h:92
std::shared_ptr< Type_Backend > getBackend() const
Definition: Type.h:103
Backends implement type providers in conjunction with Attributes, Has_Attributes, Variables and Has_V...
Definition: Type_Provider.h:36
IODA_HIDDEN std::pair< bool, hsize_t > iterativeAttributeSearch(hid_t baseObject, const char *attname, H5_index_t iteration_type)
Linear search to find an attribute.
Definition: HH-util.cpp:82
IODA_HIDDEN HH_Attribute iterativeAttributeSearchAndOpen(hid_t baseObject, H5O_type_t objType, const char *attname)
Linear search to find and open an attribute, if it exists.
Definition: HH-util.cpp:100
IODA_HIDDEN H5_index_t getAttrCreationOrder(hid_t obj, H5O_type_t objType)
Determine attribute creation order for a dataset.
Definition: HH-util.cpp:55
#define ioda_Here()