8 #ifndef TEST_INTERFACE_LINEARGETVALUES_H_
9 #define TEST_INTERFACE_LINEARGETVALUES_H_
20 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
22 #include <boost/noncopyable.hpp>
23 #include <boost/ptr_container/ptr_vector.hpp>
25 #include "eckit/config/LocalConfiguration.h"
26 #include "eckit/testing/Test.h"
37 #include "oops/util/DateTime.h"
38 #include "oops/util/dot_product.h"
39 #include "oops/util/Duration.h"
90 return theLinearGetValuesFixture;
132 linearGetValuesConfig =
135 linearGetValuesConfig));
142 ~LinearGetValuesFixture<MODEL, OBS>() {}
144 std::unique_ptr<const DateTime_>
time_;
152 std::unique_ptr<const Locations_>
locs_;
165 eckit::LocalConfiguration linearGetValuesConfig;
167 linearGetValuesConfig =
170 std::unique_ptr<const LinearGetValues_>
171 lineargetvalues(
new LinearGetValues_(Test_::resol(),
173 linearGetValuesConfig));
174 EXPECT(lineargetvalues.get());
175 oops::Log::test() <<
"Testing LinearGetValues: " << *lineargetvalues << std::endl;
176 lineargetvalues.reset();
177 EXPECT(!lineargetvalues.get());
187 Increment_ dx(Test_::resol(), Test_::statevars(), Test_::time());
190 GeoVaLs_ gv(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
192 EXPECT(dx.norm() == 0.0);
194 Test_::lineargetvalues().fillGeoVaLsTL(dx, Test_::timebeg(), Test_::timeend(), gv);
196 EXPECT(dx.norm() == 0.0);
197 EXPECT(gv.rms() == 0.0);
199 Test_::lineargetvalues().fillGeoVaLsAD(dx, Test_::timebeg(), Test_::timeend(), gv);
201 EXPECT(dx.norm() == 0.0);
202 EXPECT(gv.rms() == 0.0);
212 const double zz = 3.1415;
214 Increment_ dx1(Test_::resol(), Test_::statevars(), Test_::time());
218 EXPECT(dx1.norm() > 0.0);
219 EXPECT(dx2.norm() > 0.0);
221 GeoVaLs_ gv1(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
222 GeoVaLs_ gv2(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
225 Test_::lineargetvalues().fillGeoVaLsTL(dx1, Test_::timebeg(), Test_::timeend(), gv1);
231 Test_::lineargetvalues().fillGeoVaLsTL(dx2, Test_::timebeg(), Test_::timeend(), gv2);
233 const double tol = Test_::testconf().getDouble(
"tolerance linearity", 1.0e-11);
234 EXPECT(oops::is_close(gv1.rms(), gv2.rms(), tol));
245 const unsigned int ntest = Test_::testconf().getInt(
"iterations TL", 10);
246 double zz = Test_::testconf().getDouble(
"first multiplier TL", 1.0e-2);
249 State_ xx0(Test_::state());
250 GeoVaLs_ gv0(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
251 Test_::getvalues().fillGeoVaLs(xx0, Test_::timebeg(), Test_::timeend(), gv0);
254 Increment_ dx0(Test_::resol(), Test_::statevars(), Test_::time());
257 std::vector<double> errors;
258 for (
unsigned int jtest = 0; jtest < ntest; ++jtest) {
261 GeoVaLs_ gv(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
262 GeoVaLs_ dgv(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
267 Test_::getvalues().fillGeoVaLs(xx, Test_::timebeg(), Test_::timeend(), gv);
270 Test_::lineargetvalues().fillGeoVaLsTL(dxx, Test_::timebeg(), Test_::timeend(), dgv);
276 const double nlnorm = gvdiff.rms();
277 const double tlnorm = dgv.rms();
282 double testdot = dot_product(gvdiff, gvdiff);
283 errors.push_back(testdot);
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;
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);
306 Increment_ dx_in(Test_::resol(), Test_::statevars(), Test_::time());
307 Increment_ dx_ou(Test_::resol(), Test_::statevars(), Test_::time());
309 GeoVaLs_ gv_ou(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
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);
318 GeoVaLs_ gv_in(gv_ou);
320 EXPECT(gv_in.rms() > 0.0);
322 Test_::lineargetvalues().fillGeoVaLsAD(dx_ou, Test_::timebeg(), Test_::timeend(), gv_in);
323 EXPECT(dx_ou.norm() > 0.0);
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));
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;
338 template <
typename MODEL,
typename OBS>
345 std::string
testid()
const override {
return "test::LinearGetValues<" + MODEL::name() +
", "
346 + OBS::name() +
">";}
349 std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
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>(); });
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.
static const Variables_ & statevars()
std::unique_ptr< const DateTime_ > time_
std::unique_ptr< const DateTime_ > timeend_
static const State_ & state()
oops::GeoVaLs< OBS > GeoVaLs_
static const LocalConfig_ & testconf()
std::unique_ptr< const Variables_ > statevars_
static const Locations_ & locs()
eckit::LocalConfiguration LocalConfig_
static const GeoVaLs_ & geovals()
std::unique_ptr< LinearGetValues_ > lineargetvalues_
static const DateTime_ & time()
oops::State< MODEL > State_
static const Geometry_ & resol()
std::unique_ptr< const GetValues_ > getvalues_
static const LinearGetValues_ & lineargetvalues()
oops::GetValues< MODEL, OBS > GetValues_
static const GetValues_ & getvalues()
std::unique_ptr< const DateTime_ > timebeg_
std::vector< size_t > geovalvarsizes_
std::unique_ptr< const GeoVaLs_ > geovals_
std::unique_ptr< const Variables_ > geovalvars_
std::unique_ptr< const Geometry_ > resol_
static const Variables_ & geovalvars()
oops::Geometry< MODEL > Geometry_
std::unique_ptr< const State_ > state_
oops::Variables Variables_
oops::LinearGetValues< MODEL, OBS > LinearGetValues_
std::unique_ptr< const Locations_ > locs_
static LinearGetValuesFixture< MODEL, OBS > & getInstance()
static const DateTime_ & timebeg()
oops::Locations< OBS > Locations_
static const std::vector< size_t > & geovalvarsizes()
std::unique_ptr< const LocalConfig_ > testconf_
static const DateTime_ & timeend()
std::string testid() const override
void clear() const override
virtual ~LinearGetValues()
void register_tests() const override
static const eckit::Configuration & config()
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
logical function has(this, var)
void testLinearGetValuesAdjoint()
void testLinearGetValuesZeroPert()
void testLinearGetValuesLinearApproximation()
void testLinearGetValuesConstructor()
tests constructor and print method
void testLinearGetValuesLinearity()
CASE("test_linearmodelparameterswrapper_valid_name")