OOPS
test/interface/LinearGetValues.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020 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_LINEARGETVALUES_H_
9 #define TEST_INTERFACE_LINEARGETVALUES_H_
10 
11 #include <algorithm>
12 #include <cmath>
13 #include <iomanip>
14 #include <iostream>
15 #include <limits>
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 #include <boost/ptr_container/ptr_vector.hpp>
24 
25 #include "eckit/config/LocalConfiguration.h"
26 #include "eckit/testing/Test.h"
27 #include "oops/base/Geometry.h"
28 #include "oops/base/Increment.h"
29 #include "oops/base/State.h"
30 #include "oops/base/Variables.h"
31 #include "oops/interface/GeoVaLs.h"
35 #include "oops/mpi/mpi.h"
36 #include "oops/runs/Test.h"
37 #include "oops/util/DateTime.h"
38 #include "oops/util/dot_product.h"
39 #include "oops/util/Duration.h"
40 #include "test/TestEnvironment.h"
41 
42 namespace test {
43 
44 // =================================================================================================
45 
46 template <typename MODEL, typename OBS> class LinearGetValuesFixture : private boost::noncopyable {
47  typedef eckit::LocalConfiguration LocalConfig_;
55  typedef util::DateTime DateTime_;
56 
57  public:
58  static const DateTime_ & time() {return *getInstance().time_;}
59  static const DateTime_ & timebeg() {return *getInstance().timebeg_;}
60  static const DateTime_ & timeend() {return *getInstance().timeend_;}
61  static const GeoVaLs_ & geovals() {return *getInstance().geovals_;}
62  static const Geometry_ & resol() {return *getInstance().resol_;}
63  static const GetValues_ & getvalues() {return *getInstance().getvalues_;}
64  static const LinearGetValues_ & lineargetvalues() {return *getInstance().lineargetvalues_;}
65  static const LocalConfig_ & testconf() {return *getInstance().testconf_;}
66  static const Locations_ & locs() {return *getInstance().locs_;}
67  static const State_ & state() {return *getInstance().state_;}
68  static const Variables_ & statevars() {return *getInstance().statevars_;}
69  static const Variables_ & geovalvars() {return *getInstance().geovalvars_;}
70  static const std::vector<size_t> & geovalvarsizes() {return getInstance().geovalvarsizes_;}
71 
72  static void reset() {
73  getInstance().lineargetvalues_.reset();
74  getInstance().time_.reset();
75  getInstance().statevars_.reset();
76  getInstance().state_.reset();
77  getInstance().getvalues_.reset();
78  getInstance().geovals_.reset();
79  getInstance().timeend_.reset();
80  getInstance().timebeg_.reset();
81  getInstance().locs_.reset();
82  getInstance().geovalvars_.reset();
83  getInstance().resol_.reset();
84  getInstance().testconf_.reset();
85  }
86 
87  private:
89  static LinearGetValuesFixture<MODEL, OBS> theLinearGetValuesFixture;
90  return theLinearGetValuesFixture;
91  }
92 
94  testconf_.reset(new LocalConfig_(TestEnvironment::config(), "linear getvalues test"));
95 
96  // Geometry
97  const LocalConfig_ resolConfig(TestEnvironment::config(), "geometry");
98  resol_.reset(new Geometry_(resolConfig, oops::mpi::world()));
99 
100  // Variables
101  geovalvars_.reset(new Variables_(TestEnvironment::config(), "state variables"));
102  geovalvarsizes_ = resol_->variableSizes(*geovalvars_);
103 
104  // Locations
105  const LocalConfig_ locsConfig(TestEnvironment::config(), "locations");
106  locs_.reset(new Locations_(locsConfig, oops::mpi::world()));
107 
108  // Window times
109  timebeg_.reset(new DateTime_(locsConfig.getString("window begin")));
110  timeend_.reset(new DateTime_(locsConfig.getString("window end")));
111 
112  // GeoVaLs
114 
115  // Nonlinear GetValues
116  LocalConfig_ getvaluesConfig;
117  if (TestEnvironment::config().has("get values"))
118  getvaluesConfig = eckit::LocalConfiguration(TestEnvironment::config(), "get values");
119  getvalues_.reset(new GetValues_( *resol_, *locs_, getvaluesConfig));
120 
121  // State
122  const LocalConfig_ stateConfig(TestEnvironment::config(), "background");
123  state_.reset(new State_(*resol_, stateConfig));
124  statevars_.reset(new Variables_(state_->variables()));
125 
126  // Valid time
127  time_.reset(new DateTime_(state_->validTime()));
128 
129  // LinearGetValues
130  LocalConfig_ linearGetValuesConfig;
131  if (TestEnvironment::config().has("linear get values"))
132  linearGetValuesConfig =
133  eckit::LocalConfiguration(TestEnvironment::config(), "linear get values");
135  linearGetValuesConfig));
136 
137  // Set trajectory
139  lineargetvalues_->setTrajectory(*state_, *timebeg_, *timeend_, gvtraj);
140  }
141 
142  ~LinearGetValuesFixture<MODEL, OBS>() {}
143 
144  std::unique_ptr<const DateTime_> time_;
145  std::unique_ptr<const DateTime_> timebeg_;
146  std::unique_ptr<const DateTime_> timeend_;
147  std::unique_ptr<const GeoVaLs_> geovals_;
148  std::unique_ptr<const Geometry_> resol_;
149  std::unique_ptr<const GetValues_> getvalues_;
150  std::unique_ptr<LinearGetValues_> lineargetvalues_;
151  std::unique_ptr<const LocalConfig_> testconf_;
152  std::unique_ptr<const Locations_> locs_;
153  std::unique_ptr<const State_> state_;
154  std::unique_ptr<const Variables_> statevars_;
155  std::unique_ptr<const Variables_> geovalvars_;
156  std::vector<size_t> geovalvarsizes_;
157 };
158 
159 // =================================================================================================
160 /// \brief tests constructor and print method
161 template <typename MODEL, typename OBS> void testLinearGetValuesConstructor() {
163  typedef oops::LinearGetValues<MODEL, OBS> LinearGetValues_;
164 
165  eckit::LocalConfiguration linearGetValuesConfig;
166  if (TestEnvironment::config().has("linear get values"))
167  linearGetValuesConfig =
168  eckit::LocalConfiguration(TestEnvironment::config(), "linear get values");
169 
170  std::unique_ptr<const LinearGetValues_>
171  lineargetvalues(new LinearGetValues_(Test_::resol(),
172  Test_::locs(),
173  linearGetValuesConfig));
174  EXPECT(lineargetvalues.get());
175  oops::Log::test() << "Testing LinearGetValues: " << *lineargetvalues << std::endl;
176  lineargetvalues.reset();
177  EXPECT(!lineargetvalues.get());
178 }
179 
180 // -------------------------------------------------------------------------------------------------
181 
182 template <typename MODEL, typename OBS> void testLinearGetValuesZeroPert() {
184  typedef oops::GeoVaLs<OBS> GeoVaLs_;
185  typedef oops::Increment<MODEL> Increment_;
186 
187  Increment_ dx(Test_::resol(), Test_::statevars(), Test_::time());
188  dx.zero();
189 
190  GeoVaLs_ gv(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
191 
192  EXPECT(dx.norm() == 0.0);
193 
194  Test_::lineargetvalues().fillGeoVaLsTL(dx, Test_::timebeg(), Test_::timeend(), gv);
195 
196  EXPECT(dx.norm() == 0.0);
197  EXPECT(gv.rms() == 0.0);
198 
199  Test_::lineargetvalues().fillGeoVaLsAD(dx, Test_::timebeg(), Test_::timeend(), gv);
200 
201  EXPECT(dx.norm() == 0.0);
202  EXPECT(gv.rms() == 0.0);
203 }
204 
205 // -------------------------------------------------------------------------------------------------
206 
207 template <typename MODEL, typename OBS> void testLinearGetValuesLinearity() {
209  typedef oops::GeoVaLs<OBS> GeoVaLs_;
210  typedef oops::Increment<MODEL> Increment_;
211 
212  const double zz = 3.1415;
213 
214  Increment_ dx1(Test_::resol(), Test_::statevars(), Test_::time());
215  dx1.random();
216  Increment_ dx2(dx1);
217 
218  EXPECT(dx1.norm() > 0.0);
219  EXPECT(dx2.norm() > 0.0);
220 
221  GeoVaLs_ gv1(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
222  GeoVaLs_ gv2(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
223 
224  // Compute geovals
225  Test_::lineargetvalues().fillGeoVaLsTL(dx1, Test_::timebeg(), Test_::timeend(), gv1);
226 
227  gv1 *= zz;
228  dx2 *= zz;
229 
230  // Compute geovals
231  Test_::lineargetvalues().fillGeoVaLsTL(dx2, Test_::timebeg(), Test_::timeend(), gv2);
232 
233  const double tol = Test_::testconf().getDouble("tolerance linearity", 1.0e-11);
234  EXPECT(oops::is_close(gv1.rms(), gv2.rms(), tol));
235 }
236 
237 // -------------------------------------------------------------------------------------------------
238 
239 template <typename MODEL, typename OBS> void testLinearGetValuesLinearApproximation() {
241  typedef oops::GeoVaLs<OBS> GeoVaLs_;
242  typedef oops::Increment<MODEL> Increment_;
243  typedef oops::State<MODEL> State_;
244 
245  const unsigned int ntest = Test_::testconf().getInt("iterations TL", 10);
246  double zz = Test_::testconf().getDouble("first multiplier TL", 1.0e-2);
247 
248  // Compute nonlinear geovals
249  State_ xx0(Test_::state());
250  GeoVaLs_ gv0(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
251  Test_::getvalues().fillGeoVaLs(xx0, Test_::timebeg(), Test_::timeend(), gv0);
252 
253  // Run tangent linear
254  Increment_ dx0(Test_::resol(), Test_::statevars(), Test_::time());
255  dx0.random();
256 
257  std::vector<double> errors;
258  for (unsigned int jtest = 0; jtest < ntest; ++jtest) {
259  Increment_ dxx(dx0);
260  State_ xx(xx0);
261  GeoVaLs_ gv(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
262  GeoVaLs_ dgv(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
263  dxx *= zz;
264  xx += dxx;
265 
266  // Nonlinear
267  Test_::getvalues().fillGeoVaLs(xx, Test_::timebeg(), Test_::timeend(), gv);
268 
269  // Tangent linear
270  Test_::lineargetvalues().fillGeoVaLsTL(dxx, Test_::timebeg(), Test_::timeend(), dgv);
271 
272  GeoVaLs_ gvdiff(gv);
273  gvdiff -= gv0;
274 
275  // Print the norms
276  const double nlnorm = gvdiff.rms();
277  const double tlnorm = dgv.rms();
278 
279  // Subtract the tlm pert
280  gvdiff -= dgv;
281 
282  double testdot = dot_product(gvdiff, gvdiff);
283  errors.push_back(testdot);
284 
285  oops::Log::test() << " ||g(x+dx) - g(x)|| = " << std::setprecision(16) << nlnorm << std::endl;
286  oops::Log::test() << " ||Gdx|| = " << std::setprecision(16) << tlnorm << std::endl;
287  oops::Log::test() << "||g(x+dx)-g(x)-Gdx|| = " << std::setprecision(16) << testdot << std::endl;
288 
289  zz /= 10.0;
290  }
291 
292  // Analyze results
293  const double approx = *std::min_element(errors.begin(), errors.end());
294  oops::Log::test() << "Test LinearGetValuesTL min error = " << approx << std::endl;
295  const double tol = Test_::testconf().getDouble("tolerance TL", 1.0e-11);
296  EXPECT(approx < tol);
297 }
298 
299 // -------------------------------------------------------------------------------------------------
300 
301 template <typename MODEL, typename OBS> void testLinearGetValuesAdjoint() {
303  typedef oops::GeoVaLs<OBS> GeoVaLs_;
304  typedef oops::Increment<MODEL> Increment_;
305 
306  Increment_ dx_in(Test_::resol(), Test_::statevars(), Test_::time());
307  Increment_ dx_ou(Test_::resol(), Test_::statevars(), Test_::time());
308 
309  GeoVaLs_ gv_ou(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
310 
311  // Tangent linear
312  dx_in.random();
313  EXPECT(dx_in.norm() > 0.0);
314  Test_::lineargetvalues().fillGeoVaLsTL(dx_in, Test_::timebeg(), Test_::timeend(), gv_ou);
315  EXPECT(gv_ou.rms() > 0.0);
316 
317  // Adjoint
318  GeoVaLs_ gv_in(gv_ou);
319  gv_in.random(); // No order dependency but need a copy
320  EXPECT(gv_in.rms() > 0.0);
321  dx_ou.zero();
322  Test_::lineargetvalues().fillGeoVaLsAD(dx_ou, Test_::timebeg(), Test_::timeend(), gv_in);
323  EXPECT(dx_ou.norm() > 0.0);
324 
325  // Dot products
326  const double dot1 = dot_product(dx_in, dx_ou);
327  const double dot2 = dot_product(gv_in, gv_ou);
328  const double tol = Test_::testconf().getDouble("tolerance AD", 1.0e-11);
329  EXPECT(oops::is_close(dot1, dot2, tol));
330 
331  oops::Log::test() << "Dot Product <dx, M^Tgv> = " << std::setprecision(16) << dot1 << std::endl;
332  oops::Log::test() << "Dot Product <gv, M dx> = " << std::setprecision(16) << dot2 << std::endl;
333  oops::Log::test() << "Relative diff: " << std::setprecision(16) << (dot1-dot2)/dot1 << std::endl;
334 }
335 
336 // =================================================================================================
337 
338 template <typename MODEL, typename OBS>
339 class LinearGetValues : public oops::Test {
340  public:
343 
344  private:
345  std::string testid() const override {return "test::LinearGetValues<" + MODEL::name() + ", "
346  + OBS::name() + ">";}
347 
348  void register_tests() const override {
349  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
350 
351  ts.emplace_back(CASE("interface/LinearGetValues/testLinearGetValuesConstructor")
352  { testLinearGetValuesConstructor<MODEL, OBS>(); });
353  ts.emplace_back(CASE("interface/LinearGetValues/testLinearGetValuesZeroPert")
354  { testLinearGetValuesZeroPert<MODEL, OBS>(); });
355  ts.emplace_back(CASE("interface/LinearGetValues/testLinearGetValuesLinearity")
356  { testLinearGetValuesLinearity<MODEL, OBS>(); });
357  ts.emplace_back(CASE("interface/LinearGetValues/testLinearGetValuesLinearApproximation")
358  { testLinearGetValuesLinearApproximation<MODEL, OBS>(); });
359  ts.emplace_back(CASE("interface/LinearGetValues/testLinearGetValuesAdjoint")
360  { testLinearGetValuesAdjoint<MODEL, OBS>(); });
361  }
362 
363  void clear() const override {}
364 };
365 
366 // =================================================================================================
367 
368 } // namespace test
369 
370 #endif // TEST_INTERFACE_LINEARGETVALUES_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
Gets values from model State to observation locations (fills GeoVaLs)
Increment class used in oops.
sets trajectory and computes TL and AD for GetValues
Locations of observations for observation operator.
State class used in oops; subclass of interface class interface::State.
std::unique_ptr< const DateTime_ > time_
std::unique_ptr< const DateTime_ > timeend_
static const LocalConfig_ & testconf()
std::unique_ptr< const Variables_ > statevars_
std::unique_ptr< LinearGetValues_ > lineargetvalues_
std::unique_ptr< const GetValues_ > getvalues_
static const LinearGetValues_ & lineargetvalues()
oops::GetValues< MODEL, OBS > GetValues_
std::unique_ptr< const DateTime_ > timebeg_
std::unique_ptr< const GeoVaLs_ > geovals_
std::unique_ptr< const Variables_ > geovalvars_
std::unique_ptr< const Geometry_ > resol_
static const Variables_ & geovalvars()
std::unique_ptr< const State_ > state_
oops::LinearGetValues< MODEL, OBS > LinearGetValues_
std::unique_ptr< const Locations_ > locs_
static LinearGetValuesFixture< MODEL, OBS > & getInstance()
static const std::vector< size_t > & geovalvarsizes()
std::unique_ptr< const LocalConfig_ > testconf_
std::string testid() const override
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
logical function has(this, var)
void testLinearGetValuesZeroPert()
void testLinearGetValuesLinearApproximation()
void testLinearGetValuesConstructor()
tests constructor and print method
void testLinearGetValuesLinearity()
CASE("test_linearmodelparameterswrapper_valid_name")