IODA
Attribute_Creator.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 Attribute_Creator.h
12  * \brief Flywheel creation of ioda::Attribute. Used by ioda::Has_Attributes and
13  * ioda::VariableCreationParameters.
14  */
15 
16 #include <gsl/gsl-lite.hpp>
17 #include <iostream>
18 #include <memory>
19 #include <string>
20 #include <vector>
21 
24 #include "ioda/Misc/Dimensions.h"
25 #include "ioda/Types/Type.h"
26 #include "ioda/defs.h"
27 
28 namespace ioda {
29 namespace detail {
30 /// \brief Flywheel creation of ioda::Attribute.
31 /// \ingroup ioda_cxx_attribute
33 protected:
34  std::string name_;
35 
36 public:
38  virtual void apply(Has_Attributes& obj) const = 0;
39  Attribute_Creator_Base(const std::string& name);
40 };
41 } // namespace detail
42 
43 /// \brief Flywheel creation of ioda::Attribute.
44 /// \ingroup ioda_cxx_attribute
45 template <class DataType>
47 private:
48  ::std::vector<Dimensions_t> dimensions_;
49  ::std::vector<DataType> data_;
50 
51 public:
52  virtual ~Attribute_Creator() {}
53  void apply(Has_Attributes& obj) const override { obj.add<DataType>(name_, data_, dimensions_); }
54 
55  /// \details Totally taking advantage of vector constructors.
56  template <class DataInput, class DimensionInput>
57  Attribute_Creator(const std::string& name, DataInput data, DimensionInput dimensions)
58  : Attribute_Creator_Base(name), data_(data.begin(), data.end()), dimensions_(dimensions) {}
59 
60  template <class DimensionInput>
61  Attribute_Creator(const std::string& name, DimensionInput dimensions)
62  : Attribute_Creator_Base(name), dimensions_(dimensions) {}
63 
64  template <class DataInput_junk_param = DataType>
65  void write(const ::gsl::span<const DataType>& data) {
66  data_ = ::std::vector<DataType>(data.begin(), data.end());
67  }
68 };
69 
70 /// \brief Flywheel creation of ioda::Attribute objects.
71 /// \ingroup ioda_cxx_attribute
72 /// This is needed because you might want to make the same Attribute in multiple places.
73 class IODA_DL Attribute_Creator_Store : public detail::CanAddAttributes<Attribute_Creator_Store> {
74  std::vector<std::shared_ptr<detail::Attribute_Creator_Base>> atts_;
75 
76 public:
79 
80  void apply(Has_Attributes& obj) const;
81 
82  /// @name Convenience functions for adding attributes
83  /// @{
84  /// \see CanAddAttributes
85  /// \see CanAddAttributes::add
86 
87  template <class DataType>
88  Attribute_Creator_Store& create(const std::string& attrname, ::gsl::span<const DataType> data,
89  ::std::initializer_list<Dimensions_t> dimensions) {
90  atts_.push_back(std::make_shared<Attribute_Creator<DataType>>(attrname, data, dimensions));
91  return *this;
92  }
93 
94  template <class DataType>
95  Attribute_Creator_Store& create(const std::string& attrname,
96  ::std::initializer_list<DataType> data,
97  ::std::initializer_list<Dimensions_t> dimensions) {
98  atts_.push_back(std::make_shared<Attribute_Creator<DataType>>(attrname, data, dimensions));
99  return *this;
100  }
101 
102  template <class DataType2>
103  struct AttWrapper {
104  std::shared_ptr<Attribute_Creator<DataType2>> inner;
105  AttWrapper(std::shared_ptr<Attribute_Creator<DataType2>> d) : inner(d) {}
106  template <class DataInput_junk_param = DataType2>
107  void write(const ::gsl::span<const DataType2>& data) {
108  inner->write(data);
109  }
110  };
111 
112  template <class DataType>
113  AttWrapper<DataType> create(const std::string& attrname, ::std::vector<Dimensions_t> dimensions) {
114  auto res = std::make_shared<Attribute_Creator<DataType>>(attrname, dimensions);
115  atts_.push_back(res);
116  return AttWrapper<DataType>{res};
117  }
118 
119  /// @}
120 };
121 
122 } // namespace ioda
123 
124 /// @} // End Doxygen block
Interfaces for ioda::Attribute and related classes.
Describe the dimensions of a ioda::Attribute or ioda::Variable.
Interfaces for ioda::Has_Attributes and related classes.
Interfaces for ioda::Type and related classes.
Flywheel creation of ioda::Attribute objects.This is needed because you might want to make the same A...
std::vector< std::shared_ptr< detail::Attribute_Creator_Base > > atts_
Attribute_Creator_Store & create(const std::string &attrname, ::std::initializer_list< DataType > data, ::std::initializer_list< Dimensions_t > dimensions)
Attribute_Creator_Store & create(const std::string &attrname, ::gsl::span< const DataType > data, ::std::initializer_list< Dimensions_t > dimensions)
AttWrapper< DataType > create(const std::string &attrname, ::std::vector< Dimensions_t > dimensions)
Flywheel creation of ioda::Attribute.
void write(const ::gsl::span< const DataType > &data)
::std::vector< Dimensions_t > dimensions_
Attribute_Creator(const std::string &name, DimensionInput dimensions)
::std::vector< DataType > data_
Attribute_Creator(const std::string &name, DataInput data, DimensionInput dimensions)
void apply(Has_Attributes &obj) const override
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
Flywheel creation of ioda::Attribute.
Attribute_Creator_Base(const std::string &name)
virtual void apply(Has_Attributes &obj) const =0
Describes the functions that can add attributes.
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.
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
std::shared_ptr< Attribute_Creator< DataType2 > > inner
void write(const ::gsl::span< const DataType2 > &data)
AttWrapper(std::shared_ptr< Attribute_Creator< DataType2 >> d)