IODA
Has_Attributes.h
Go to the documentation of this file.
1 #pragma once
2 /*
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_cxx_attribute
9  *
10  * @{
11  * \file Has_Attributes.h
12  * \brief Interfaces for ioda::Has_Attributes and related classes.
13  */
14 
15 #include <gsl/gsl-lite.hpp>
16 #include <iostream>
17 #include <memory>
18 #include <string>
19 #include <utility>
20 #include <vector>
21 
23 #include "ioda/Exception.h"
24 #include "ioda/Misc/Dimensions.h"
25 #include "ioda/Misc/Eigen_Compat.h"
26 #include "ioda/Types/Type.h"
27 #include "ioda/defs.h"
28 
29 namespace ioda {
30 class Has_Attributes;
31 namespace detail {
32 class Has_Attributes_Backend;
33 class Has_Attributes_Base;
34 class Variable_Backend;
35 class Group_Backend;
36 
37 /// \brief Describes the functions that can add attributes
38 /// \ingroup ioda_cxx_attribute
39 /// \details Using the (Curiously Recurring Template
40 /// Pattern)[https://en.wikipedia.org/wiki/Curiously_recurring_template_pattern] to implement this
41 /// since we want to use the same functions for placeholder attribute construction.
42 /// \see Has_Attributes
43 /// \see Attribute_Creator
44 template <class DerivedHasAtts>
46 public:
47  /// @name Convenience functions for adding attributes
48  /// @{
49 
50  /// \brief Create and write an Attribute, for arbitrary dimensions.
51  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
52  /// \param attrname is the name of the Attribute to be created.
53  /// \param data is a gsl::span (a pointer-length pair) that contains the data to be written.
54  /// \param dimensions is an initializer list representing the size of the metadata. Each element
55  /// is a dimension with a certain size.
56  /// \returns Another instance of this Has_Attribute. Used for operation chaining.
57  /// \throws jedi::xError if data.size() does not match the number of total elements
58  /// described by dimensions.
59  /// \see gsl::span for details of how to make a span.
60  template <class DataType>
61  DerivedHasAtts add(const std::string& attrname, ::gsl::span<const DataType> data,
62  const ::std::vector<Dimensions_t>& dimensions) {
63  auto derivedThis = static_cast<DerivedHasAtts*>(this);
64  auto att = derivedThis->template create<DataType>(attrname, dimensions);
65  att.template write<DataType>(data);
66  return *derivedThis;
67  }
68 
69  /// \brief Create and write an Attribute, for arbitrary dimensions.
70  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
71  /// \param attrname is the name of the Attribute to be created.
72  /// \param data is an initializer list that contains the data to be written.
73  /// \param dimensions is an initializer list representing the size of the metadata. Each
74  /// element is a dimension with a certain size.
75  /// \returns Another instance of this Has_Attribute. Used for operation chaining.
76  /// \throws jedi::xError if data.size() does not match the number of total
77  /// elements described by dimensions.
78  template <class DataType>
79  DerivedHasAtts add(const std::string& attrname, ::std::initializer_list<DataType> data,
80  const ::std::vector<Dimensions_t>& dimensions) {
81  auto derivedThis = static_cast<DerivedHasAtts*>(this);
82  auto att = derivedThis->template create<DataType>(attrname, dimensions);
83  att.template write<DataType>(data);
84  return *derivedThis;
85  }
86 
87  /// \brief Create and write an Attribute, for a single-dimensional span of 1-D data.
88  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
89  /// \param attrname is the name of the Attribute to be created.
90  /// \param data is a gsl::span (a pointer-length pair) that contains the data to be written.
91  /// The new Attribute will be one-dimensional and the length of the overall span.
92  /// \returns Another instance of this Has_Attribute. Used for operation chaining.
93  /// \see gsl::span for details of how to make a span.
94  /// \see gsl::make_span
95  template <class DataType>
96  DerivedHasAtts add(const std::string& attrname, ::gsl::span<const DataType> data) {
97  return add(attrname, data, {gsl::narrow<Dimensions_t>(data.size())});
98  }
99 
100  /// \brief Create and write an Attribute, for a 1-D initializer list.
101  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
102  /// \param attrname is the name of the Attribute to be created.
103  /// \param data is an initializer list that contains the data to be written.
104  /// The new Attribute will be one-dimensional and the length of the overall span.
105  /// \returns Another instance of this Has_Attribute. Used for operation chaining.
106  template <class DataType>
107  DerivedHasAtts add(const std::string& attrname, ::std::initializer_list<DataType> data) {
108  return add(attrname, data, {gsl::narrow<Dimensions_t>(data.size())});
109  }
110 
111  /// \brief Create and write a single datum of an Attribute.
112  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
113  /// \param attrname is the name of the Attribute to be created.
114  /// \param data is the data to be written. The new Attribute will be zero-dimensional and will
115  /// contain only this datum.
116  /// \note Even 0-dimensional data have a type, which may be a compound
117  /// array (i.e. a single string of variable length).
118  /// \returns Another instance of this Has_Attribute. Used for operation chaining.
119  template <class DataType>
120  DerivedHasAtts add(const std::string& attrname, const DataType& data) {
121  return add(attrname, gsl::make_span(&data, 1), {1});
122  }
123 
124  template <class EigenClass>
125  DerivedHasAtts addWithEigenRegular(const std::string& attrname, const EigenClass& data,
126  bool is2D = true) {
127 #if 1 //__has_include("Eigen/Dense")
128  typedef typename EigenClass::Scalar ScalarType;
129  // If d is already in Row Major form, then this is optimized out.
130  Eigen::Array<ScalarType, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor> dout;
131  dout.resize(data.rows(), data.cols());
132  dout = data;
133  const auto& dconst = dout; // To make some compilers happy.
134  auto sp = gsl::make_span(dconst.data(), static_cast<int>(data.rows() * data.cols()));
135 
136  if (is2D)
137  return add(attrname, sp,
138  {gsl::narrow<Dimensions_t>(data.rows()), gsl::narrow<Dimensions_t>(data.cols())});
139  else
140  return add(attrname, sp);
141 #else
142  static_assert(false, "The Eigen headers cannot be found, so this function cannot be used.");
143 #endif
144  }
145 
146  template <class EigenClass>
147  DerivedHasAtts addWithEigenTensor(const std::string& attrname, const EigenClass& data) {
148 #if 1 //__has_include("unsupported/Eigen/CXX11/Tensor")
149  typedef typename EigenClass::Scalar ScalarType;
151 
152  auto derived_this = static_cast<DerivedHasAtts*>(this);
153  Attribute att = derived_this->template create<ScalarType>(attrname, dims.dimsCur);
154 
155  att.writeWithEigenTensor(data);
156  return *derived_this;
157 #else
158  static_assert(
159  false, "The Eigen unsupported/ headers cannot be found, so this function cannot be used.");
160 #endif
161  }
162 
163  /// @}
164 };
165 
166 /// \brief Describes the functions that can read attributes
167 /// \ingroup ioda_cxx_attribute
168 /// \details Uses the (CRTP)[https://en.wikipedia.org/wiki/Curiously_recurring_template_pattern]
169 /// to implement this as we use the same functions for placeholder attribute construction.
170 /// \see Has_Attributes
171 template <class DerivedHasAtts>
173 protected:
175 
176 public:
177  virtual ~CanReadAttributes() {}
178 
179  /// @name Convenience functions for reading attributes
180  /// @{
181 
182  /// \brief Open and read an Attribute, for expected dimensions.
183  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
184  /// \param attrname is the name of the Attribute to be read.
185  /// \param data is a pointer-size pair to the data buffer that is filled with the metadata's
186  /// contents. It should be pre-sized to accomodate all of the matadata. See
187  /// getDimensions().numElements. Data will be filled in row-major order.
188  /// \throws jedi::xError on a size mismatch between Attribute dimensions and data.size().
189  template <class DataType>
190  const DerivedHasAtts read(const std::string& attrname, gsl::span<DataType> data) const {
191  // Attribute att = this->open(attrname);
192  Attribute att = static_cast<const DerivedHasAtts*>(this)->open(attrname);
193  att.read(data);
194  return *(static_cast<const DerivedHasAtts*>(this));
195  }
196 
197  /// \brief Open and read an Attribute, with unknown dimensions.
198  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
199  /// \param attrname is the name of the Attribute to be read.
200  /// \param data is a vector acting as a data buffer that is filled with the metadata's
201  /// contents. It gets resized as needed. data will be filled in row-major order.
202  template <class DataType>
203  const DerivedHasAtts read(const std::string& attrname, std::vector<DataType>& data) const {
204  Attribute att = static_cast<const DerivedHasAtts*>(this)->open(attrname);
205  att.read(data);
206  return *(static_cast<const DerivedHasAtts*>(this));
207  }
208 
209  /// \brief Open and read an Attribute, with unknown dimensions.
210  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
211  /// \param attrname is the name of the Attribute to be read.
212  /// \param data is a valarray acting as a data buffer that is filled with the metadata's
213  /// contents. It gets resized as needed. data will be filled in row-major order.
214  template <class DataType>
215  const DerivedHasAtts read(const std::string& attrname, std::valarray<DataType>& data) const {
216  Attribute att = static_cast<const DerivedHasAtts*>(this)->open(attrname);
217  att.read(data);
218  return *(static_cast<const DerivedHasAtts*>(this));
219  }
220 
221  /// \brief Read a datum of an Attribute.
222  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
223  /// \param attrname is the name of the Attribute to be read.
224  /// \param data is a datum of type DataType.
225  /// \throws jedi::xError if the underlying data have size > 1.
226  template <class DataType>
227  const DerivedHasAtts read(const std::string& attrname, DataType& data) const {
228  Attribute att = static_cast<const DerivedHasAtts*>(this)->open(attrname);
229  att.read<DataType>(data);
230  return *(static_cast<const DerivedHasAtts*>(this));
231  }
232 
233  /// \brief Read a datum of an Attribute.
234  /// \tparam DataType is the type of the data. I.e. float, int, int32_t, uint16_t, std::string, etc.
235  /// \param attrname is the name of the Attribute to be read.
236  /// \returns A datum of type DataType.
237  /// \throws jedi::xError if the underlying data have size > 1.
238  template <class DataType>
239  DataType read(const std::string& attrname) const {
240  Attribute att = static_cast<const DerivedHasAtts*>(this)->open(attrname);
241  return att.read<DataType>();
242  }
243 
244  template <class EigenClass, bool Resize = detail::EigenCompat::CanResize<EigenClass>::value>
245  DerivedHasAtts readWithEigenRegular(const std::string& attrname, EigenClass& data) {
246 #if 1 // __has_include("Eigen/Dense")
247  Attribute att = static_cast<const DerivedHasAtts*>(this)->open(attrname);
248  att.readWithEigenRegular<EigenClass, Resize>(data);
249  return *(static_cast<const DerivedHasAtts*>(this));
250 #else
251  static_assert(false, "The Eigen headers cannot be found, so this function cannot be used.")
252 #endif
253  }
254 
255  template <class EigenClass>
256  DerivedHasAtts readWithEigenTensor(const std::string& attrname, EigenClass& data) {
257 #if 1 //__has_include("unsupported/Eigen/CXX11/Tensor")
258  Attribute att = static_cast<const DerivedHasAtts*>(this)->open(attrname);
259  att.readWithEigenTensor<EigenClass>(data);
260  return *(static_cast<const DerivedHasAtts*>(this));
261 #else
262  static_assert(
263  false, "The Eigen unsupported/ headers cannot be found, so this function cannot be used.");
264 #endif
265  }
266 
267  /// @}
268 };
269 
270 /// \ingroup ioda_cxx_attribute
272 private:
273  /// Using an opaque object to implement the backend.
274  std::shared_ptr<Has_Attributes_Backend> backend_;
275 
276  // Backend classes do occasionally need access to the Attribute_Backend object.
277  friend class Group_Backend;
278  friend class Variable_Backend;
279 
280 protected:
281  Has_Attributes_Base(std::shared_ptr<Has_Attributes_Backend>);
282 
283 public:
285 
286  /// Query the backend and get the type provider.
287  virtual detail::Type_Provider* getTypeProvider() const;
288 
289  /// @name General Functions
290  /// @{
291  ///
292 
293  /// List all attributes
294  /// \returns an unordered vector of attribute names for an object.
295  virtual std::vector<std::string> list() const;
296  /// List all attributes
297  /// \returns an unordered vector of attribute names for an object.
298  inline std::vector<std::string> operator()() const { return list(); }
299 
300  /// \brief Does an Attribute with the specified name exist?
301  /// \param attname is the name of the Attribute that we are looking for.
302  /// \returns true if it exists.
303  /// \returns false otherwise.
304  virtual bool exists(const std::string& attname) const;
305  /// \brief Delete an Attribute with the specified name.
306  /// \param attname is the name of the Attribute that we are deleting.
307  /// \throws jedi::xError if no such attribute exists.
308  virtual void remove(const std::string& attname);
309  /// \brief Open an Attribute by name
310  /// \param name is the name of the Attribute to be opened.
311  /// \returns An instance of an Attribute that can be queried (with getDimensions()) and read.
312  virtual Attribute open(const std::string& name) const;
313  /// \brief Open Attribute by name
314  /// \param name is the name of the Attribute to be opened.
315  /// \returns An instance of Attribute that can be queried (with getDimensions()) and read.
316  inline Attribute operator[](const std::string& name) const { return open(name); }
317 
318  /// \brief Open all attributes in an object.
319  /// \details This is a collective call, optimized for performance.
320  /// \return A sequence of (name, Attribute) pairs.
321  virtual std::vector<std::pair<std::string, Attribute>> openAll() const;
322 
323  /// \brief Create an Attribute without setting its data.
324  /// \param attrname is the name of the Attribute.
325  /// \param dimensions is a vector representing the size of the metadata.
326  /// Each element of the vector is a dimension with a certain size.
327  /// \param in_memory_datatype is the runtime description of the Attribute's data type.
328  /// \returns An instance of Attribute that can be written to.
329  virtual Attribute create(const std::string& attrname, const Type& in_memory_dataType,
330  const std::vector<Dimensions_t>& dimensions = {1});
331 
332  /// Python compatability function
333  inline Attribute _create_py(const std::string& attrname, BasicTypes dataType,
334  const std::vector<Dimensions_t>& dimensions = {1}) {
335  return create(attrname, ::ioda::Type(dataType, getTypeProvider()), dimensions);
336  }
337 
338  /// \brief Create an Attribute without setting its data.
339  /// \tparam DataType is the type of the data. I.e. float, int, uint16_t, std::string, etc.
340  /// \param attrname is the name of the Attribute.
341  /// \param dimensions is a vector representing the size of the metadata. Each element of the
342  /// vector is a dimension with a certain size.
343  /// \returns An instance of Attribute that can be written to.
344  template <class DataType, class TypeWrapper = Types::GetType_Wrapper<DataType>>
345  Attribute create(const std::string& attrname,
346  const std::vector<Dimensions_t>& dimensions = {1}) {
347  try {
348  Type in_memory_dataType = TypeWrapper::GetType(getTypeProvider());
349  auto att = create(attrname, in_memory_dataType, dimensions);
350  return att;
351  } catch (...) {
352  std::throw_with_nested(Exception(ioda_Here()));
353  }
354  }
355 
356  /// \brief Rename an Attribute
357  /// \param oldName is the name of the Attribute to be changed.
358  /// \param newName is the new name of the Attribute.
359  /// \throws jedi::xError if oldName is not found.
360  /// \throws jedi::xError if newName already exists.
361  virtual void rename(const std::string& oldName, const std::string& newName);
362 
363  /// @}
364 };
365 
367 protected:
369 
370 public:
372 
373  /// \brief Default implementation of Has_Attributes_Base::openAll.
374  /// \see Has_Attributes_Base::openAll.
375  std::vector<std::pair<std::string, Attribute>> openAll() const override;
376 };
377 } // namespace detail
378 
379 /** \brief This class exists inside of ioda::Group or ioda::Variable and provides
380  * the interface to manipulating Attributes.
381  * \ingroup ioda_cxx_attribute
382  *
383  * \note It should only be constructed inside of a Group or Variable.
384  * It has no meaning elsewhere.
385  * \see Attribute for the class that represents individual attributes.
386  * \throws jedi::xError on all exceptions.
387  **/
388 class IODA_DL Has_Attributes : public detail::CanAddAttributes<Has_Attributes>,
389  public detail::CanReadAttributes<Has_Attributes>,
391 public:
392  Has_Attributes();
393  Has_Attributes(std::shared_ptr<detail::Has_Attributes_Backend>);
394  virtual ~Has_Attributes();
395 };
396 
397 } // namespace ioda
398 
399 /// @} // End Doxygen block
Interfaces for ioda::Attribute and related classes.
Describe the dimensions of a ioda::Attribute or ioda::Variable.
Convenience functions to work with Eigen objects.
IODA's error system.
Interfaces for ioda::Type and related classes.
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
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
virtual ~Has_Attributes()
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
Attribute_Implementation writeWithEigenTensor(const EigenClass &d)
Write an Eigen Tensor-like object.
Definition: Attribute.h:195
Attribute_Implementation readWithEigenRegular(EigenClass &res) const
Read data into an Eigen::Array, Eigen::Matrix, Eigen::Map, etc.
Definition: Attribute.h:347
Attribute_Implementation readWithEigenTensor(EigenClass &res) const
Read data into an Eigen::Array, Eigen::Matrix, Eigen::Map, etc.
Definition: Attribute.h:401
virtual Attribute_Implementation read(gsl::span< char > data, const Type &in_memory_dataType) const
The fundamental read function. Backends overload this function to implement all read operations.
Definition: Attribute.cpp:75
Describes the functions that can add attributes.
DerivedHasAtts addWithEigenTensor(const std::string &attrname, const EigenClass &data)
DerivedHasAtts add(const std::string &attrname, ::gsl::span< const DataType > data, const ::std::vector< Dimensions_t > &dimensions)
Create and write an Attribute, for arbitrary dimensions.
DerivedHasAtts add(const std::string &attrname, ::gsl::span< const DataType > data)
Create and write an Attribute, for a single-dimensional span of 1-D data.
DerivedHasAtts addWithEigenRegular(const std::string &attrname, const EigenClass &data, bool is2D=true)
DerivedHasAtts add(const std::string &attrname, ::std::initializer_list< DataType > data)
Create and write an Attribute, for a 1-D initializer list.
DerivedHasAtts add(const std::string &attrname, ::std::initializer_list< DataType > data, const ::std::vector< Dimensions_t > &dimensions)
Create and write an Attribute, for arbitrary dimensions.
DerivedHasAtts add(const std::string &attrname, const DataType &data)
Create and write a single datum of an Attribute.
Describes the functions that can read attributes.
DataType read(const std::string &attrname) const
Read a datum of an Attribute.
DerivedHasAtts readWithEigenTensor(const std::string &attrname, EigenClass &data)
const DerivedHasAtts read(const std::string &attrname, std::valarray< DataType > &data) const
Open and read an Attribute, with unknown dimensions.
const DerivedHasAtts read(const std::string &attrname, gsl::span< DataType > data) const
Open and read an Attribute, for expected dimensions.
DerivedHasAtts readWithEigenRegular(const std::string &attrname, EigenClass &data)
const DerivedHasAtts read(const std::string &attrname, std::vector< DataType > &data) const
Open and read an Attribute, with unknown dimensions.
const DerivedHasAtts read(const std::string &attrname, DataType &data) const
Read a datum of an Attribute.
std::vector< std::string > operator()() const
Attribute create(const std::string &attrname, const std::vector< Dimensions_t > &dimensions={1})
Create an Attribute without setting its data.
Attribute operator[](const std::string &name) const
Open Attribute by name.
std::shared_ptr< Has_Attributes_Backend > backend_
Using an opaque object to implement the backend.
Attribute _create_py(const std::string &attrname, BasicTypes dataType, const std::vector< Dimensions_t > &dimensions={1})
Python compatability function.
Backends implement type providers in conjunction with Attributes, Has_Attributes, Variables and Has_V...
Definition: Type_Provider.h:36
Variable backends inherit from this.
Definition: Variable.h:710
Common preprocessor definitions used throughout IODA.
#define IODA_DL
A preprocessor tag that indicates that a symbol is to be exported/imported.
Definition: defs.h:110
Type GetType(gsl::not_null< const ::ioda::detail::Type_Provider * > t, std::initializer_list< Dimensions_t > Adims={}, typename std::enable_if<!is_string< DataType >::value >::type *=0)
For fundamental, non-string types.
Definition: Type.h:189
Dimensions getTensorDimensions(EigenClass &e)
Definition: Eigen_Compat.h:72
BasicTypes
Definition: Type.h:37
#define ioda_Here()
Describes the dimensions of an Attribute or Variable.
Definition: Dimensions.h:22
std::vector< Dimensions_t > dimsCur
The dimensions of the data.
Definition: Dimensions.h:23