OOPS
test/interface/LinearVariableChange.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
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  * In applying this licence, ECMWF does not waive the privileges and immunities
7  * granted to it by virtue of its status as an intergovernmental organisation nor
8  * does it submit to any jurisdiction.
9  */
10 
11 #ifndef TEST_INTERFACE_LINEARVARIABLECHANGE_H_
12 #define TEST_INTERFACE_LINEARVARIABLECHANGE_H_
13 
14 #include <cmath>
15 #include <iostream>
16 #include <memory>
17 #include <string>
18 #include <vector>
19 
20 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
21 
22 #include <boost/noncopyable.hpp>
23 
24 #include "eckit/config/Configuration.h"
25 #include "eckit/testing/Test.h"
26 #include "oops/base/Geometry.h"
27 #include "oops/base/Increment.h"
29 #include "oops/base/State.h"
30 #include "oops/base/Variables.h"
32 #include "oops/mpi/mpi.h"
33 #include "oops/runs/Test.h"
34 #include "oops/util/DateTime.h"
35 #include "oops/util/dot_product.h"
36 #include "oops/util/Expect.h"
37 #include "oops/util/Logger.h"
38 #include "test/TestEnvironment.h"
39 
40 namespace test {
41 
42 // -----------------------------------------------------------------------------
43 
44 template <typename MODEL> class LinearVariableChangeFixture : private boost::noncopyable {
47  typedef util::DateTime DateTime_;
48 
49  public:
50  static std::vector<eckit::LocalConfiguration> & confs() {return getInstance().confs_;}
51  static const State_ & xx() {return *getInstance().xx_;}
52  static const Geometry_ & resol() {return *getInstance().resol_;}
53  static const DateTime_ & time() {return *getInstance().time_;}
54  static void reset() {
55  getInstance().time_.reset();
56  getInstance().xx_.reset();
57  getInstance().resol_.reset();
58  }
59 
60  private:
62  static LinearVariableChangeFixture<MODEL> theLinearVariableChangeFixture;
63  return theLinearVariableChangeFixture;
64  }
65 
67  oops::instantiateVariableChangeFactory<MODEL>();
68 
69  const eckit::LocalConfiguration resolConfig(TestEnvironment::config(), "geometry");
70  resol_.reset(new Geometry_(resolConfig, oops::mpi::world()));
71 
72  const eckit::LocalConfiguration fgconf(TestEnvironment::config(), "background");
73  xx_.reset(new State_(*resol_, fgconf));
74 
75  time_.reset(new util::DateTime(xx_->validTime()));
76 
77  TestEnvironment::config().get("linear variable change tests", confs_);
78  }
79 
80  ~LinearVariableChangeFixture<MODEL>() {}
81 
82  std::vector<eckit::LocalConfiguration> confs_;
83  std::unique_ptr<const State_ > xx_;
84  std::unique_ptr<const Geometry_> resol_;
85  std::unique_ptr<const util::DateTime> time_;
86 };
87 
88 // -----------------------------------------------------------------------------
89 
90 template <typename MODEL> void testLinearVariableChangeZero() {
92  typedef oops::Increment<MODEL> Increment_;
93  typedef oops::LinearVariableChangeBase<MODEL> LinearVariableChange_;
94  typedef oops::LinearVariableChangeFactory<MODEL> LinearVariableChangeFactory_;
95 
96  for (std::size_t jj = 0; jj < Test_::confs().size(); ++jj) {
97  oops::Variables varin(Test_::confs()[jj], "input variables");
98  oops::Variables varout(Test_::confs()[jj], "output variables");
99 
100  std::unique_ptr<LinearVariableChange_> changevar(LinearVariableChangeFactory_::create(
101  Test_::xx(), Test_::xx(),
102  Test_::resol(), Test_::confs()[jj]));
103  oops::Log::test() << "Testing linear variable change: " << *changevar << std::endl;
104  Increment_ dxinTlIAd(Test_::resol(), varin, Test_::time());
105  Increment_ dxinAdInv(Test_::resol(), varout, Test_::time());
106  Increment_ dxoutTlIAd(Test_::resol(), varout, Test_::time());
107  Increment_ dxoutAdInv(Test_::resol(), varin, Test_::time());
108 
109  // dxinTlIAd = 0, check if K.dxinTlIAd = 0
110  dxinTlIAd.zero();
111  dxoutTlIAd = changevar->multiply(dxinTlIAd);
112  EXPECT(dxoutTlIAd.norm() == 0.0);
113 
114  // dxinAdInv = 0, check if K^T.dxinAdInv = 0
115  dxinAdInv.zero();
116  dxoutAdInv = changevar->multiplyAD(dxinAdInv);
117  EXPECT(dxoutAdInv.norm() == 0.0);
118 
119  const bool testinverse = Test_::confs()[jj].getBool("test inverse", true);
120  if (testinverse)
121  {
122  oops::Log::test() << "Doing zero test for inverse" << std::endl;
123  dxinTlIAd.zero();
124  dxoutTlIAd = changevar->multiplyInverseAD(dxinTlIAd);
125  EXPECT(dxoutTlIAd.norm() == 0.0);
126 
127  dxinAdInv.zero();
128  dxoutAdInv = changevar->multiplyInverse(dxinAdInv);
129  EXPECT(dxoutAdInv.norm() == 0.0);
130  } else {
131  oops::Log::test() << "Not doing zero test for inverse" << std::endl;
132  }
133  }
134 }
135 // -----------------------------------------------------------------------------
136 
137 template <typename MODEL> void testLinearVariableChangeAdjoint() {
139  typedef oops::Increment<MODEL> Increment_;
140  typedef oops::LinearVariableChangeBase<MODEL> LinearVariableChange_;
141  typedef oops::LinearVariableChangeFactory<MODEL> LinearVariableChangeFactory_;
142 
143  for (std::size_t jj = 0; jj < Test_::confs().size(); ++jj) {
144  oops::Variables varin(Test_::confs()[jj], "input variables");
145  oops::Variables varout(Test_::confs()[jj], "output variables");
146 
147  std::unique_ptr<LinearVariableChange_> changevar(LinearVariableChangeFactory_::create(
148  Test_::xx(), Test_::xx(),
149  Test_::resol(), Test_::confs()[jj]));
150 
151  Increment_ dxinAdInv(Test_::resol(), varout, Test_::time());
152  Increment_ dxinTlIAd(Test_::resol(), varin, Test_::time());
153  Increment_ dxoutAdInv(Test_::resol(), varin, Test_::time());
154  Increment_ dxoutTlIAd(Test_::resol(), varout, Test_::time());
155 
156  dxinAdInv.random();
157  dxinTlIAd.random();
158 
159  Increment_ dxinAdInv0(dxinAdInv);
160  Increment_ dxinTlIAd0(dxinTlIAd);
161 
162  dxoutTlIAd = changevar->multiply(dxinTlIAd);
163  dxoutAdInv = changevar->multiplyAD(dxinAdInv);
164 
165  // zz1 = <dxoutTlIAd,dxinAdInv>
166  double zz1 = dot_product(dxoutTlIAd, dxinAdInv0);
167  // zz2 = <dxout,dxoutAdInv>
168  double zz2 = dot_product(dxinTlIAd0, dxoutAdInv);
169 
170  oops::Log::test() << "<dxout,KTdxin>-<Kdxout,dxin>/<dxout,KTdxin>="
171  << (zz1-zz2)/zz1 << std::endl;
172  oops::Log::test() << "<dxout,KTdxin>-<Kdxout,dxin>/<Kdxout,dxin>="
173  << (zz1-zz2)/zz2 << std::endl;
174  const double tol = 1e-10;
175  EXPECT(oops::is_close(zz1, zz2, tol));
176  const bool testinverse = Test_::confs()[jj].getBool("test inverse", true);
177  if (testinverse)
178  {
179  oops::Log::test() << "Doing adjoint test for inverse" << std::endl;
180  dxoutAdInv.zero();
181  dxoutTlIAd.zero();
182  dxinAdInv.random();
183  dxinTlIAd.random();
184  dxinAdInv0 = dxinAdInv;
185  dxinTlIAd0 = dxinTlIAd;
186  dxoutTlIAd = changevar->multiplyInverseAD(dxinTlIAd);
187  dxoutAdInv = changevar->multiplyInverse(dxinAdInv);
188  zz1 = dot_product(dxoutTlIAd, dxinAdInv0);
189  zz2 = dot_product(dxinTlIAd0, dxoutAdInv);
190  oops::Log::test() << "<dxout,KinvTdxin>-<Kinvdxout,dxin>/<dxout,KinvTdxin>="
191  << (zz1-zz2)/zz1 << std::endl;
192  oops::Log::test() << "<dxout,KinvTdxin>-<Kinvdxout,dxin>/<Kinvdxout,dxin>="
193  << (zz1-zz2)/zz2 << std::endl;
194  EXPECT(oops::is_close(zz1, zz2, tol));
195  } else {
196  oops::Log::test() << "Not doing adjoint test for inverse" << std::endl;
197  }
198  }
199 }
200 
201 // -----------------------------------------------------------------------------
202 
203 template <typename MODEL> void testLinearVariableChangeInverse() {
205  typedef oops::Increment<MODEL> Increment_;
206  typedef oops::LinearVariableChangeBase<MODEL> LinearVariableChange_;
207  typedef oops::LinearVariableChangeFactory<MODEL> LinearVariableChangeFactory_;
208 
209  for (std::size_t jj = 0; jj < Test_::confs().size(); ++jj) {
210  oops::Variables varin(Test_::confs()[jj], "input variables");
211  oops::Variables varout(Test_::confs()[jj], "output variables");
212 
213  const double tol = Test_::confs()[jj].getDouble("tolerance inverse");
214 
215  const bool testinverse = Test_::confs()[jj].getBool("test inverse", false);
216  if (testinverse)
217  {
218  oops::Log::test() << "Testing multiplyInverse" << std::endl;
219  std::unique_ptr<LinearVariableChange_> changevar(LinearVariableChangeFactory_::create(
220  Test_::xx(), Test_::xx(),
221  Test_::resol(), Test_::confs()[jj]));
222 
223  Increment_ dxinInv(Test_::resol(), varout, Test_::time());
224  Increment_ dxoutInv(Test_::resol(), varin, Test_::time());
225  Increment_ dxout(Test_::resol(), varout, Test_::time());
226 
227  dxinInv.random();
228 
229  dxoutInv = changevar->multiplyInverse(dxinInv);
230  dxout = changevar->multiply(dxoutInv);
231 
232  const double zz1 = dxinInv.norm();
233  const double zz2 = dxout.norm();
234 
235  oops::Log::test() << "<x>, <KK^{-1}x>=" << zz1 << " " << zz2 << std::endl;
236  oops::Log::test() << "<x>-<KK^{-1}x>=" << zz1-zz2 << std::endl;
237 
238  EXPECT((zz1-zz2) < tol);
239  } else {
240  oops::Log::test() << "multiplyInverse test not executed" << std::endl;
241  EXPECT(1.0 < 2.0);
242  }
243  }
244 }
245 
246 // -----------------------------------------------------------------------------
247 
248 template <typename MODEL> void testLinearVariableChangeParametersWrapperValidName() {
250  for (const eckit::Configuration &config : Test_::confs()) {
252  EXPECT_NO_THROW(parameters.validateAndDeserialize(config));
253  }
254 }
255 
256 // -----------------------------------------------------------------------------
257 
259  eckit::LocalConfiguration config;
260  config.set("variable change", "###INVALID###");
262  if (oops::Parameters::isValidationSupported())
263  EXPECT_THROWS_MSG(parameters.validate(config), "unrecognized enum value");
264  EXPECT_THROWS_MSG(parameters.deserialize(config),
265  "does not exist in LinearVariableChangeFactory");
266 }
267 
268 // -----------------------------------------------------------------------------
269 
270 template <typename MODEL> void testLinearVariableChangeFactoryGetMakerNames() {
272  const std::vector<std::string> registeredNames =
274  for (const eckit::Configuration &config : Test_::confs()) {
275  const std::string validName = config.getString("variable change");
276  const bool found = std::find(registeredNames.begin(), registeredNames.end(), validName) !=
277  registeredNames.end();
278  EXPECT(found);
279  }
280 }
281 
282 // -----------------------------------------------------------------------------
283 
284 template <typename MODEL>
286  public:
289 
290  private:
291  std::string testid() const override {return "test::LinearVariableChange<" + MODEL::name() + ">";}
292 
293  void register_tests() const override {
294  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
295 
296  ts.emplace_back(CASE("interface/LinearVariableChange/testLinearVariableChangeZero")
297  { testLinearVariableChangeZero<MODEL>(); });
298  ts.emplace_back(CASE("interface/LinearVariableChange/testLinearVariableChangeAdjoint")
299  { testLinearVariableChangeAdjoint<MODEL>(); });
300  ts.emplace_back(CASE("interface/LinearVariableChange/testLinearVariableChangeInverse")
301  { testLinearVariableChangeInverse<MODEL>(); });
302  ts.emplace_back(CASE("interface/LinearVariableChange/"
303  "testLinearVariableChangeParametersWrapperValidName")
304  { testLinearVariableChangeParametersWrapperValidName<MODEL>(); });
305  ts.emplace_back(CASE("interface/LinearVariableChange/"
306  "testLinearVariableChangeParametersWrapperInvalidName")
307  { testLinearVariableChangeParametersWrapperInvalidName<MODEL>(); });
308  ts.emplace_back(CASE("interface/LinearVariableChange/"
309  "testLinearVariableChangeFactoryGetMakerNames")
310  { testLinearVariableChangeFactoryGetMakerNames<MODEL>(); });
311  }
312 
313  void clear() const override {}
314 };
315 
316 // -----------------------------------------------------------------------------
317 
318 } // namespace test
319 
320 #endif // TEST_INTERFACE_LINEARVARIABLECHANGE_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
static std::vector< std::string > getMakerNames()
Return the names of all linear variable changes that can be created by one of the registered makers.
Contains a polymorphic parameter holding an instance of a subclass of LinearVariableChangeParametersB...
State class used in oops; subclass of interface class interface::State.
std::vector< eckit::LocalConfiguration > confs_
static LinearVariableChangeFixture< MODEL > & getInstance()
static std::vector< eckit::LocalConfiguration > & confs()
std::unique_ptr< const util::DateTime > time_
static const eckit::Configuration & config()
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
Definition: oops/mpi/mpi.cc:84
void testLinearVariableChangeParametersWrapperInvalidName()
void testLinearVariableChangeParametersWrapperValidName()
void testLinearVariableChangeFactoryGetMakerNames()
CASE("test_linearmodelparameterswrapper_valid_name")