OOPS
ParameterTraitsVariables.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020 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 
8 #ifndef OOPS_BASE_PARAMETERTRAITSVARIABLES_H_
9 #define OOPS_BASE_PARAMETERTRAITSVARIABLES_H_
10 
11 #include <string>
12 #include <vector>
13 
14 #include "oops/base/Variables.h"
15 #include "oops/util/CompositePath.h"
16 #include "oops/util/parameters/ParameterTraits.h"
17 #include "oops/util/stringFunctions.h"
18 
19 /// \file ParameterTraitsVariables.h
20 /// This file needs to be included before any uses of (Required/Optional)Parameter<Variables>.
21 
22 namespace oops {
23 
24 /// \brief Returns the list of "base" variable names (i.e. names without channel suffixes)
25 /// stored in \p variables.
26 ///
27 /// Throws an exception if some variables have different channel suffixes than others.
28 std::vector<std::string> getVariableNamesWithoutChannelSuffix(const Variables &variables);
29 
30 /// \brief Specialization of ParameterTraits needed for serialization and deserialization of
31 /// instances of Variables to/from Parameter objects.
32 template <>
33 struct ParameterTraits<Variables> {
34  static boost::optional<Variables> get(util::CompositePath &path,
35  const eckit::Configuration &config,
36  const std::string& name) {
37  if (config.has(name)) {
38  return Variables(config, name);
39  } else {
40  return boost::none;
41  }
42  }
43 
44  static void set(eckit::LocalConfiguration &config,
45  const std::string &name,
46  const Variables &value) {
47  // Try to remove channel indices from variable names before storing the latter in the
48  // LocalConfiguration object.
49  //
50  // This will work if 'value' hasn't been modified after construction, but if it has, it may
51  // not. For example, a sum of two sets of variables, each with a different set of channels,
52  // cannot be represented by a single Configuration object. In these cases
53  // getVariableNamesWithoutChannelSuffix() will throw an exception.
54  const std::vector<std::string> baseVariableNames = getVariableNamesWithoutChannelSuffix(value);
55 
56  config.set(name, baseVariableNames);
57  const std::vector<int> &channels = value.channels();
58  if (!channels.empty()) {
59  const std::string channelsAsString = util::stringfunctions::join(
60  ",", channels.begin(), channels.end(), [](int n) { return std::to_string(n); });
61  config.set("channels", channelsAsString);
62  }
63  }
64 
65  static ObjectJsonSchema jsonSchema(const std::string &name) {
66  ObjectJsonSchema nameSchema = ParameterTraits<std::vector<std::string>>::jsonSchema("");
67  return ObjectJsonSchema({{name, nameSchema.properties().at("")},
68  {"channels", {{"type", "[\"string\", \"integer\"]"}}}});
69  }
70 };
71 
72 } // namespace oops
73 
74 #endif // OOPS_BASE_PARAMETERTRAITSVARIABLES_H_
const std::vector< int > & channels() const
The namespace for the main oops code.
std::vector< std::string > getVariableNamesWithoutChannelSuffix(const Variables &variables)
Returns the list of "base" variable names (i.e. names without channel suffixes) stored in variables.
static ObjectJsonSchema jsonSchema(const std::string &name)
static void set(eckit::LocalConfiguration &config, const std::string &name, const Variables &value)
static boost::optional< Variables > get(util::CompositePath &path, const eckit::Configuration &config, const std::string &name)