OOPS
ParameterTraitsVariables.cc
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 #include <map>
9 #include <set>
10 #include <string>
11 #include <utility>
12 #include <vector>
13 
15 #include "oops/base/Variables.h"
16 
17 namespace oops {
18 
19 std::vector<std::string> getVariableNamesWithoutChannelSuffix(const Variables &variables) {
20  if (variables.channels().empty())
21  return variables.variables();
22 
23  auto throwException = [&variables] {
24  std::stringstream msg;
25  msg << "The Variables object " << variables << " cannot be stored in a Configuration object: "
26  "not all variables have the same channels";
27  throw std::runtime_error(msg.str());
28  };
29 
30  if (variables.variables().size() % variables.channels().size() != 0)
31  throwException(); // clearly, not all variables have the same channel suffixes
32 
33  typedef std::set<std::string> ChannelSuffixes;
34  ChannelSuffixes expectedChannels;
35  for (int channel : variables.channels())
36  expectedChannels.insert(std::to_string(channel));
37 
38  // Split each variable name into a base name and a channel suffix
39  const char channelSeparator = '_';
40  std::vector<std::string> uniqueBaseNames;
41  std::map<std::string, ChannelSuffixes> channelsPerVariable;
42  for (const std::string& name : variables.variables()) {
43  const std::string::size_type separatorPosition = name.find_last_of(channelSeparator);
44  if (separatorPosition == std::string::npos)
45  throwException(); // no channel suffix
46  std::string baseName = name.substr(0, separatorPosition);
47  std::string channel = name.substr(separatorPosition + 1);
48 
49  std::map<std::string, ChannelSuffixes>::iterator it = channelsPerVariable.find(baseName);
50  if (it == channelsPerVariable.end()) {
51  // It's the first time we're encountering this base name
52  uniqueBaseNames.push_back(baseName);
53  it = channelsPerVariable.insert(std::make_pair(baseName, ChannelSuffixes())).first;
54  }
55  it->second.insert(std::move(channel));
56  }
57 
58  if (std::any_of(channelsPerVariable.begin(), channelsPerVariable.end(),
59  [&expectedChannels]
60  (const std::pair<const std::string, ChannelSuffixes> &nameAndChannels)
61  { return nameAndChannels.second != expectedChannels; }))
62  throwException(); // some variables have different channel suffixes than expected
63 
64  return uniqueBaseNames;
65 }
66 
67 } // namespace oops
const std::vector< std::string > & variables() const
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.