11 #ifndef TEST_INTERFACE_LINEARVARIABLECHANGE_H_
12 #define TEST_INTERFACE_LINEARVARIABLECHANGE_H_
20 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
22 #include <boost/noncopyable.hpp>
24 #include "eckit/config/Configuration.h"
25 #include "eckit/testing/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"
63 return theLinearVariableChangeFixture;
67 oops::instantiateVariableChangeFactory<MODEL>();
75 time_.reset(
new util::DateTime(
xx_->validTime()));
80 ~LinearVariableChangeFixture<MODEL>() {}
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_;
96 for (std::size_t jj = 0; jj < Test_::confs().size(); ++jj) {
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());
111 dxoutTlIAd = changevar->multiply(dxinTlIAd);
112 EXPECT(dxoutTlIAd.norm() == 0.0);
116 dxoutAdInv = changevar->multiplyAD(dxinAdInv);
117 EXPECT(dxoutAdInv.norm() == 0.0);
119 const bool testinverse = Test_::confs()[jj].getBool(
"test inverse",
true);
122 oops::Log::test() <<
"Doing zero test for inverse" << std::endl;
124 dxoutTlIAd = changevar->multiplyInverseAD(dxinTlIAd);
125 EXPECT(dxoutTlIAd.norm() == 0.0);
128 dxoutAdInv = changevar->multiplyInverse(dxinAdInv);
129 EXPECT(dxoutAdInv.norm() == 0.0);
131 oops::Log::test() <<
"Not doing zero test for inverse" << std::endl;
143 for (std::size_t jj = 0; jj < Test_::confs().size(); ++jj) {
147 std::unique_ptr<LinearVariableChange_> changevar(LinearVariableChangeFactory_::create(
148 Test_::xx(), Test_::xx(),
149 Test_::resol(), Test_::confs()[jj]));
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());
159 Increment_ dxinAdInv0(dxinAdInv);
160 Increment_ dxinTlIAd0(dxinTlIAd);
162 dxoutTlIAd = changevar->multiply(dxinTlIAd);
163 dxoutAdInv = changevar->multiplyAD(dxinAdInv);
166 double zz1 = dot_product(dxoutTlIAd, dxinAdInv0);
168 double zz2 = dot_product(dxinTlIAd0, dxoutAdInv);
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);
179 oops::Log::test() <<
"Doing adjoint test for inverse" << std::endl;
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));
196 oops::Log::test() <<
"Not doing adjoint test for inverse" << std::endl;
209 for (std::size_t jj = 0; jj < Test_::confs().size(); ++jj) {
213 const double tol = Test_::confs()[jj].getDouble(
"tolerance inverse");
215 const bool testinverse = Test_::confs()[jj].getBool(
"test inverse",
false);
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]));
223 Increment_ dxinInv(Test_::resol(), varout, Test_::time());
224 Increment_ dxoutInv(Test_::resol(), varin, Test_::time());
225 Increment_ dxout(Test_::resol(), varout, Test_::time());
229 dxoutInv = changevar->multiplyInverse(dxinInv);
230 dxout = changevar->multiply(dxoutInv);
232 const double zz1 = dxinInv.norm();
233 const double zz2 = dxout.norm();
235 oops::Log::test() <<
"<x>, <KK^{-1}x>=" << zz1 <<
" " << zz2 << std::endl;
236 oops::Log::test() <<
"<x>-<KK^{-1}x>=" << zz1-zz2 << std::endl;
238 EXPECT((zz1-zz2) < tol);
240 oops::Log::test() <<
"multiplyInverse test not executed" << std::endl;
250 for (
const eckit::Configuration &config : Test_::confs()) {
252 EXPECT_NO_THROW(parameters.validateAndDeserialize(config));
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");
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();
284 template <
typename MODEL>
291 std::string
testid()
const override {
return "test::LinearVariableChange<" + MODEL::name() +
">";}
294 std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
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>(); });
Geometry class used in oops; subclass of interface class interface::Geometry.
Increment class used in oops.
LinearVariableChange factory.
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.
oops::State< MODEL > State_
std::vector< eckit::LocalConfiguration > confs_
static const Geometry_ & resol()
std::unique_ptr< const Geometry_ > resol_
static LinearVariableChangeFixture< MODEL > & getInstance()
static std::vector< eckit::LocalConfiguration > & confs()
oops::Geometry< MODEL > Geometry_
std::unique_ptr< const State_ > xx_
static const State_ & xx()
static const DateTime_ & time()
std::unique_ptr< const util::DateTime > time_
std::string testid() const override
void clear() const override
virtual ~LinearVariableChange()
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)
void testLinearVariableChangeParametersWrapperInvalidName()
void testLinearVariableChangeAdjoint()
void testLinearVariableChangeParametersWrapperValidName()
void testLinearVariableChangeInverse()
void testLinearVariableChangeFactoryGetMakerNames()
void testLinearVariableChangeZero()
CASE("test_linearmodelparameterswrapper_valid_name")