IODA
ObsStore-variables.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020-2021 UCAR
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  */
7 /*! \addtogroup ioda_internals_engines_obsstore
8  *
9  * @{
10  * \file ObsStore-variables.cpp
11  * \brief Functions for ioda::Variable and ioda::Has_Variables backed by ObsStore
12  */
13 #include "./ObsStore-variables.h"
14 #include "ioda/Exception.h"
15 
16 namespace ioda {
17 namespace Engines {
18 namespace ObsStore {
19 //*************************************************************************
20 // ObsStore_Variable_Backend functions
21 //*************************************************************************
24 ObsStore_Variable_Backend::ObsStore_Variable_Backend(std::shared_ptr<ioda::ObsStore::Variable> b)
25  : backend_(b) {
26  std::shared_ptr<ioda::ObsStore::Has_Attributes> b_atts(backend_->atts);
27  atts = Has_Attributes(std::make_shared<ObsStore_HasAttributes_Backend>(b_atts));
28 
29  std::shared_ptr<ioda::ObsStore::Has_Attributes> b_impl_atts(backend_->impl_atts);
30  impl_atts_ = Has_Attributes(std::make_shared<ObsStore_HasAttributes_Backend>(b_impl_atts));
31 }
32 
35 }
36 
38  auto backend_type = backend_->dtype();
39  ObsTypeInfo typ{backend_type.first, backend_type.second};
40  return Type{std::make_shared<ObsStore_Type>(typ), typeid(ObsStore_Type)};
41 }
42 
44  auto typeBackend = std::dynamic_pointer_cast<ObsStore_Type>(lhs.getBackend());
45  ioda::ObsStore::ObsTypes dtype = typeBackend->dtype();
46  return backend_->isOfType(dtype);
47 }
48 
50  try {
51  return backend_->hasFillValue();
52  } catch (std::bad_cast) {
53  std::throw_with_nested(Exception("Bad cast.", ioda_Here()));
54  }
55 }
56 
58  try {
59  return backend_->getFillValue();
60  } catch (std::bad_cast) {
61  std::throw_with_nested(Exception("Bad cast.", ioda_Here()));
62  }
63 }
64 
65 std::vector<Dimensions_t> ObsStore_Variable_Backend::getChunkSizes() const {
66  if (!impl_atts_.exists("_chunks")) return {};
67  std::vector<Dimensions_t> chunks;
68  impl_atts_.read<Dimensions_t>("_chunks", chunks);
69  return chunks;
70 }
71 
72 std::pair<bool, int> ObsStore_Variable_Backend::getGZIPCompression() const {
73  if (!impl_atts_.exists("_gzip")) return std::pair<bool, int>(false, 0);
74  return std::pair<bool, int>(true, impl_atts_.read<int>("_gzip"));
75 }
76 
77 std::tuple<bool, unsigned, unsigned> ObsStore_Variable_Backend::getSZIPCompression() const {
78  if (!impl_atts_.exists("_szip")) return std::tuple<bool, unsigned, unsigned>(false, 0, 0);
79  std::vector<unsigned> sz;
80  impl_atts_.read<unsigned>("_szip", sz);
81  Expects(sz.size() == 2);
82  return std::tuple<bool, unsigned, unsigned>(true, sz[0], sz[1]);
83 }
84 
86  // Convert to Dimensions types
87  std::vector<Dimensions_t> iodaDims = backend_->get_dimensions();
88  std::size_t numElems = 1;
89  for (std::size_t i = 0; i < iodaDims.size(); ++i) {
90  numElems *= iodaDims[i];
91  }
92 
93  // Create and return a Dimensions object
94  std::vector<Dimensions_t> iodaMaxDims = backend_->get_max_dimensions();
95  auto iodaRank = gsl::narrow<Dimensions_t>(iodaDims.size());
96  auto iodaNumElems = gsl::narrow<Dimensions_t>(numElems);
97  Dimensions dims(iodaDims, iodaMaxDims, iodaRank, iodaNumElems);
98  return dims;
99 }
100 
101 Variable ObsStore_Variable_Backend::resize(const std::vector<Dimensions_t>& newDims) {
102  backend_->resize(newDims);
103  return Variable{shared_from_this()};
104 }
105 
107  const Variable& scale) {
108  auto scaleBackendBase = scale.get();
109  auto scaleBackendDerived = std::dynamic_pointer_cast<ObsStore_Variable_Backend>(scaleBackendBase);
110  backend_->attachDimensionScale(DimensionNumber, scaleBackendDerived->backend_);
111 
112  return Variable{shared_from_this()};
113 }
114 
116  const Variable& scale) {
117  auto scaleBackendBase = scale.get();
118  auto scaleBackendDerived = std::dynamic_pointer_cast<ObsStore_Variable_Backend>(scaleBackendBase);
119  backend_->detachDimensionScale(DimensionNumber, scaleBackendDerived->backend_);
120 
121  return Variable{shared_from_this()};
122 }
123 
124 bool ObsStore_Variable_Backend::isDimensionScale() const { return backend_->isDimensionScale(); }
125 
126 Variable ObsStore_Variable_Backend::setIsDimensionScale(const std::string& dimensionScaleName) {
127  backend_->setIsDimensionScale(dimensionScaleName);
128  return Variable{shared_from_this()};
129 }
130 
132  backend_->getDimensionScaleName(res);
133  return Variable{std::make_shared<ObsStore_Variable_Backend>(*this)};
134 }
135 
136 /// Is a dimension scale attached to this Variable in a certain position?
137 bool ObsStore_Variable_Backend::isDimensionScaleAttached(unsigned int DimensionNumber,
138  const Variable& scale) const {
139  auto scaleBackendBase = scale.get();
140  auto scaleBackendDerived = std::dynamic_pointer_cast<ObsStore_Variable_Backend>(scaleBackendBase);
141  return backend_->isDimensionScaleAttached(DimensionNumber, scaleBackendDerived->backend_);
142 }
143 
144 Variable ObsStore_Variable_Backend::write(gsl::span<char> data, const Type& in_memory_dataType,
145  const Selection& mem_selection,
146  const Selection& file_selection) {
147  // Convert to an obs store data type
148  auto typeBackend = std::dynamic_pointer_cast<ObsStore_Type>(in_memory_dataType.getBackend());
149  ioda::ObsStore::ObsTypes dtype = typeBackend->dtype();
150  std::size_t dtype_size = typeBackend->dtype_size();
151 
152  // Convert to obs store selection
153  //
154  // Need to record dimension sizes in the ObsStore Selection object.
155  // The memory select comes from the frontend, and has its extent_
156  // member set to the dimension sizes of the frontend data. In the
157  // case where all points are selected, extent_ will be empty so create
158  // dimension sizes from the size data.
159  //
160  // The file select comes from the backend, and gets its dimension
161  // sizes from this variable.
162 
163  // Copy the selection extent (dim sizes)
164  std::size_t extSize = mem_selection.extent().size();
165  std::vector<Dimensions_t> dim_sizes(extSize);
166  if (extSize == 0) {
167  dim_sizes.push_back(data.size() / dtype_size);
168  } else {
169  for (std::size_t i = 0; i < extSize; ++i) {
170  dim_sizes[i] = mem_selection.extent()[i];
171  }
172  }
173 
174  ioda::ObsStore::Selection m_select = createObsStoreSelection(mem_selection, dim_sizes);
176  = createObsStoreSelection(file_selection, backend_->get_dimensions());
177 
178  // Check the number of points in the selections. Data transfer is going
179  // from memory to file so make sure the memory npoints is not greater
180  // than the file npoints.
181  std::size_t m_npts = m_select.npoints();
182  std::size_t f_npts = f_select.npoints();
183  if (m_npts > f_npts)
184  throw Exception("Number of points from memory is greater than that of file", ioda_Here())
185  .add("m_select.npoints()", m_npts)
186  .add("f_select.npoints()", f_npts);
187 
188  backend_->write(data, dtype, m_select, f_select);
189  return Variable{shared_from_this()};
190 }
191 
192 Variable ObsStore_Variable_Backend::read(gsl::span<char> data, const Type& in_memory_dataType,
193  const Selection& mem_selection,
194  const Selection& file_selection) const {
195  // Convert to an obs store data type
196  auto typeBackend = std::dynamic_pointer_cast<ObsStore_Type>(in_memory_dataType.getBackend());
197  ioda::ObsStore::ObsTypes dtype = typeBackend->dtype();
198  std::size_t dtype_size = typeBackend->dtype_size();
199 
200  // Convert to obs store selection
201  //
202  // Need to record dimension sizes in the ObsStore Selection object.
203  // The memory select comes from the frontend, and has its extent_
204  // member set to the dimension sizes of the frontend data. In the
205  // case where all points are selected, extent_ will be empty so create
206  // dimension sizes from the size data.
207  //
208  // The file select comes from the backend, and gets its dimension
209  // sizes from this variable.
210 
211  // Copy the selection extent (dim sizes)
212  std::size_t extSize = mem_selection.extent().size();
213  std::vector<Dimensions_t> dim_sizes(extSize);
214  if (extSize == 0) {
215  dim_sizes.push_back(data.size() / dtype_size);
216  } else {
217  for (std::size_t i = 0; i < extSize; ++i) {
218  dim_sizes[i] = mem_selection.extent()[i];
219  }
220  }
221 
222  ioda::ObsStore::Selection m_select = createObsStoreSelection(mem_selection, dim_sizes);
224  = createObsStoreSelection(file_selection, backend_->get_dimensions());
225 
226  // Check the number of points in the selections. Data transfer is going
227  // from file to memory so make sure the file npoints is not greater
228  // than the memory npoints.
229  std::size_t m_npts = m_select.npoints();
230  std::size_t f_npts = f_select.npoints();
231  if (m_npts > f_npts)
232  throw Exception("Number of points from file is greater than that of memory", ioda_Here())
233  .add("m_select.npoints()", m_npts)
234  .add("f_select.npoints()", f_npts);
235 
236  backend_->read(data, dtype, m_select, f_select);
237  // Need to construct a shared_ptr to "this", instead of using
238  // shared_from_this() because of the const qualifier on this method.
239  return Variable{std::make_shared<ObsStore_Variable_Backend>(*this)};
240 }
241 
242 //*************************************************************************
243 // ObsStore_HasVariables_Backend functions
244 //*************************************************************************
247  std::shared_ptr<ioda::ObsStore::Has_Variables> b)
248  : backend_(b) {}
250 
253 }
254 
255 bool ObsStore_HasVariables_Backend::exists(const std::string& name) const {
256  return backend_->exists(name);
257 }
258 
259 void ObsStore_HasVariables_Backend::remove(const std::string& name) { backend_->remove(name); }
260 
262  auto res = backend_->open(name);
263  auto b = std::make_shared<ObsStore_Variable_Backend>(res);
264  Variable var{b};
265  return var;
266 }
267 
268 std::vector<std::string> ObsStore_HasVariables_Backend::list() const { return backend_->list(); }
269 
271  const Type& in_memory_dataType,
272  const std::vector<Dimensions_t>& dimensions,
273  const std::vector<Dimensions_t>& max_dimensions,
275  // Convert to an obs store data type
276  auto typeBackend = std::dynamic_pointer_cast<ObsStore_Type>(in_memory_dataType.getBackend());
277  ioda::ObsStore::ObsTypes dtype = typeBackend->dtype();
278 
279  // If max_dimensions not specified (empty), then copy from dimensions
280  std::vector<Dimensions_t> max_dims;
281  if (max_dimensions.empty()) {
282  max_dims = dimensions;
283  } else {
284  max_dims = max_dimensions;
285  }
286 
287  // Convert to obs store create parameters.
289  os_params.dtype_size = typeBackend->dtype_size();
290 
291  os_params.fvdata = params.fillValue_;
292  const auto fvdata_final = params.finalize(); // Using in a span. Keep in scope.
293  if (os_params.fvdata.set_) {
294  os_params.fill_value
295  = gsl::make_span<char>((char*)&(fvdata_final), sizeof(fvdata_final)); // NOLINT
296  }
297 
298  // Call backend create
299  auto res = backend_->create(name, dtype, dimensions, max_dims, os_params);
300  auto b = std::make_shared<ObsStore_Variable_Backend>(res);
301 
302  // Also set the chunking and compression parameters
303  if (params.chunk) b->impl_atts_.add<Dimensions_t>("_chunks", params.getChunks(dimensions));
304  if (params.gzip_) b->impl_atts_.add<int>("_gzip", params.gzip_level_);
305  if (params.szip_)
306  b->impl_atts_.add<unsigned>("_szip", {params.szip_options_, params.szip_PixelsPerBlock_});
307 
308  Variable var{b};
309 
310  return var;
311 }
312 } // namespace ObsStore
313 } // namespace Engines
314 } // namespace ioda
315 
316 /// @}
IODA's error system.
Functions for ioda::Variable and ioda::Has_Variables backed by ObsStore.
std::vector< std::string > list() const final
return list of variables in this container
Variable open(const std::string &name) const final
open variable (throws exception if not found)
std::shared_ptr< ioda::ObsStore::Has_Variables > backend_
ObsStore Has_Variables.
detail::Type_Provider * getTypeProvider() const final
return an ObsStore type marker
Variable create(const std::string &name, const Type &in_memory_dataType, const std::vector< Dimensions_t > &dimensions={1}, const std::vector< Dimensions_t > &max_dimensions={}, const VariableCreationParameters &params=VariableCreationParameters()) final
create a new variable
bool exists(const std::string &name) const final
return true if variable exists
void remove(const std::string &name) final
remove variable
static ObsStore_Type_Provider * instance()
create instance of a type provider for frontend
Type getType() const final
Encapsulate the internal ObsStore_Type object as a Type.
FillValueData_t getFillValue() const final
Get the fill value associated with the Variable.
bool isDimensionScale() const final
is this variable a dimension scale (ie, hold coordinate values)
Variable resize(const std::vector< Dimensions_t > &newDims) final
resize dimensions
std::pair< bool, int > getGZIPCompression() const final
Get GZIP compression information.
Variable write(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection, const Selection &file_selection) final
transfer data into the ObsStore Variable
std::shared_ptr< ioda::ObsStore::Variable > backend_
ObsStore Variable.
detail::Type_Provider * getTypeProvider() const final
return an ObsStore type marker
std::vector< Dimensions_t > getChunkSizes() const final
Get chunking information.
Variable setIsDimensionScale(const std::string &dimensionScaleName) final
set flag to denote this variable as a dimension scale
Dimensions getDimensions() const final
return dimensions of this variable
Variable attachDimensionScale(unsigned int DimensionNumber, const Variable &scale) final
attach dimension to this variable
bool hasFillValue() const final
Does the Variable have an associated fill value?
bool isDimensionScaleAttached(unsigned int DimensionNumber, const Variable &scale) const final
is the given dimension scale attached to the given dimension number
Variable read(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection, const Selection &file_selection) const final
transfer data from the ObsStore Variable
Variable detachDimensionScale(unsigned int DimensionNumber, const Variable &scale) final
detach dimensions to this variable
std::tuple< bool, unsigned, unsigned > getSZIPCompression() const final
Get SZIP compression information.
The ioda exception class.
Definition: Exception.h:54
Exception & add(const std::string &key, const T value)
Add a key-value pair to the error message.
Definition: Exception.h:75
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
std::size_t npoints() const
returns number of points in selection
A Selection represents the bounds of the data, in ioda or in userspace, that you are reading or writi...
Definition: Selection.h:48
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
Variables store data!
Definition: Variable.h:680
const DerivedHasAtts read(const std::string &attrname, gsl::span< DataType > data) const
Open and read an Attribute, for expected dimensions.
virtual bool exists(const std::string &attname) const
Does an Attribute with the specified name exist?
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
bool isA() const
Convenience function to check a Variable's storage type.
Definition: Variable.h:99
Has_Attributes atts
Attributes.
Definition: Variable.h:71
std::string getDimensionScaleName() const
Get the name of this Variable's defined dimension scale.
Definition: Variable.h:217
std::shared_ptr< Variable_Backend > get() const
Gets a handle to the underlying object that implements the backend functionality.
Definition: Variable.cpp:21
@ ObsStore
ObsStore in-memory.
Selection & extent(const VecDimensions_t &sz)
Provide the dimensions of the object that you are selecting from.
Definition: Selection.h:111
ObsTypes
ObsStore data type markers.
Definition: Types.hpp:30
ioda::ObsStore::Selection createObsStoreSelection(const ioda::Selection &selection, const std::vector< Dimensions_t > &dim_sizes)
translate a ioda::Selection to and ObsStore Selection
list newDims
Definition: 05-ObsGroup.py:95
std::pair< ioda::ObsStore::ObsTypes, std::size_t > ObsTypeInfo
#define ioda_Here()
Describes the dimensions of an Attribute or Variable.
Definition: Dimensions.h:22
parameters for creating a new variable
Definition: Variables.hpp:38
gsl::span< char > fill_value
Definition: Variables.hpp:45
detail::FillValueData_t fvdata
Definition: Variables.hpp:44
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57
Container used to store and manipulate fill values.
Definition: Fill.h:35