OOPS
test/interface/VariableChange.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2018-2021 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 #ifndef TEST_INTERFACE_VARIABLECHANGE_H_
9 #define TEST_INTERFACE_VARIABLECHANGE_H_
10 
11 #include <cmath>
12 #include <iostream>
13 #include <memory>
14 #include <string>
15 #include <vector>
16 
17 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
18 
19 #include <boost/noncopyable.hpp>
20 
21 #include "eckit/config/Configuration.h"
22 #include "eckit/testing/Test.h"
23 #include "oops/base/Geometry.h"
24 #include "oops/base/State.h"
27 #include "oops/mpi/mpi.h"
28 #include "oops/runs/Test.h"
29 #include "oops/util/Expect.h"
30 #include "oops/util/Logger.h"
31 #include "test/TestEnvironment.h"
32 
33 namespace test {
34 
35 // -------------------------------------------------------------------------------------------------
36 
37 template <typename MODEL> class VariableChangeFixture : private boost::noncopyable {
40 
41  public:
42  static std::vector<eckit::LocalConfiguration> & confs() {return getInstance().confs_;}
43  static const Geometry_ & resol() {return *getInstance().resol_;}
44  static void reset() {
45  getInstance().resol_.reset();
46  }
47 
48  private:
50  static VariableChangeFixture<MODEL> theVariableChangeFixture;
51  return theVariableChangeFixture;
52  }
53 
55  oops::instantiateVariableChangeFactory<MODEL>();
56 
57  // Geometry for the test
58  const eckit::LocalConfiguration resolConfig(TestEnvironment::config(), "geometry");
59  resol_.reset(new Geometry_(resolConfig, oops::mpi::world()));
60 
61  // Configuration (list of all variable changes)
62  TestEnvironment::config().get("variable change tests", confs_);
63  }
64 
65  ~VariableChangeFixture<MODEL>() {}
66 
67  std::vector<eckit::LocalConfiguration> confs_;
68  std::unique_ptr<const Geometry_> resol_;
69 };
70 
71 // -------------------------------------------------------------------------------------------------
72 
73 template <typename MODEL> void testVariableChangeInverse() {
74  typedef VariableChangeFixture<MODEL> Test_;
75  typedef oops::State<MODEL> State_;
76  typedef oops::VariableChange<MODEL> VariableChange_;
77 
78  // Loop over all variable changes
79  for (std::size_t jj = 0; jj < Test_::confs().size(); ++jj) {
80  // Construct variable change
81  VariableChange_ changevar(Test_::resol(), Test_::confs()[jj]);
82 
83  oops::Log::test() << "Testing VariableChange: " << changevar << std::endl;
84  // User specified tolerance for pass/fail
85  const double tol = Test_::confs()[jj].getDouble("tolerance inverse");
86 
87  // Create states with input and output variables
88  const eckit::LocalConfiguration initialConfig(Test_::confs()[jj], "state");
89  State_ xx(Test_::resol(), initialConfig);
90 
91  const double xxnorm_ref = xx.norm();
92 
93  // Order, inverse first or not (default)
94  // Note: switch input and output variables in configuration if true
95  const bool inverseFirst = Test_::confs()[jj].getBool("inverse first", false);
96 
97  // Convert from input to output variables and back (or vice versa)
98  if (inverseFirst) {
99  oops::Variables varin(Test_::confs()[jj], "input variables");
100  State_ xin(Test_::resol(), varin, xx.validTime());
101  changevar.changeVarInverse(xx, xin);
102 // xx.zero(); Test for GEOS fails if uncommented
103  changevar.changeVar(xin, xx);
104  } else {
105  oops::Variables varout(Test_::confs()[jj], "output variables");
106  State_ xout(Test_::resol(), varout, xx.validTime());
107  changevar.changeVar(xx, xout);
108 // xx.zero(); Test for GEOS fails if uncommented
109  changevar.changeVarInverse(xout, xx);
110  }
111 
112  // Compute norms of the result and reference
113  const double xxnorm_tst = xx.norm();
114 
115  // Print the input and final state
116  oops::Log::test() << "<xin>, <K^{-1}[K(xin)]>, (<xin>-<K^{-1}[K(xin)]<xin>)/>=" << xxnorm_ref <<
117  " " << xxnorm_tst << " " << (xxnorm_ref - xxnorm_tst)/xxnorm_ref <<std::endl;
118 
119  // Is result similar to the reference
120  EXPECT(oops::is_close(xxnorm_tst, xxnorm_ref, tol));
121  }
122 }
123 
124 // -------------------------------------------------------------------------------------------------
125 
126 template <typename MODEL> void testVariableChangeParametersWrapperValidName() {
127  typedef VariableChangeFixture<MODEL> Test_;
128  for (const eckit::Configuration &config : Test_::confs()) {
130  EXPECT_NO_THROW(parameters.validateAndDeserialize(config));
131  }
132 }
133 
134 // -------------------------------------------------------------------------------------------------
135 
136 template <typename MODEL> void testVariableChangeParametersWrapperInvalidName() {
137  eckit::LocalConfiguration config;
138  config.set("variable change", "###INVALID###");
140  if (oops::Parameters::isValidationSupported())
141  EXPECT_THROWS_MSG(parameters.validate(config), "unrecognized enum value");
142  EXPECT_THROWS_MSG(parameters.deserialize(config),
143  "does not exist in VariableChangeFactory");
144 }
145 
146 // -------------------------------------------------------------------------------------------------
147 
148 template <typename MODEL> void testVariableChangeFactoryGetMakerNames() {
149  typedef VariableChangeFixture<MODEL> Test_;
150  const std::vector<std::string> registeredNames =
152  for (const eckit::Configuration &config : Test_::confs()) {
153  const std::string validName = config.getString("variable change");
154  const bool found = std::find(registeredNames.begin(), registeredNames.end(), validName) !=
155  registeredNames.end();
156  EXPECT(found);
157  }
158 }
159 
160 // -------------------------------------------------------------------------------------------------
161 
162 template <typename MODEL> class VariableChange : public oops::Test {
163  public:
166  private:
167  std::string testid() const override {return "test::VariableChange<" + MODEL::name() + ">";}
168 
169  void register_tests() const override {
170  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
171 
172  ts.emplace_back(CASE("interface/VariableChange/testVariableChangeInverse")
173  { testVariableChangeInverse<MODEL>(); });
174  ts.emplace_back(CASE("interface/VariableChange/"
175  "testVariableChangeParametersWrapperValidName")
176  { testVariableChangeParametersWrapperValidName<MODEL>(); });
177  ts.emplace_back(CASE("interface/VariableChange/"
178  "testVariableChangeParametersWrapperInvalidName")
179  { testVariableChangeParametersWrapperInvalidName<MODEL>(); });
180  ts.emplace_back(CASE("interface/VariableChange/"
181  "testVariableChangeFactoryGetMakerNames")
182  { testVariableChangeFactoryGetMakerNames<MODEL>(); });
183  }
184 
185  void clear() const override {}
186 };
187 
188 // -------------------------------------------------------------------------------------------------
189 
190 } // namespace test
191 
192 #endif // TEST_INTERFACE_VARIABLECHANGE_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
State class used in oops; subclass of interface class interface::State.
static std::vector< std::string > getMakerNames()
Return the names of all variable changes that can be created by one of the registered makers.
Encapsulates the nonlinear variable change Note: to see methods that need to be implemented in the im...
Contains a polymorphic parameter holding an instance of a subclass of VariableChangeParametersBase.
static const eckit::Configuration & config()
static std::vector< eckit::LocalConfiguration > & confs()
std::vector< eckit::LocalConfiguration > confs_
std::unique_ptr< const Geometry_ > resol_
static VariableChangeFixture< MODEL > & getInstance()
void register_tests() const override
std::string testid() const override
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
Definition: oops/mpi/mpi.cc:84
void testVariableChangeParametersWrapperValidName()
void testVariableChangeFactoryGetMakerNames()
void testVariableChangeInverse()
void testVariableChangeParametersWrapperInvalidName()
CASE("test_linearmodelparameterswrapper_valid_name")