IODA
Layout_ObsGroup_ODB.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2021 Met Office UK
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 /// \file Layout_ObsGroup_ODB.cpp
8 /// \brief Contains implementations for how ODB data are arranged in ioda internally.
9 
11 
13 
14 #include "ioda/Group.h"
15 #include "ioda/Layout.h"
16 #include "ioda/Misc/StringFuncs.h"
17 #include "ioda/defs.h"
18 
19 #include "boost/none_t.hpp"
20 #include "boost/optional.hpp"
21 #include "eckit/config/Configuration.h"
22 #include "eckit/config/LocalConfiguration.h"
23 #include "eckit/config/YAMLConfiguration.h"
24 #include "eckit/exception/Exceptions.h"
25 #include "eckit/filesystem/LocalPathName.h"
26 #include "eckit/filesystem/PathName.h"
27 #include "oops/util/parameters/Parameters.h"
28 
29 #include <exception>
30 #include <string>
31 #include <unordered_map>
32 #include <utility>
33 #include <vector>
34 
35 namespace ioda {
36 namespace detail {
38 
40  const std::string &fileMappingName, const std::vector<std::string> &nonODBVariables)
41  : mappingParams_(new ODBLayoutParameters())
42 {
43  parseMappingFile(fileMappingName);
44  for (auto const &str : nonODBVariables) {
46  }
47 }
48 
50  const std::string &method) {
51  if (method != "concat") {
52  throw eckit::MethodNotYetImplemented("Concatenation is the only supported merge method.");
53  }
55 }
56 
57 void DataLayoutPolicy_ObsGroup_ODB::parseMappingFile(const std::string &nameMapFile) {
58  eckit::PathName yamlPath = nameMapFile;
59  eckit::YAMLConfiguration conf(yamlPath);
60  eckit::LocalConfiguration ioda(conf, "ioda");
61  mappingParams_->validateAndDeserialize(ioda);
62  if (mappingParams_->variables.value() != boost::none)
64  if (mappingParams_->complementaryVariables.value() != boost::none)
66 }
67 
69 {
70  for (VariableParameters const& variable : *(mappingParams_->variables.value())) {
71  if (variable.unit.value())
72  Mapping[variable.source] = {variable.name, {true, *(variable.unit.value())}};
73  else
74  Mapping[variable.source] = {variable.name, {false, ""}};
75  //Mapping[variable.source] = {variable.name, variable.unit};
76  }
77 }
78 
79 /// Add an unchanged variable to the mapping. Used to ensure that
80 /// all of the fundamental variables do not falsely throw an exception.
82  if (isComplementary(str) || isMapped(str) || isMapOutput(str)) {
83  throw Exception("Attempting to re-add existing variable to mapping", ioda_Here());
84  }
85  Mapping[str] = {str, {false, ""}};
86 }
87 
89 {
90  for (ComplementaryVariablesParameters const& variable :
91  *(mappingParams_->complementaryVariables.value())) {
92  if (variable.outputVariableDataType.value() != "string") {
93  throw eckit::MethodNotYetImplemented("YAML mapping file: the output variable "
94  "data type for a derived variable is not "
95  "'string'");
96  }
97  MergeMethod mergeMethod = parseMergeMethod(variable.mergeMethod);
98  if (std::find(variable.inputNames.value().begin(), variable.inputNames.value().end(),
99  variable.outputName.value()) != variable.inputNames.value().end()) {
100  throw eckit::ReadError(std::string("YAML mapping file has a complementary variable name") +
101  std::string("matching a derived variable name."));
102  }
103  std::type_index outputTypeIndex = typeid(std::string);
104 
105  std::shared_ptr<ComplementaryVariableOutputMetadata> sharedOutputMetaData(
107  variable.outputName, outputTypeIndex, mergeMethod, 0});
108  size_t inputIndex = 0;
109  for (const std::string &input : variable.inputNames.value()) {
110  complementaryVariableDataMap[input] = std::make_pair(inputIndex, sharedOutputMetaData);
111  inputIndex++;
112  }
113  sharedOutputMetaData->inputVariableCount = inputIndex;
114  }
115 }
116 
118  // First, set an attribute to indicate that the data are managed
119  // by this data policy.
120  g.atts.add<std::string>("_ioda_layout", std::string("ObsGroup_ODB"));
121  g.atts.add<int32_t>("_ioda_layout_version", ObsGroup_ODB_Layout_Version);
122 
123  // Create the default containers - currently ignored as these are
124  // dynamically created.
125  /*
126  g.create("MetaData");
127  g.create("ObsBias");
128  g.create("ObsError");
129  g.create("ObsValue");
130  g.create("PreQC");
131  */
132 }
133 
134 std::string DataLayoutPolicy_ObsGroup_ODB::doMap(const std::string &str) const {
135  // If the string contains '@', then it needs to be broken into
136  // components and reversed. Additionally, if the string is a key (ODB name) in the mapping file,
137  // it is replaced with its value. All other strings are passed through untouched.
138  std::string mappedStr;
139  auto it = Mapping.find(str);
140  if (it != Mapping.end()) {
141  mappedStr = (it->second).iodaName;
142  } else {
143  mappedStr = str;
144  }
145  mappedStr = convertV1PathToV2Path(mappedStr);
146  return mappedStr;
147 }
148 
149 bool DataLayoutPolicy_ObsGroup_ODB::isComplementary(const std::string &inputVariable) const
150 {
151  return (complementaryVariableDataMap.find(inputVariable) != complementaryVariableDataMap.end());
152 }
153 
154 bool DataLayoutPolicy_ObsGroup_ODB::isMapped(const std::string &input) const {
155  return (Mapping.find(input) != Mapping.end());
156 }
157 
158 bool DataLayoutPolicy_ObsGroup_ODB::isMapOutput(const std::string &output) const
159 {
160  for (const std::pair<std::string, variableStorageInformation> &entry : Mapping) {
161  if (entry.second.iodaName == output)
162  return true;
163  }
164  for (const std::pair<std::string, complementaryVariableMetaData> &entry :
166  if (entry.second.second->outputName == output)
167  return true;
168  }
169  return false;
170 }
171 
172 size_t DataLayoutPolicy_ObsGroup_ODB::getComplementaryPosition(const std::string &input) const
173 {
174  if (!isComplementary(input))
175  throw eckit::ReadError(input + " was not found to be a complementary variable.");
176  return complementaryVariableDataMap.at(input).first;
177 }
178 
179 size_t DataLayoutPolicy_ObsGroup_ODB::getInputsNeeded(const std::string &input) const
180 {
181  if (!isComplementary(input))
182  throw eckit::ReadError(input + std::string(" was not found to be a complementary variable."));
183  return complementaryVariableDataMap.at(input).second->inputVariableCount;
184 }
185 
187  const std::string &input) const
188 {
189  if (!isComplementary(input))
190  throw eckit::ReadError(input + " was not found to be a complementary variable.");
191  return complementaryVariableDataMap.at(input).second->outputName;
192 }
193 
195  const std::string &input) const
196 {
197  if (!isComplementary(input))
198  throw eckit::ReadError(input + " was not found to be a complementary variable.");
199  return complementaryVariableDataMap.at(input).second->outputVariableDataType;
200 }
201 
203  const std::string &input) const
204 {
205  if (!isComplementary(input))
206  throw eckit::ReadError(input + " was not found to be a complementary variable.");
207  return complementaryVariableDataMap.at(input).second->mergeMethod;
208 }
209 
210 std::pair<bool, std::string> DataLayoutPolicy_ObsGroup_ODB::getUnit(
211  const std::string & input) const {
212  if (Mapping.find(input) == Mapping.end()) {
213  throw eckit::ReadError(input + " was not found to to be an ODB source variable.");
214  }
215  return (Mapping.at(input)).inputUnit;
216 }
217 
218 std::string DataLayoutPolicy_ObsGroup_ODB::name() const { return std::string{"ObsGroup ODB v1"}; }
219 
220 } // namespace detail
221 } // namespace ioda
Interfaces for ioda::Group and related classes.
Contains definitions for how data are arranged in ioda internally.
Contains definitions for how ODB data are arranged in ioda internally.
Defines all of the information which should be stored in the YAML mapping file.
The ioda exception class.
Definition: Exception.h:54
DataLayoutPolicy::MergeMethod getMergeMethod(const std::string &) const override
std::string doMap(const std::string &) const override
Map a user-specified Variable path to the correct location.
std::unordered_map< std::string, variableStorageInformation > Mapping
Mapping with ODB equivalents as keys and IODA naming/unit pairs as values.
DataLayoutPolicy::MergeMethod parseMergeMethod(const std::string &)
size_t getInputsNeeded(const std::string &) const override
std::string getOutputNameFromComponent(const std::string &) const override
const int32_t ObsGroup_ODB_Layout_Version
Record versioning information for this layout in the ioda object. Provides forward compatability.
DataLayoutPolicy_ObsGroup_ODB(const std::string &, const std::vector< std::string > &={})
void initializeStructure(Group_Base &) const override
bool isComplementary(const std::string &) const override
Check if the named variable will be a part of a derived variable.
std::string name() const override
A descriptive name for the policy.
std::pair< bool, std::string > getUnit(const std::string &) const override
std::unordered_map< std::string, complementaryVariableMetaData > complementaryVariableDataMap
bool isMapped(const std::string &) const override
Check if the named variable is in the Variables section of the ODB mapping file.
size_t getComplementaryPosition(const std::string &) const override
std::shared_ptr< ODBLayoutParameters > mappingParams_
std::type_index getOutputVariableDataType(const std::string &) const override
bool isMapOutput(const std::string &) const override
Check if the named variable matches one of the output (ioda) names.
@ Concat
Concatenate complementary variables entry-by-entry.
Hidden base class to prevent constructor confusion.
Definition: Group.h:42
Common preprocessor definitions used throughout IODA.
IODA_DL std::string convertV1PathToV2Path(const std::string &path)
Split path into substrings separated by @ characters, then concatenate them in reverse order,...
Definition: StringFuncs.cpp:85
#define ioda_Here()
Metadata for generating a variable in IODA from multiple component variables (same across components)...