IODA
Variable.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  */
9 #include "ioda/Exception.h"
10 
11 namespace ioda {
12 namespace detail {
13 
14 template <>
15 Variable_Base<>::Variable_Base(std::shared_ptr<Variable_Backend> backend)
16  : backend_(backend), atts((backend) ? backend->atts : Has_Attributes()) {}
17 template <>
19 
20 template <>
21 std::shared_ptr<Variable_Backend> Variable_Base<>::get() const {
22  return backend_;
23 }
24 
25 template <>
26 bool Variable_Base<>::isA(Type lhs) const {
27  try {
28  if (backend_ == nullptr)
29  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
30  return backend_->isA(lhs);
31  } catch (...) {
32  std::throw_with_nested(Exception(
33  "An exception occurred inside ioda while checking variable type.", ioda_Here()));
34  }
35 }
36 template <>
38  try {
39  if (backend_ == nullptr)
40  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
41  return backend_->getTypeProvider();
42  } catch (...) {
43  std::throw_with_nested(Exception(
44  "An exception occurred inside ioda while getting a backend type provider.", ioda_Here()));
45  }
46 }
47 
48 template <>
50  try {
51  if (backend_ == nullptr)
52  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
53  return backend_->getType();
54  } catch (...) {
55  std::throw_with_nested(Exception(
56  "An exception occurred inside ioda while determining variable type.", ioda_Here()));
57  }
58 }
59 
60 /// \note This is very inefficient. Refactor?
61 template <>
63  try {
64  if (isA(BasicTypes::float_)) return BasicTypes::float_;
65  if (isA(BasicTypes::int32_)) return BasicTypes::int32_;
66  if (isA<double>()) return BasicTypes::double_;
67  if (isA<int16_t>()) return BasicTypes::int16_;
68  if (isA<int64_t>()) return BasicTypes::int64_;
69  if (isA<uint16_t>()) return BasicTypes::uint16_;
70  if (isA<uint32_t>()) return BasicTypes::uint32_;
71  if (isA<uint64_t>()) return BasicTypes::uint64_;
72  if (isA<std::string>()) return BasicTypes::str_;
73  if (isA<long double>()) return BasicTypes::ldouble_;
74  if (isA<char>()) return BasicTypes::char_;
75  if (isA<bool>()) return BasicTypes::bool_;
76 
78  } catch (...) {
79  std::throw_with_nested(Exception(
80  "An exception occurred inside ioda while determining variable type.", ioda_Here()));
81  }
82 }
83 
84 template <>
86  try {
87  if (backend_ == nullptr)
88  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
89  return backend_->getCreationParameters(doAtts, doDims);
90  } catch (...) {
91  std::throw_with_nested(Exception(
92  "An exception occurred inside ioda while getting creation-time metadata of a variable.",
93  ioda_Here()));
94  }
95 }
96 
97 template <>
99  try {
100  if (backend_ == nullptr)
101  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
102  return backend_->hasFillValue();
103  } catch (...) {
104  std::throw_with_nested(Exception(
105  "An exception occurred inside ioda while determining if a variable has a fill value.",
106  ioda_Here()));
107  }
108 }
109 
110 template <>
112  try {
113  if (backend_ == nullptr)
114  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
115  return backend_->getFillValue();
116  } catch (...) {
117  std::throw_with_nested(Exception(
118  "An exception occurred inside ioda while reading a variable's fill value.", ioda_Here()));
119  }
120 }
121 
122 template <>
123 std::vector<Dimensions_t> Variable_Base<>::getChunkSizes() const {
124  try {
125  if (backend_ == nullptr)
126  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
127  return backend_->getChunkSizes();
128  } catch (...) {
129  std::throw_with_nested(Exception(
130  "An exception occurred inside ioda while determining a variable's chunking options.",
131  ioda_Here()));
132  }
133 }
134 
135 template <>
136 std::pair<bool, int> Variable_Base<>::getGZIPCompression() const {
137  try {
138  if (backend_ == nullptr)
139  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
140  return backend_->getGZIPCompression();
141  } catch (...) {
142  std::throw_with_nested(Exception(
143  "An exception occurred inside ioda while reading GZIP compression options.", ioda_Here()));
144  }
145 }
146 
147 template <>
148 std::tuple<bool, unsigned, unsigned> Variable_Base<>::getSZIPCompression() const {
149  try {
150  if (backend_ == nullptr)
151  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
152  return backend_->getSZIPCompression();
153  } catch (...) {
154  std::throw_with_nested(Exception(
155  "An exception occurred inside ioda while reading SZIP compression options.", ioda_Here()));
156  }
157 }
158 
159 template <>
161  try {
162  if (backend_ == nullptr)
163  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
164  return backend_->getDimensions();
165  } catch (...) {
166  std::throw_with_nested(Exception(
167  "An exception occurred inside ioda while reading a variable's dimensions.", ioda_Here()));
168  }
169 }
170 
171 template <>
172 Variable Variable_Base<>::resize(const std::vector<Dimensions_t>& newDims) {
173  try {
174  if (backend_ == nullptr)
175  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
176  return backend_->resize(newDims);
177  } catch (...) {
178  std::throw_with_nested(Exception(
179  "An exception occurred inside ioda while resizing a variable.", ioda_Here()));
180  }
181 }
182 
183 template <>
184 Variable Variable_Base<>::attachDimensionScale(unsigned int DimensionNumber,
185  const Variable& scale) {
186  try {
187  if (backend_ == nullptr)
188  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
189  return backend_->attachDimensionScale(DimensionNumber, scale);
190  } catch (...) {
191  std::throw_with_nested(Exception(
192  "An exception occurred inside ioda while attaching a dimension scale to a variable.",
193  ioda_Here()));
194  }
195 }
196 
197 template <>
198 Variable Variable_Base<>::detachDimensionScale(unsigned int DimensionNumber,
199  const Variable& scale) {
200  try {
201  if (backend_ == nullptr)
202  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
203  return backend_->detachDimensionScale(DimensionNumber, scale);
204  } catch (...) {
205  std::throw_with_nested(Exception(
206  "An exception occurred inside ioda while detaching a dimension "
207  "scale from a variable.", ioda_Here()));
208  }
209 }
210 
211 template <>
212 Variable Variable_Base<>::setDimScale(const std::vector<Variable>& vdims) {
213  try {
214  for (unsigned int i = 0; i < vdims.size(); ++i)
215  attachDimensionScale(i, vdims[i]);
216  return Variable{backend_};
217  } catch (...) {
218  std::throw_with_nested(Exception(
219  "An exception occurred inside ioda while setting dimension scales on a variable.",
220  ioda_Here()));
221  }
222 }
223 
224 template <>
225 Variable Variable_Base<>::setDimScale(const std::vector<Named_Variable>& vdims) {
226  try {
227  for (unsigned int i = 0; i < vdims.size(); ++i) attachDimensionScale(i, vdims[i].var);
228  return Variable{backend_};
229  } catch (...) {
230  std::throw_with_nested(
231  Exception("An exception occurred inside ioda while setting dimension scales on a variable.",
232  ioda_Here()));
233  }
234 
235 }
236 template <>
238  return setDimScale(std::vector<Variable>{dims});
239 }
240 template <>
242  return setDimScale(std::vector<Variable>{dim1, dim2});
243 }
244 template <>
246  const Variable& dim3) {
247  return setDimScale(std::vector<Variable>{dim1, dim2, dim3});
248 }
249 
250 template <>
252  try {
253  if (backend_ == nullptr)
254  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
255  return backend_->isDimensionScale();
256  } catch (...) {
257  std::throw_with_nested(Exception(
258  "An exception occurred inside ioda while checking if a variable is a dimension scale.",
259  ioda_Here()));
260  }
261 }
262 
263 template <>
264 Variable Variable_Base<>::setIsDimensionScale(const std::string& dimensionScaleName) {
265  try {
266  if (backend_ == nullptr)
267  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
268  return backend_->setIsDimensionScale(dimensionScaleName);
269  } catch (...) {
270  std::throw_with_nested(Exception(
271  "An exception occurred inside ioda while making a variable a dimension scale.",
272  ioda_Here()));
273  }
274 }
275 template <>
277  try {
278  if (backend_ == nullptr)
279  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
280  return backend_->getDimensionScaleName(res);
281  } catch (...) {
282  std::throw_with_nested(Exception(
283  "An exception occurred inside ioda while determining the human-readable "
284  "name of a dimension scale.", ioda_Here()));
285  }
286 }
287 template <>
288 bool Variable_Base<>::isDimensionScaleAttached(unsigned int DimensionNumber,
289  const Variable& scale) const {
290  try {
291  if (backend_ == nullptr)
292  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
293  return backend_->isDimensionScaleAttached(DimensionNumber, scale);
294  } catch (...) {
295  std::throw_with_nested(Exception(
296  "An exception occurred inside ioda while determining if a dimension scale is "
297  "attached to a variable at a specified dimension.", ioda_Here())
298  .add("DimensionNumber", DimensionNumber));
299  }
300 }
301 
302 template <>
303 std::vector<std::vector<Named_Variable>> Variable_Base<>::getDimensionScaleMappings(
304  const std::list<Named_Variable>& scalesToQueryAgainst, bool firstOnly) const {
305  try {
306  if (backend_ == nullptr)
307  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
308  return backend_->getDimensionScaleMappings(scalesToQueryAgainst, firstOnly);
309  } catch (...) {
310  std::throw_with_nested(Exception(
311  "An exception occurred inside ioda while determining which scales are attached to "
312  "which dimensions of a variable.", ioda_Here()));
313  }
314 }
315 
316 template <>
317 Variable Variable_Base<>::write(gsl::span<char> data, const Type& in_memory_dataType,
318  const Selection& mem_selection, const Selection& file_selection) {
319  try {
320  if (backend_ == nullptr)
321  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
322  return backend_->write(data, in_memory_dataType, mem_selection, file_selection);
323  } catch (...) {
324  std::throw_with_nested(Exception(
325  "An exception occurred inside ioda while writing data to a variable.", ioda_Here()));
326  }
327 }
328 
329 template <>
330 Variable Variable_Base<>::read(gsl::span<char> data, const Type& in_memory_dataType,
331  const Selection& mem_selection,
332  const Selection& file_selection) const {
333  try {
334  if (backend_ == nullptr)
335  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
336  return backend_->read(data, in_memory_dataType, mem_selection, file_selection);
337  } catch (...) {
338  std::throw_with_nested(Exception(
339  "An exception occurred inside ioda while reading data from a variable.", ioda_Here()));
340  }
341 }
342 
343 template <>
345  try {
346  if (backend_ == nullptr)
347  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
348  return backend_->instantiateSelection(sel);
349  } catch (...) {
350  std::throw_with_nested(Exception(
351  "An exception occurred inside ioda.", ioda_Here()));
352  }
353 }
354 
355 template class Variable_Base<Variable>; // NOLINT: Bad check result
356 
359 
360 std::vector<std::vector<Named_Variable>> Variable_Backend::getDimensionScaleMappings(
361  const std::list<Named_Variable>& scalesToQueryAgainst, bool firstOnly) const {
362  try {
363  auto dims = this->getDimensions();
364  std::vector<std::vector<Named_Variable>> res(gsl::narrow<size_t>(dims.dimensionality));
365  for (unsigned i = 0; i < gsl::narrow<unsigned>(dims.dimensionality); ++i) {
366  for (const auto& s : scalesToQueryAgainst) {
367  if (this->isDimensionScaleAttached(i, s.var)) {
368  res[i].push_back(s);
369  if (firstOnly) break;
370  }
371  }
372  }
373 
374  return res;
375  } catch (...) {
376  std::throw_with_nested(Exception(
377  "An exception occurred inside ioda.", ioda_Here()));
378  }
379 }
380 
382  try {
384 
385  // Get chunking
386  auto chunkinfo = getChunkSizes();
387  if (chunkinfo.size()) {
388  res.chunk = true;
389  res.chunks = chunkinfo;
390  }
391  // Get compression
392  auto gz = getGZIPCompression();
393  if (gz.first) res.compressWithGZIP(gz.second);
394  auto sz = getSZIPCompression();
395  if (std::get<0>(sz)) res.compressWithSZIP(std::get<1>(sz), std::get<2>(sz));
396  // Get fill value
397  res.fillValue_ = getFillValue();
398  // Attributes (optional)
399  if (doAtts) {
400  throw Exception("Unimplemented doAtts option.", ioda_Here());
401  }
402  // Dimensions (optional)
403  if (doDims) {
404  throw Exception("Unimplemented doDims option.", ioda_Here());
405  }
406 
407  return res;
408  } catch (...) {
409  std::throw_with_nested(Exception(
410  "An exception occurred inside ioda while determining creation-time parameters of a "
411  "variable .", ioda_Here()));
412  }
413 }
414 
415 } // namespace detail
416 
417 Variable::~Variable() = default;
419  : detail::Variable_Base<Variable>(nullptr),
420  _py_isA{this},
421  _py_readVector{this},
422  _py_readNPArray{this},
423  _py_writeVector{this},
424  _py_writeNPArray{this},
425  _py_scales{this} {}
426 
427 Variable::Variable(std::shared_ptr<detail::Variable_Backend> b)
428  : detail::Variable_Base<Variable>(b),
429  _py_isA{this},
430  _py_readVector{this},
431  _py_readNPArray{this},
432  _py_writeVector{this},
433  _py_writeNPArray{this},
434  _py_scales{this} {}
435 
436 Variable::Variable(const Variable& r)
437  : Variable_Base{r.backend_},
438  _py_isA{this},
439  _py_readVector{this},
440  _py_readNPArray{this},
441  _py_writeVector{this},
442  _py_writeNPArray{this},
443  _py_scales{this} {}
444 
446  if (this == &r) return *this;
447  backend_ = r.backend_;
448  atts = r.atts;
451 
454 
457  return *this;
458 }
459 } // namespace ioda
IODA's error system.
Interfaces for ioda::Has_Variables and related classes.
Interfaces for ioda::Variable and related classes.
The ioda exception class.
Definition: Exception.h:54
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
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
virtual ~Variable()
detail::python_bindings::VariableReadVector< Variable > _py_readVector
Definition: Variable.h:697
detail::python_bindings::VariableScales< Variable > _py_scales
Definition: Variable.h:703
detail::python_bindings::VariableWriteNPArray< Variable > _py_writeNPArray
Definition: Variable.h:701
detail::python_bindings::VariableWriteVector< Variable > _py_writeVector
Definition: Variable.h:700
detail::python_bindings::VariableIsA< Variable > _py_isA
Definition: Variable.h:695
detail::python_bindings::VariableReadNPArray< Variable > _py_readNPArray
Definition: Variable.h:698
Variable & operator=(const Variable &)
Definition: Variable.cpp:445
Backends implement type providers in conjunction with Attributes, Has_Attributes, Variables and Has_V...
Definition: Type_Provider.h:36
VariableCreationParameters getCreationParameters(bool doAtts=true, bool doDims=true) const override
Default implementation. Customizable by backends for performance.
Definition: Variable.cpp:381
std::vector< std::vector< Named_Variable > > getDimensionScaleMappings(const std::list< Named_Variable > &scalesToQueryAgainst, bool firstOnly=true) const override
Default, trivial implementation. Customizable by backends for performance.
Definition: Variable.cpp:360
virtual Variable attachDimensionScale(unsigned int DimensionNumber, const Variable &scale)
Attach a dimension scale to this Variable.
Definition: Variable.cpp:184
Variable setDimScale(const std::vector< Variable > &dims)
Set dimensions (convenience function to several invocations of attachDimensionScale).
Definition: Variable.cpp:212
virtual Type getType() const
Get type.
Definition: Variable.cpp:49
virtual FillValueData_t getFillValue() const
Retrieve the fill value.
Definition: Variable.cpp:111
virtual Variable setIsDimensionScale(const std::string &dimensionScaleName)
Designate this table as a dimension scale.
Definition: Variable.cpp:264
virtual std::vector< Dimensions_t > getChunkSizes() const
Retrieve the chunking options for the Variable.
Definition: Variable.cpp:123
virtual std::pair< bool, int > getGZIPCompression() const
Retrieve the GZIP compression options for the Variable.
Definition: Variable.cpp:136
std::shared_ptr< Variable_Backend > backend_
Using an opaque object to implement the backend.
Definition: Variable.h:56
bool isA() const
Convenience function to check a Variable's storage type.
Definition: Variable.h:99
Has_Attributes atts
Attributes.
Definition: Variable.h:71
virtual bool isDimensionScaleAttached(unsigned int DimensionNumber, const Variable &scale) const
Is a dimension scale attached to this Variable in a certain position?
Definition: Variable.cpp:288
virtual bool hasFillValue() const
Check if a variable has a fill value set.
Definition: Variable.cpp:98
virtual bool isDimensionScale() const
Is this Variable used as a dimension scale?
Definition: Variable.cpp:251
virtual Variable detachDimensionScale(unsigned int DimensionNumber, const Variable &scale)
Detach a dimension scale.
Definition: Variable.cpp:198
BasicTypes getBasicType() const
Convenience function to query type.
virtual Dimensions getDimensions() const
Definition: Variable.cpp:160
virtual std::tuple< bool, unsigned, unsigned > getSZIPCompression() const
Retrieve the SZIP compression options for the Variable.
Definition: Variable.cpp:148
virtual std::vector< std::vector< Named_Variable > > getDimensionScaleMappings(const std::list< Named_Variable > &scalesToQueryAgainst, bool firstOnly=true) const
Which dimensions are attached at which positions? This function may offer improved performance on som...
Definition: Variable.cpp:303
virtual VariableCreationParameters getCreationParameters(bool doAtts=true, bool doDims=true) const
Convenience function to get fill value, attributes, chunk sizes, and compression in a collective call...
Definition: Variable.cpp:85
virtual detail::Type_Provider * getTypeProvider() const
Query the backend and get the type provider.
Definition: Variable.cpp:37
std::string getDimensionScaleName() const
Get the name of this Variable's defined dimension scale.
Definition: Variable.h:217
virtual Variable read(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all) const
Read the Variable - as char array. Ordering is row-major.
Definition: Variable.cpp:330
Variable_Base(std::shared_ptr< Variable_Backend >)
Definition: Variable.cpp:15
virtual Variable write(gsl::span< char > data, const Type &in_memory_dataType, const Selection &mem_selection=Selection::all, const Selection &file_selection=Selection::all)
The fundamental write function. Backends overload this function to implement all write operations.
Definition: Variable.cpp:317
std::shared_ptr< Variable_Backend > get() const
Gets a handle to the underlying object that implements the backend functionality.
Definition: Variable.cpp:21
virtual Variable resize(const std::vector< Dimensions_t > &newDims)
Resize the variable.
Definition: Variable.cpp:172
virtual Selections::SelectionBackend_t instantiateSelection(const Selection &sel) const
Convert a selection into its backend representation.
Definition: Variable.cpp:344
std::shared_ptr< InstantiatedSelection > SelectionBackend_t
Definition: Selection.h:35
list newDims
Definition: 05-ObsGroup.py:95
BasicTypes
Definition: Type.h:37
@ undefined_
Internal use only.
#define ioda_Here()
Describes the dimensions of an Attribute or Variable.
Definition: Dimensions.h:22
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57
detail::FillValueData_t fillValue_
Definition: Has_Variables.h:66
std::vector< Dimensions_t > chunks
Manually specify the chunks. Never directly use. Use getChunks(...) instead.
Definition: Has_Variables.h:87
void compressWithSZIP(unsigned PixelsPerBlock=16, unsigned options=4)
bool chunk
Do we chunk this variable? Required for extendible / compressible Variables.
Definition: Has_Variables.h:84
Container used to store and manipulate fill values.
Definition: Fill.h:35