IODA
Layout.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 /*! \addtogroup ioda_cxx_layout
9  *
10  * @{
11  * \file Layout.h
12  * \brief Contains definitions for how data are arranged in ioda internally.
13  */
14 
15 #include <memory>
16 #include <string>
17 #include <vector>
18 #include <typeindex>
19 #include <utility>
20 
21 #include "ioda/defs.h"
22 
23 namespace ioda {
24 class Group;
25 
26 namespace detail {
27 class Group_Base;
28 class Group_Backend;
29 class DataLayoutPolicy;
30 
31 /// \brief Policy used for setting locations for Variable access
32 /// \ingroup ioda_cxx_layout
33 /// \note Using std::enable_shared_from_this as part of the pybind11 interface.
34 /// We pass this to ObsGroup as a shared_ptr.
35 /// See https://pybind11.readthedocs.io/en/stable/advanced/smart_ptrs.html#std-shared-ptr
36 class IODA_DL DataLayoutPolicy : public std::enable_shared_from_this<DataLayoutPolicy> {
37 public:
38  virtual ~DataLayoutPolicy();
39  enum class Policies {
40  /// Do no manipulation of the Group / Variable layout.
41  None,
42  /// Transform "Variable@Group" into "Group/Variable". Ensure that
43  /// group names match a few predefined keys.
44  ObsGroup,
45  /// Uses an auxiliary YAML dictionary to convert ODB variable/group naming conventions to
46  /// IODA equivalents. Transform "Variable@Group" into "Group/Variable". Ensure that
47  /// the new group names match a few predefined keys.
48  ObsGroupODB
49  };
50  enum class MergeMethod {
51  /// Concatenate complementary variables entry-by-entry
52  Concat
53  };
54 
55  /// Factory generator.
56  static std::shared_ptr<const DataLayoutPolicy> generate(const std::string &polid = "");
57  /// Factory generator (ODB-specific)
58  /// \p mapPath path to a yaml file that defines how input file variables should be renamed
59  /// upon import to ioda.
60  /// \p nonODBVariables variables such as nlocs which are not declared as either keys or
61  /// values in the mapping yaml file. Needs to be filled out to prevent an exception.
62  static std::shared_ptr<const DataLayoutPolicy> generate(
63  const std::string &polid, const std::string &mapPath,
64  const std::vector<std::string> &nonODBVariables = {});
65  /// Factory generator.
66  static std::shared_ptr<const DataLayoutPolicy> generate(Policies pol = Policies::None);
67  /// Factory generator (ODB-specific)
68  /// \p mapPath path to a yaml file that defines how input file variables should be renamed
69  /// upon import to ioda.
70  /// \p nonODBVariables variables such as nlocs which are not declared as either keys or
71  /// values in the mapping yaml file. Needs to be filled out to prevent an exception.
72  static std::shared_ptr<const DataLayoutPolicy> generate(
73  Policies pol, const std::string &mapPath,
74  const std::vector<std::string> &nonODBVariables = {});
75  /// \internal pybind11 overload casts do not work on some compilers.
76  static inline std::shared_ptr<const DataLayoutPolicy> _py_generate1(const std::string &polid) {
77  return generate(polid);
78  }
79  /// \internal pybind11 overload casts do not work on some compilers.
80  static inline std::shared_ptr<const DataLayoutPolicy> _py_generate2(Policies pol) {
81  return generate(pol);
82  }
83 
84  /// Create default groups and write default attributes upon object
85  /// creation / initialization.
86  virtual void initializeStructure(Group_Base &) const;
87  /// \brief Map a user-specified Variable path to the correct location.
88  /// \details This allows us to keep the frontend paths consistent, and we
89  /// can instead do a path transformation to hide implementation details
90  /// from end users.
91  ///
92  /// The default policy is to pass paths expressed with forward slashes ("MetaData/Longitude")
93  /// unchanged. If we pass paths using '@' notation, then reverse the path component
94  /// (i.e. "TB@ObsValue" becomes "ObsValue/TB").
95  /// \note We can apply these policies in both the frontend and inside of the engines.
96  /// \param inStr is the user-provided string. Ex: "TB@ObsValue" or "MetaData/Latitude",
97  /// or even a fundamental dimension ("ChannelNumber").
98  /// \returns A canonical path, always of the form "Group/Variable". In case of a dimension
99  /// scale, then there is no group name, but for every other Variable, there is a Group name.
100  virtual std::string doMap(const std::string &) const;
101 
102  /// Check if the named variable will be a part of a derived variable
103  virtual bool isComplementary(const std::string &) const;
104 
105  /// Check if the named variable is in the Variables section of the ODB mapping file.
106  virtual bool isMapped(const std::string &) const;
107 
108  /// Check if the named variable matches one of the output (ioda) names.
109  virtual bool isMapOutput(const std::string &) const;
110 
111  /// Returns the position of the input variable in the derived variable.
112  /// \throws If the input is not part of a derived variable.
113  virtual size_t getComplementaryPosition(const std::string &) const;
114 
115  /// Returns the derived variable name to be used in ioda.
116  /// \throws If the input is not part of a derived variable.
117  virtual std::string getOutputNameFromComponent(const std::string &) const;
118 
119  /// Returns the data type of the derived variable.
120  /// \throws If the input is not part of a derived variable.
121  virtual std::type_index getOutputVariableDataType(const std::string &) const;
122 
123  /// Returns the merge method for derived variables.
124  /// \throws If the input is not part of a derived variable.
125  virtual MergeMethod getMergeMethod(const std::string &) const;
126 
127  /// Returns the count of input variables needed.
128  /// \throws If the input is not part of a derived variable.
129  virtual size_t getInputsNeeded(const std::string &) const;
130 
131  /// Returns the variable's unit if it has been specified.
132  /// \returns A pair of (found, unit) indicating if a unit was found and what it is.
133  /// \throws If the input is not listed in Variables section of mapping file.
134  virtual std::pair<bool, std::string> getUnit(const std::string &) const;
135 
136  /// A descriptive name for the policy.
137  virtual std::string name() const;
138 
140 };
141 } // namespace detail
142 } // namespace ioda
143 
144 /// @}
An ObsGroup is a specialization of a ioda::Group. It provides convenience functions and guarantees th...
Definition: ObsGroup.h:32
Policy used for setting locations for Variable access.
Definition: Layout.h:36
static std::shared_ptr< const DataLayoutPolicy > _py_generate1(const std::string &polid)
Definition: Layout.h:76
static std::shared_ptr< const DataLayoutPolicy > _py_generate2(Policies pol)
Definition: Layout.h:80
Hidden base class to prevent constructor confusion.
Definition: Group.h:42
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