IODA
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 Variables.cpp
11  * \brief Functions for ObsStore Variable and Has_Variables
12  */
13 
14 #include "./Variables.hpp"
15 
16 #include <exception>
17 #include <functional>
18 #include <numeric>
19 
20 #include "./Group.hpp"
21 #include "ioda/Exception.h"
22 
23 namespace ioda {
24 namespace ObsStore {
25 //***************************************************************************
26 // Variable methods
27 //****************************************************************************
28 Variable::Variable(const std::vector<Dimensions_t>& dimensions,
29  const std::vector<Dimensions_t>& max_dimensions, const ObsTypes& dtype,
30  const VarCreateParams& params)
31  : dimensions_(dimensions),
32  max_dimensions_(max_dimensions),
33  dtype_(dtype),
34  dtype_size_(0),
35  var_data_(),
36  is_scale_(false),
37  atts(std::make_shared<Has_Attributes>()),
38  impl_atts(std::make_shared<Has_Attributes>()) {
39  // Get a typed storage object based on dtype
41  dtype_size_ = params.dtype_size;
42 
43  // If have a fill value, save in an attribute. Do this before resizing
44  // because resize() will check for the fill value.
45  if (params.fvdata.set_) {
46  auto fv = impl_atts->create("_fillValue", dtype_, {1});
47  fv->write(params.fill_value, dtype_);
48 
49  this->fvdata_ = params.fvdata;
50  }
51 
52  // Set the size of the variable value
53  this->resize(dimensions_);
54 
55  // Anticipate attaching dimension scales. The dim_scales_ data member won't
56  // be used if this variable is a dimension scale.
57  dim_scales_.assign(dimensions_.size(), nullptr);
58 }
59 
60 std::vector<Dimensions_t> Variable::get_dimensions() const { return dimensions_; }
61 
62 std::vector<Dimensions_t> Variable::get_max_dimensions() const { return max_dimensions_; }
63 
64 void Variable::resize(const std::vector<Dimensions_t>& new_dim_sizes) {
65  // Check new_dim_sizes versus max_dimensions.
66  for (std::size_t i = 0; i < max_dimensions_.size(); ++i) {
67  if (max_dimensions_[i] >= 0) {
68  if (new_dim_sizes[i] > max_dimensions_[i]) {
69  throw Exception("new_dim_sizes exceeds max_dimensions_", ioda_Here())
70  .add("dimension index", i)
71  .add("max_dims[i]", max_dimensions_[i])
72  .add("new_dim_sizes[i]", new_dim_sizes[i]);
73  }
74  }
75  }
76 
77  // Set the dimensions_ data member
78  dimensions_ = new_dim_sizes;
79 
80  // Allow for the total number of elements to change. If there are
81  // addtional elements (total size is growing), then fill those elements
82  // with the variable's fill value (if exists).
83  std::size_t numElements = std::accumulate(new_dim_sizes.begin(), new_dim_sizes.end(), (size_t)1,
84  std::multiplies<std::size_t>());
85 
86  if (impl_atts->exists("_fillValue")) {
87  std::vector<char> fvalue(dtype_size_);
88  gsl::span<char> fillValue(fvalue.data(), dtype_size_);
89  impl_atts->open("_fillValue")->read(fillValue, dtype_);
90  var_data_->resize(numElements, fillValue);
91  } else {
92  var_data_->resize(numElements);
93  }
94 }
95 
96 bool Variable::isOfType(ObsTypes dtype) const { return (dtype == dtype_); }
97 
98 bool Variable::hasFillValue() const { return fvdata_.set_; }
99 
101 
102 void Variable::attachDimensionScale(const std::size_t dim_number,
103  const std::shared_ptr<Variable> scale) {
104  dim_scales_[dim_number] = scale;
105 }
106 
107 void Variable::detachDimensionScale(const std::size_t dim_number,
108  const std::shared_ptr<Variable> scale) {
109  if (dim_scales_[dim_number] == scale) {
110  dim_scales_[dim_number] = nullptr;
111  } else
112  throw Exception("specified scale is not found", ioda_Here()).add("dim_number", dim_number);
113 }
114 
115 bool Variable::isDimensionScale() const { return is_scale_; }
116 
117 void Variable::setIsDimensionScale(const std::string& name) {
118  is_scale_ = true;
119  scale_name_ = name;
120 }
121 
122 void Variable::getDimensionScaleName(std::string& name) const { name = scale_name_; }
123 
124 bool Variable::isDimensionScaleAttached(const std::size_t dim_number,
125  const std::shared_ptr<Variable> scale) const {
126  return (dim_scales_[dim_number] == scale);
127 }
128 
129 std::shared_ptr<Variable> Variable::write(gsl::span<char> data, ObsTypes dtype, Selection& m_select,
130  Selection& f_select) {
131  if (dtype != dtype_)
132  throw Exception("Requested data type not equal to storage datatype", ioda_Here());
133 
134  var_data_->write(data, m_select, f_select);
135  return shared_from_this();
136 }
137 
138 std::shared_ptr<Variable> Variable::read(gsl::span<char> data, ObsTypes dtype, Selection& m_select,
139  Selection& f_select) {
140  if (dtype != dtype_)
141  throw Exception("Requested data type not equal to storage datatype.", ioda_Here());
142 
143  var_data_->read(data, m_select, f_select);
144  return shared_from_this();
145 }
146 
147 //***************************************************************************
148 // Has_Variable methods
149 //****************************************************************************
150 std::shared_ptr<Variable> Has_Variables::create(const std::string& name,
152  const std::vector<Dimensions_t>& dims,
153  const std::vector<Dimensions_t>& max_dims,
154  const VarCreateParams& params) {
155  // If have hiearchical name, then go to the parent group to create the
156  // intermediate groups and create the varible at the group at the end
157  // of the intermediate group chain. Otherwise create the varible in this
158  // this group.
159  std::shared_ptr<Variable> var;
160  std::vector<std::string> splitPaths = splitGroupVar(name);
161  if (splitPaths.size() > 1) {
162  // Have intermediate groups, create variable in "bottom" group
163  std::shared_ptr<Group> parentGroup = parent_group_.lock();
164  std::shared_ptr<Group> group = parentGroup->create(splitPaths[0]);
165  var = group->vars->create(splitPaths[1], dtype, dims, max_dims, params);
166  } else {
167  // No intermediate groups, create variable here
168  var = std::make_shared<Variable>(dims, max_dims, dtype, params);
169  variables_.insert(std::pair<std::string, std::shared_ptr<Variable>>(name, var));
170  }
171  return var;
172 }
173 
174 std::shared_ptr<Variable> Has_Variables::open(const std::string& name) const {
175  std::shared_ptr<Variable> var;
176  std::vector<std::string> splitPaths = splitGroupVar(name);
177  if (splitPaths.size() > 1) {
178  std::shared_ptr<Group> parentGroup = parent_group_.lock();
179  std::shared_ptr<Group> group = parentGroup->open(splitPaths[0]);
180  var = group->vars->open(splitPaths[1]);
181  } else {
182  auto ivar = variables_.find(name);
183  if (ivar == variables_.end())
184  throw Exception("Variable not found.", ioda_Here()).add("name", name);
185  var = ivar->second;
186  }
187  return var;
188 }
189 
190 bool Has_Variables::exists(const std::string& name) const {
191  bool varExists = false;
192  std::vector<std::string> splitPaths = splitGroupVar(name);
193  if (splitPaths.size() > 1) {
194  std::shared_ptr<Group> parentGroup = parent_group_.lock();
195  if (parentGroup->exists(splitPaths[0])) {
196  std::shared_ptr<Group> group = parentGroup->open(splitPaths[0]);
197  varExists = group->vars->exists(splitPaths[1]);
198  }
199  } else {
200  varExists = (variables_.find(name) != variables_.end());
201  }
202  return varExists;
203 }
204 
205 void Has_Variables::remove(const std::string& name) {
206  std::vector<std::string> splitPaths = splitGroupVar(name);
207  if (splitPaths.size() > 1) {
208  std::shared_ptr<Group> parentGroup = parent_group_.lock();
209  std::shared_ptr<Group> group = parentGroup->open(splitPaths[0]);
210  group->vars->remove(splitPaths[1]);
211  } else {
212  variables_.erase(name);
213  }
214 }
215 
216 void Has_Variables::rename(const std::string& oldName, const std::string& newName) {
217  std::vector<std::string> splitPaths = splitGroupVar(oldName);
218  if (splitPaths.size() > 1) {
219  std::shared_ptr<Group> parentGroup = parent_group_.lock();
220  std::shared_ptr<Group> group = parentGroup->open(splitPaths[0]);
221  group->vars->rename(splitPaths[1], newName);
222  } else {
223  std::shared_ptr<Variable> var = open(oldName);
224  variables_.erase(oldName);
225  variables_.insert(std::pair<std::string, std::shared_ptr<Variable>>(newName, var));
226  }
227 }
228 
229 std::vector<std::string> Has_Variables::list() const {
230  std::vector<std::string> varList;
231  for (auto ivar = variables_.begin(); ivar != variables_.end(); ++ivar) {
232  varList.push_back(ivar->first);
233  }
234  return varList;
235 }
236 
237 void Has_Variables::setParentGroup(const std::shared_ptr<Group>& parentGroup) {
238  parent_group_ = parentGroup;
239 }
240 
241 // private methods
242 std::vector<std::string> Has_Variables::splitGroupVar(const std::string& path) {
243  std::vector<std::string> splitPath;
244  auto pos = path.find_last_of('/');
245  splitPath.push_back(path.substr(0, pos));
246  if (pos != std::string::npos) {
247  splitPath.push_back(path.substr(pos + 1));
248  }
249  return splitPath;
250 }
251 } // namespace ObsStore
252 } // namespace ioda
253 
254 /// @}
IODA's error system.
Functions for ObsStore Group and Has_Groups.
Functions for ObsStore Variable and Has_Variables.
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
std::vector< std::string > list() const
returns a list of names of the variables in the container
Definition: Variables.cpp:229
static std::vector< std::string > splitGroupVar(const std::string &path)
split a path into groups and variable pieces
Definition: Variables.cpp:242
bool exists(const std::string &name) const
returns true if variable exists in the container
Definition: Variables.cpp:190
std::shared_ptr< Variable > create(const std::string &name, const ioda::ObsStore::ObsTypes &dtype, const std::vector< Dimensions_t > &dims, const std::vector< Dimensions_t > &max_dims, const VarCreateParams &params)
create a new variable
Definition: Variables.cpp:150
std::shared_ptr< Variable > open(const std::string &name) const
open an existing variable (throws exception if not found)
Definition: Variables.cpp:174
void remove(const std::string &name)
remove variable
Definition: Variables.cpp:205
std::map< std::string, std::shared_ptr< Variable > > variables_
container of variables
Definition: Variables.hpp:153
void rename(const std::string &oldName, const std::string &newName)
rename variable
Definition: Variables.cpp:216
std::weak_ptr< Group > parent_group_
pointer to parent group
Definition: Variables.hpp:156
void setParentGroup(const std::shared_ptr< Group > &parentGroup)
set parent group pointer
Definition: Variables.cpp:237
detail::FillValueData_t fvdata_
Fill value information.
Definition: Variables.hpp:61
std::shared_ptr< Has_Attributes > impl_atts
implementation-specific attribute storage. Fill values, chunking, compression settings,...
Definition: Variables.hpp:87
std::string scale_name_
alias for this variable when it is serving as a dimension scale
Definition: Variables.hpp:73
void detachDimensionScale(const std::size_t dim_number, const std::shared_ptr< Variable > scale)
detach another variable that is servingas a scale (coordinate values)
Definition: Variables.cpp:107
detail::FillValueData_t getFillValue() const
Get the fill value.
Definition: Variables.cpp:100
void attachDimensionScale(const std::size_t dim_number, const std::shared_ptr< Variable > scale)
attach another variable to serve as a scale (holds coordinate values)
Definition: Variables.cpp:102
bool hasFillValue() const
Is there an associated fill value?
Definition: Variables.cpp:98
bool isDimensionScaleAttached(const std::size_t dim_number, const std::shared_ptr< Variable > scale) const
return true if the scale is attached to this variable
Definition: Variables.cpp:124
std::size_t dtype_size_
ObsStore data type.
Definition: Variables.hpp:58
bool isOfType(ObsTypes dtype) const
returns true if requested type matches stored type
Definition: Variables.cpp:96
void resize(const std::vector< Dimensions_t > &new_dim_sizes)
resizes dimensions (but cannot change dimensions themselves)
Definition: Variables.cpp:64
std::vector< Dimensions_t > get_dimensions() const
returns dimension sizes
Definition: Variables.cpp:60
std::vector< Dimensions_t > dimensions_
dimension sizes (length is rank of dimensions)
Definition: Variables.hpp:52
std::shared_ptr< Variable > write(gsl::span< char > data, ObsTypes dtype, Selection &m_select, Selection &f_select)
transfer data to variable storage
Definition: Variables.cpp:129
std::pair< ObsTypes, size_t > dtype() const
returns the data type.
Definition: Variables.hpp:99
void setIsDimensionScale(const std::string &name)
set this variable as a dimension scale
Definition: Variables.cpp:117
bool isDimensionScale() const
returns true if this is being used as a scale for another variable
Definition: Variables.cpp:115
std::vector< std::shared_ptr< Variable > > dim_scales_
pointers to associated dimension scales
Definition: Variables.hpp:67
std::vector< Dimensions_t > max_dimensions_
maximum dimension sizes (for resizing)
Definition: Variables.hpp:54
std::unique_ptr< VarAttrStore_Base > var_data_
container for variable data values
Definition: Variables.hpp:64
void getDimensionScaleName(std::string &name) const
get the dimension scale name
Definition: Variables.cpp:122
std::shared_ptr< Variable > read(gsl::span< char > data, ObsTypes dtype, Selection &m_select, Selection &f_select)
transfer data from variable storage
Definition: Variables.cpp:138
std::vector< Dimensions_t > get_max_dimensions() const
returns maximum dimension sizes
Definition: Variables.cpp:62
ObsTypes dtype_
ObsStore data type.
Definition: Variables.hpp:56
bool is_scale_
true if this variable is a dimension scale
Definition: Variables.hpp:70
IODA_DL std::vector< std::string > splitPaths(const std::string &p)
Split a string based on occurances of the '/' character.
Definition: StringFuncs.cpp:14
@ ObsStore
ObsStore in-memory.
VarAttrStore_Base * createVarAttrStore(ObsTypes dtype)
factory style function to create a new templated object
ObsTypes
ObsStore data type markers.
Definition: Types.hpp:30
#define ioda_Here()
parameters for creating a new variable
Definition: Variables.hpp:38
detail::FillValueData_t fvdata
Definition: Variables.hpp:44
Container used to store and manipulate fill values.
Definition: Fill.h:35