IODA
Group.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 /*! \defgroup ioda_cxx_layout Groups and Data Layout
9  * \brief Public API for ioda::Group, ioda::ObsGroup, and data layout policies.
10  * \ingroup ioda_cxx_api
11  *
12  * @{
13  * \file Group.h
14  * \brief Interfaces for ioda::Group and related classes.
15  */
16 
17 #include <gsl/gsl-lite.hpp>
18 #include <map>
19 #include <memory>
20 #include <string>
21 #include <vector>
22 
26 #include "ioda/defs.h"
27 
28 namespace ioda {
29 class Group;
30 
31 namespace Engines {
32 struct Capabilities;
33 } // namespace Engines
34 
35 namespace detail {
36 class Group_Base;
37 class Group_Backend;
38 
39 /// \brief Hidden base class to prevent constructor confusion.
40 /// \ingroup ioda_cxx_layout
41 /// \note enable_shared_from_this is a hint to pybind11.
43  std::shared_ptr<Group_Backend> backend_;
44 
45 protected:
46  Group_Base(std::shared_ptr<Group_Backend>);
47 
48 public:
49  virtual ~Group_Base();
50 
51  /// Get capabilities of the Engine backing this Group
52  virtual ::ioda::Engines::Capabilities getCapabilities() const;
53 
54  /// \brief Get the fill value policy used for Variables within this Group
55  /// \details The backend has to be consulted for this operation. Storage of this policy is
56  /// backend-dependent.
57  virtual FillValuePolicy getFillValuePolicy() const;
58 
59  /// \brief List all one-level child groups in this group
60  /// \details This function exists to provide the same calling semantics as
61  /// vars.list() and atts.list(). It is useful for human exploration of the contents
62  /// of a Group.
63  /// \returns A vector of strings listing the child groups. The strings are unordered.
64  /// \see listObjects if you need to enumerate both Groups and Variables, or
65  /// if you want to do a recursive search. This function is more sophisticated, and
66  /// depending on the backend it may be faster than recursive calls to
67  /// list() and vars.list().
68  std::vector<std::string> list() const;
69  /// Same as list(). Uniform semantics with atts() and vars().
70  inline std::vector<std::string> groups() const { return list(); }
71 
72  /// \brief List all objects (groups + variables) within this group
73  /// \param recurse indicates whether the search should be one-level or recursive. If
74  /// multiple possible paths exist for an object, only one is actually returned.
75  /// \param filter allows you to search for only a certain type of object, such as a
76  /// Group or Variable.
77  /// \returns a map of ObjectTypes. Each mapping contains a vector of strings indicating
78  /// object names. If a filter is provided, then this map only contains the filtered
79  /// object type. Otherwise, the map always has vectors for each element of ObjectType
80  /// (Group, Variable, etc.), although these vectors may be empty if no object of a
81  /// certain type is found.
82  /// \todo This function should list all *distinct* objects. In the future, this will mean
83  /// that 1) hard-linked duplicate objects will only be listed once, 2) soft
84  /// links pointing to the same object will only be traversed once, and
85  /// 3) multiple external links to the same object will only be traversed once.
86  /// If you would want to list *indistinct* objects, then there will be a
87  /// listLinks function.
88  /// \todo Once links are implemented, add an option to auto-resolve
89  /// soft and external links.
90  virtual std::map<ObjectType, std::vector<std::string>> listObjects(ObjectType filter
91  = ObjectType::Ignored,
92  bool recurse = false) const;
93 
94  template <ObjectType objectClass>
95  std::vector<std::string> listObjects(bool recurse = false) const {
96  return listObjects(objectClass, recurse)[objectClass];
97  }
98 
99  /// Does a group exist at the specified path?
100  /// \param name is the group name.
101  /// \returns true if a group does exist.
102  /// \return false if a group does not exist.
103  virtual bool exists(const std::string& name) const;
104 
105  /// \brief Create a group
106  /// \param name is the group name.
107  /// \returns an invalid handle on failure. (i.e. return.isGroup() == false).
108  /// \returns a scoped handle to the group on success.
109  virtual Group create(const std::string& name);
110 
111  /// \brief Open a group
112  /// \returns an invalid handle if an error occurred. (i.e. return.isGroup() == false).
113  /// \returns a scoped handle to the group upon success.
114  /// \note It is possible to have multiple handles opened for the group
115  /// simultaneously.
116  /// \param name is the name of the child group to open.
117  virtual Group open(const std::string& name) const;
118 
119  /// Use this to access the metadata for the group / ObsSpace.
121 
122  /// Use this to access variables
124 };
125 
127 public:
128  virtual ~Group_Backend();
129  /// Default fill value policy is NETCDF4. Overridable on a per-backend basis.
130  FillValuePolicy getFillValuePolicy() const override;
131 
132  /// @}
133 
134 protected:
135  Group_Backend();
136 };
137 
138 } // namespace detail
139 
140 /** \brief Groups are a new implementation of ObsSpaces.
141  * \ingroup ioda_cxx_layout
142  *
143  * \see \ref Groups for a examples.
144  *
145  * A group can be thought of as a folder that contains Variables and Metadata.
146  * A group can also contain child groups, allowing for our ObsSpaces to exist in a
147  * nested tree-like structure, which removes the need for having an ObsSpaceContainer.
148  *
149  * Groups are implemented in several backends, such as the in-memory IODA store, the HDF5 disk
150  * backend, the HDF5 in-memory backend, the ATLAS backend, et cetera. The root Group is mounted
151  * using one of these backends (probably as a File object, which is a special type of Group),
152  * and additional backends may be mounted into the tree structure.
153  *
154  * \throws std::logic_error if the backend handle points to something other than a group.
155  * \throws std::logic_error if the backend handle is invalid.
156  * \throws std::logic_error if any error has occurred.
157  * \author Ryan Honeyager (honeyage@ucar.edu)
158  **/
160 public:
161  Group();
162  Group(std::shared_ptr<detail::Group_Backend>);
163  virtual ~Group();
164 };
165 
166 } // namespace ioda
167 
168 /// @}
Default fill values for ioda files.
Interfaces for ioda::Has_Attributes and related classes.
Interfaces for ioda::Has_Variables and related classes.
Groups are a new implementation of ObsSpaces.
Definition: Group.h:159
virtual ~Group()
This class exists inside of ioda::Group or ioda::Variable and provides the interface to manipulating ...
This class exists inside of ioda::Group and provides the interface to manipulating Variables.
Hidden base class to prevent constructor confusion.
Definition: Group.h:42
Has_Attributes atts
Use this to access the metadata for the group / ObsSpace.
Definition: Group.h:120
Has_Variables vars
Use this to access variables.
Definition: Group.h:123
std::shared_ptr< Group_Backend > backend_
Definition: Group.h:43
std::vector< std::string > groups() const
Same as list(). Uniform semantics with atts() and vars().
Definition: Group.h:70
std::vector< std::string > listObjects(bool recurse=false) const
Definition: Group.h:95
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
IODA_DL Capabilities getCapabilities()
Get capabilities of the ObsStore engine.
Definition: ObsStore.cpp:29
FillValuePolicy
This option describes the default fill values that will be used if the user does not manually specify...
Definition: FillPolicy.h:28