OOPS
oops/base/Variables.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2017-2018 UCAR
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 "oops/base/Variables.h"
9 
10 #include <algorithm>
11 #include <iostream>
12 #include <set>
13 #include <string>
14 #include <vector>
15 
16 #include "eckit/config/Configuration.h"
17 #include "eckit/exception/Exceptions.h"
18 #include "eckit/types/Types.h"
19 
20 #include "oops/util/IntSetParser.h"
21 #include "oops/util/Logger.h"
22 
23 // -----------------------------------------------------------------------------
24 namespace oops {
25 // -----------------------------------------------------------------------------
26 
28  : convention_(""), vars_(0) {
29  Log::trace() << "Variables::Variables" << std::endl;
30 }
31 
32 // -----------------------------------------------------------------------------
33 
34 Variables::Variables(const eckit::Configuration & conf, const std::string & name)
35  : convention_(""), vars_(0), channels_(0) {
36  Log::trace() << "Variables::Variables start " << conf << std::endl;
37  std::vector<std::string> vars;
38  if (!conf.get(name, vars)) {
39  Log::error() << name << " not found in " << conf << std::endl;
40  throw eckit::BadParameter("Undefined variable: '" + name + "'");
41  }
42  // hack to read channels
43  if (conf.has("channels")) {
44  std::string chlist = conf.getString("channels");
45  std::set<int> channels = parseIntSet(chlist);
46  std::copy(channels.begin(), channels.end(), std::back_inserter(channels_));
47  // assuming the same channel subsetting applies to all variables
48  for (size_t jvar = 0; jvar < vars.size(); ++jvar) {
49  for (size_t jch = 0; jch < channels_.size(); ++jch) {
50  vars_.push_back(vars[jvar]+"_"+std::to_string(channels_[jch]));
51  }
52  }
53  } else {
54  vars_ = vars;
55  }
56  Log::trace() << "Variables::Variables done" << std::endl;
57 }
58 
59 // -----------------------------------------------------------------------------
60 
61 Variables::Variables(const std::vector<std::string> & vars, const std::string & conv)
62  : convention_(conv), vars_(vars) {
63  Log::trace() << "Variables::Variables start " << vars << std::endl;
64  Log::trace() << "Variables::Variables done" << std::endl;
65 }
66 
67 // -----------------------------------------------------------------------------
68 
69 Variables::Variables(const std::vector<std::string> & vars, const std::vector<int> & channels)
70  : convention_(""), vars_(0), channels_(channels) {
71  Log::trace() << "Variables::Variables start " << vars << " @ " << channels << std::endl;
72  if (channels.empty()) {
73  vars_ = vars;
74  } else {
75  for (size_t jvar = 0; jvar < vars.size(); ++jvar) {
76  for (size_t jch = 0; jch < channels_.size(); ++jch) {
77  vars_.push_back(vars[jvar]+"_"+std::to_string(channels_[jch]));
78  }
79  }
80  }
81  Log::trace() << "Variables::Variables done" << std::endl;
82 }
83 
84 // -----------------------------------------------------------------------------
85 
87  : convention_(other.convention_), vars_(other.vars_), channels_(other.channels_)
88 {}
89 
90 // -----------------------------------------------------------------------------
91 
93  ASSERT(convention_ == rhs.convention_);
94  vars_.insert(vars_.end(), rhs.vars_.begin(), rhs.vars_.end());
95  // revisit late, should we add channels this way ?
96  channels_.insert(channels_.end(), rhs.channels_.begin(), rhs.channels_.end());
97  // remove duplicated variables and channels
98  std::sort(vars_.begin(), vars_.end());
99  vars_.erase(std::unique(vars_.begin(), vars_.end() ), vars_.end());
100  std::sort(channels_.begin(), channels_.end());
101  channels_.erase(std::unique(channels_.begin(), channels_.end() ), channels_.end());
102  return *this;
103 }
104 
105 // -----------------------------------------------------------------------------
106 
107 bool Variables::operator==(const Variables & rhs) const {
108  return convention_ == rhs.convention_
109  && vars_ == rhs.vars_
110  && channels_ == rhs.channels_;
111 }
112 
113 // -----------------------------------------------------------------------------
114 
115 bool Variables::operator<=(const Variables & rhs) const {
116  ASSERT(convention_ == rhs.convention_);
117  ASSERT(channels_.empty());
118  bool is_in_rhs = true;
119  for (size_t jj = 0; jj < vars_.size(); ++jj) {
120  is_in_rhs = is_in_rhs && rhs.has(vars_[jj]);
121  }
122  return is_in_rhs;
123 }
124 
125 // -----------------------------------------------------------------------------
126 
127 bool Variables::has(const std::string & var) const {
128  bool found = false;
129  for (size_t jj = 0; jj < vars_.size(); ++jj) {
130  found = found || vars_[jj] == var;
131  }
132  return found;
133 }
134 
135 // -----------------------------------------------------------------------------
136 
137 size_t Variables::find(const std::string & var) const {
138  size_t ii = vars_.size();
139  for (size_t jj = 0; jj < vars_.size(); ++jj) {
140  if (vars_[jj] == var) ii = jj;
141  }
142  ASSERT(ii < vars_.size());
143  return ii;
144 }
145 
146 // -----------------------------------------------------------------------------
147 
148 void Variables::push_back(const std::string & vname) {
149  vars_.push_back(vname);
150 }
151 
152 // -----------------------------------------------------------------------------
153 
155 
156 // -----------------------------------------------------------------------------
157 
158 void Variables::print(std::ostream & os) const {
159  os << vars_.size() << " variables: ";
160  for (size_t jj = 0; jj < vars_.size(); ++jj) {
161  if (jj > 0) os << ", ";
162  os << vars_[jj];
163  }
164  if (!convention_.empty()) os << " (" << convention_ << ")";
165 }
166 
167 // -----------------------------------------------------------------------------
168 
169 } // namespace oops
bool operator==(const Variables &) const
bool has(const std::string &) const
std::string convention_
void push_back(const std::string &)
const std::vector< int > & channels() const
void print(std::ostream &) const
std::vector< std::string > vars_
size_t find(const std::string &) const
bool operator<=(const Variables &) const
Variables & operator+=(const Variables &)
std::vector< int > channels_
int error
Definition: compare.py:168
The namespace for the main oops code.