IODA
ObsGroup.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020 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 #include "ioda/ObsGroup.h"
8 
9 #include "ioda/Engines/HH.h"
10 #include "ioda/Exception.h"
11 #include "ioda/Layout.h"
12 
13 namespace ioda {
15 
16 ObsGroup::~ObsGroup() = default;
17 
18 ObsGroup::ObsGroup() = default;
19 
20 ObsGroup::ObsGroup(Group g, std::shared_ptr<const detail::DataLayoutPolicy> layout) : Group(g) {
21  try {
22  if (layout)
23  this->setLayout(layout);
24  else
25  this->setLayout(
27  } catch (...) {
28  std::throw_with_nested(Exception(
29  "An exception occurred inside ioda while constructing an ObsGroup.", ioda_Here()));
30  }
31 }
32 
33 void ObsGroup::setLayout(std::shared_ptr<const detail::DataLayoutPolicy> policy) {
34  try {
35  layout_ = policy;
36  this->vars.setLayout(policy);
37  } catch (...) {
38  std::throw_with_nested(Exception(
39  "An exception occurred inside ioda while setting a layout policy.", ioda_Here()));
40  }
41 }
42 
43 void ObsGroup::setup(const NewDimensionScales_t& fundamentalDims,
44  std::shared_ptr<const detail::DataLayoutPolicy>) {
45  try {
46  layout_->initializeStructure(*this);
47 
48  for (const auto& f : fundamentalDims) {
50  p.chunk = true;
53  out = {f->chunkingSize_};
54  return true;
55  };
56  p.atts.add("suggested_chunk_dim", f->chunkingSize_);
57 
58  Type t = (f->dataTypeKnown_.isValid())
59  ? f->dataTypeKnown_
60  : vars.getTypeProvider()->makeFundamentalType(f->dataType_);
61 
62  auto newvar = vars.create(f->name_, t, {f->size_}, {f->maxSize_}, p);
63  newvar.setIsDimensionScale(f->name_);
64  f->writeInitialData(newvar);
65  }
66  } catch (...) {
67  std::throw_with_nested(Exception(
68  "An exception occurred inside ioda while building a new ObsGroup.", ioda_Here()));
69  }
70 }
71 
72 ObsGroup ObsGroup::generate(Group& emptyGroup, const NewDimensionScales_t& fundamentalDims,
73  std::shared_ptr<const detail::DataLayoutPolicy> layout) {
74  try {
75  ObsGroup res{emptyGroup, layout};
76  res.setup(fundamentalDims, layout);
77 
78  return res;
79  } catch (...) {
80  std::throw_with_nested(Exception(
81  "An exception occurred inside ioda while building a new ObsGroup.", ioda_Here()));
82  }
83 }
84 
85 void ObsGroup::resize(const std::vector<std::pair<Variable, ioda::Dimensions_t>>& newDims) {
86  try {
87  // Resize the dimension variables
88  for (std::size_t i = 0; i < newDims.size(); ++i) {
89  Variable var = newDims[i].first;
90  var.resize({newDims[i].second});
91  }
92  // Recursively traverse group structure and resize all variables using
93  // the given dimensions.
94  resizeVars(*this, newDims);
95  } catch (...) {
96  std::throw_with_nested(Exception(
97  "An exception occurred inside ioda while resizing an ObsGroup.", ioda_Here()));
98  }
99 }
100 
101 void ObsGroup::resizeVars(Group& g,
102  const std::vector<std::pair<Variable, ioda::Dimensions_t>>& newDims)
103 {
104  try {
105  // Visit the variables in this group, and resize according to
106  // what's in the newDims vector.
107  auto groupVars = g.listObjects(ObjectType::Variable, true)[ObjectType::Variable];
108  for (std::size_t i = 0; i < groupVars.size(); ++i) {
109  // Want to create new sizes for the variable dimesions that
110  // consider both the current dimension sizes and the new
111  // dimension sizes in newDims. For each dimension position,
112  // if a variable in newDims is a Dimension Scale for that position
113  // use the new size in newDims, otherwise use the current size.
114  //
115  // Do not attempt to resize Dimension Scale variables. Those have
116  // been taken care of in the ObsGroup::resize function. Plus,
117  // Dimension Scale variables cannot have Dimension Scales attached
118  // to them, so the isDimensionScaleAttached call will throw an error
119  // if attempted to be run on a Dimension Scale variable.
120  Variable var = g.vars.open(groupVars[i]);
121  if (!var.isDimensionScale()) {
122  std::vector<Dimensions_t> varDims = var.getDimensions().dimsCur;
123  std::vector<Dimensions_t> varNewDims(varDims);
124  for (std::size_t idim = 0; idim < varDims.size(); ++idim) {
125  for (std::size_t inewdim = 0; inewdim < newDims.size(); ++inewdim) {
126  if (var.isDimensionScaleAttached(gsl::narrow<unsigned>(idim), newDims[inewdim].first)) {
127  varNewDims[idim] = newDims[inewdim].second;
128  }
129  }
130  }
131  var.resize(varNewDims);
132  }
133  }
134  } catch (...) {
135  std::throw_with_nested(Exception(
136  "An exception occurred inside ioda while resizing an ObsGroup.", ioda_Here()));
137  }
138 }
139 } // namespace ioda
IODA's error system.
HDF5 engine.
Contains definitions for how data are arranged in ioda internally.
Interfaces for ioda::ObsGroup and related classes.
The ioda exception class.
Definition: Exception.h:54
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
An ObsGroup is a specialization of a ioda::Group. It provides convenience functions and guarantees th...
Definition: ObsGroup.h:32
void setup(const NewDimensionScales_t &fundamentalDims, std::shared_ptr< const detail::DataLayoutPolicy > layout)
Create ObsGroup objects.
Definition: ObsGroup.cpp:43
std::shared_ptr< const detail::DataLayoutPolicy > layout_
Mapping policy.
Definition: ObsGroup.h:39
void setLayout(std::shared_ptr< const detail::DataLayoutPolicy >)
Set the mapping policy to determine the Layout of Variables stored under this Group.
Definition: ObsGroup.cpp:33
virtual ~ObsGroup()
Represents the "type" (i.e. integer, string, float) of a piece of data.
Definition: Type.h:123
Variables store data!
Definition: Variable.h:680
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.
static std::shared_ptr< const DataLayoutPolicy > generate(const std::string &polid="")
Factory generator.
Definition: Layout.cpp:28
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
virtual 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())
Create a Variable without setting its data.
virtual void setLayout(std::shared_ptr< const detail::DataLayoutPolicy >)
virtual Type_Provider * getTypeProvider() const
Query the backend and get the type provider.
virtual Type makeFundamentalType(std::type_index type) const
Make a basic object type, like a double, a float, or a char.
virtual Variable setIsDimensionScale(const std::string &dimensionScaleName)
Designate this table as a dimension scale.
Definition: Variable.cpp:264
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 isDimensionScale() const
Is this Variable used as a dimension scale?
Definition: Variable.cpp:251
virtual Dimensions getDimensions() const
Definition: Variable.cpp:160
virtual Variable resize(const std::vector< Dimensions_t > &newDims)
Resize the variable.
Definition: Variable.cpp:172
std::vector< Dimensions_t > VecDimensions_t
Definition: Selection.h:50
list newDims
Definition: 05-ObsGroup.py:95
std::vector< std::shared_ptr< NewDimensionScale_Base > > NewDimensionScales_t
#define ioda_Here()
std::vector< Dimensions_t > dimsCur
The dimensions of the data.
Definition: Dimensions.h:23
Used to specify Variable creation-time properties.
Definition: Has_Variables.h:57
Attribute_Creator_Store atts
Set any initial attributes here.
std::function< bool(const std::vector< Dimensions_t > &, std::vector< Dimensions_t > &)> fChunkingStrategy
Set variable chunking strategy. Used only if chunk == true and chunks.size() == 0.
Definition: Has_Variables.h:90
bool chunk
Do we chunk this variable? Required for extendible / compressible Variables.
Definition: Has_Variables.h:84