Loading [MathJax]/extensions/tex2jax.js
OOPS
All Classes Namespaces Files Functions Variables Typedefs Macros Pages
ObsErrorCovariance.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_BASE_OBSERRORCOVARIANCE_H_
12 #define TEST_BASE_OBSERRORCOVARIANCE_H_
13 
14 #include <memory>
15 #include <string>
16 #include <vector>
17 
18 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
19 
20 #include "eckit/testing/Test.h"
21 #include "oops/base/ObsError.h"
22 #include "oops/base/ObsVector.h"
24 #include "oops/runs/Test.h"
26 #include "test/TestEnvironment.h"
27 
28 namespace test {
29 
30 // -----------------------------------------------------------------------------
31 /// Tests creation and destruction of ObsErrorCovariances
32 template <typename OBS> void testConstructor() {
33  typedef ObsTestsFixture<OBS> Test_;
34  typedef oops::ObsError<OBS> Covar_;
35  typedef oops::ObsErrorParametersWrapper<OBS> Parameters_;
36 
37  oops::instantiateObsErrorFactory<OBS>();
38 
39  std::vector<eckit::LocalConfiguration> conf;
40  TestEnvironment::config().get("observations", conf);
41 
42  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
43  const eckit::LocalConfiguration rconf(conf[jj], "obs error");
44  Parameters_ rparams;
45  rparams.validateAndDeserialize(rconf);
46  std::unique_ptr<Covar_> R = std::make_unique<Covar_>(rparams.obsErrorParameters,
47  Test_::obspace()[jj]);
48  EXPECT(R.get());
49  oops::Log::test() << "Testing ObsError: " << *R << std::endl;
50  R.reset();
51  EXPECT(!R.get());
52  }
53 }
54 
55 // -----------------------------------------------------------------------------
56 /// Tests that \f$R*R^{-1}*dy = dy\f$ and \f$R^{-1}*R*dy = dy\f$
57 template <typename OBS> void testMultiplies() {
58  typedef ObsTestsFixture<OBS> Test_;
59  typedef oops::ObsError<OBS> Covar_;
60  typedef oops::ObsErrorParametersWrapper<OBS> Parameters_;
61  typedef oops::ObsVector<OBS> ObsVector_;
62 
63  oops::instantiateObsErrorFactory<OBS>();
64 
65  std::vector<eckit::LocalConfiguration> conf;
66  TestEnvironment::config().get("observations", conf);
67 
68  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
69  ObsVector_ obserr(Test_::obspace()[jj], "ObsError");
70 
71  const eckit::LocalConfiguration rconf(conf[jj], "obs error");
72  Parameters_ rparams;
73  rparams.validateAndDeserialize(rconf);
74  Covar_ R(rparams.obsErrorParameters, Test_::obspace()[jj]);
75 
76  // RMSE should be equal to the rms that was read from the file
77  EXPECT(oops::is_close(R.getRMSE(), obserr.rms(), 1.e-10));
78 
79  // create random vector dy and its copies dy1, dy2
80  ObsVector_ dy(Test_::obspace()[jj]);
81  R.randomize(dy);
82  ObsVector_ dy1(dy);
83  ObsVector_ dy2(dy);
84  oops::Log::test() << "Random vector dy: " << dy << std::endl;
85 
86  R.multiply(dy1);
87  oops::Log::test() << "R*dy: " << dy1 << std::endl;
88  R.inverseMultiply(dy1);
89  // dy1 = R^{-1}*R*dy
90  oops::Log::test() << "R^{-1}*R*dy: " << dy1 << std::endl;
91  EXPECT(oops::is_close(dy1.rms(), dy.rms(), 1.e-10));
92 
93  R.inverseMultiply(dy2);
94  oops::Log::test() << "R^{-1}*dy: " << dy2 << std::endl;
95  R.multiply(dy2);
96  // dy2 = R*R^P-1}*dy
97  oops::Log::test() << "R*R^{-1}*dy: " << dy2 << std::endl;
98  EXPECT(oops::is_close(dy2.rms(), dy.rms(), 1.e-10));
99  }
100 }
101 
102 // -----------------------------------------------------------------------------
103 /// Tests that the methods obserrors(), inverseVariance update() and save()
104 /// do what is expected.
105 template <typename OBS> void testAccessors() {
106  typedef ObsTestsFixture<OBS> Test_;
107  typedef oops::ObsError<OBS> Covar_;
108  typedef oops::ObsErrorParametersWrapper<OBS> Parameters_;
109  typedef oops::ObsVector<OBS> ObsVector_;
110 
111  oops::instantiateObsErrorFactory<OBS>();
112 
113  std::vector<eckit::LocalConfiguration> conf;
114  TestEnvironment::config().get("observations", conf);
115 
116  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
117  ObsVector_ obserr(Test_::obspace()[jj], "ObsError");
118 
119  const eckit::LocalConfiguration rconf(conf[jj], "obs error");
120  Parameters_ rparams;
121  rparams.validateAndDeserialize(rconf);
122  Covar_ R(rparams.obsErrorParameters, Test_::obspace()[jj]);
123 
124  ObsVector_ dy(R.obserrors());
125  oops::Log::test() << "ObsError: " << dy << std::endl;
126  EXPECT(oops::is_close(dy.rms(), obserr.rms(), 1.e-10));
127 
128  ObsVector_ dy1(R.inverseVariance());
129  oops::Log::test() << "inverseVariance: " << dy1 << std::endl;
130  dy *= dy;
131  dy.invert();
132  EXPECT(oops::is_close(dy.rms(), dy1.rms(), 1.e-10));
133 
134  dy.ones();
135  R.update(dy);
136  oops::Log::test() << "R filled with ones: " << R.obserrors() << std::endl;
137  EXPECT(oops::is_close(R.obserrors().rms(), R.inverseVariance().rms(), 1.e-10));
138  EXPECT(oops::is_close(R.obserrors().rms(), dy.rms(), 1.e-10));
139 
140  R.save("Ones");
141  ObsVector_ testOnes(Test_::obspace()[jj], "Ones");
142  EXPECT(oops::is_close(dy.rms(), testOnes.rms(), 1.e-10));
143  }
144 }
145 
146 // -----------------------------------------------------------------------------
147 
148 template <typename OBS>
151  public:
153  virtual ~ObsErrorCovariance() {}
154  private:
155  std::string testid() const override {return "test::ObsErrorCovariance<" + OBS::name() + ">";}
156 
157  void register_tests() const override {
158  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
159 
160  ts.emplace_back(CASE("interface/ObsErrorCovariance/testConstructor")
161  { testConstructor<OBS>(); });
162  ts.emplace_back(CASE("interface/ObsErrorCovariance/testMultiplies")
163  { testMultiplies<OBS>(); });
164  ts.emplace_back(CASE("interface/ObsErrorCovariance/testAccessors")
165  { testAccessors<OBS>(); });
166  }
167 
168  void clear() const override {
169  Test_::reset();
170  }
171 };
172 
173 // -----------------------------------------------------------------------------
174 
175 } // namespace test
176 
177 #endif // TEST_BASE_OBSERRORCOVARIANCE_H_
Observation error covariance matrix of observations from a single ObsSpace.
Definition: ObsError.h:29
Contains a polymorphic parameter holding an instance of a subclass of ObsErrorParametersBase.
ObsVector class used in oops; subclass of interface class interface::ObsVector.
void register_tests() const override
ObsTestsFixture< OBS > Test_
void clear() const override
std::string testid() const override
static const eckit::Configuration & config()
void testAccessors()
void testMultiplies()
Tests that and .
CASE("test_linearmodelparameterswrapper_valid_name")
void testConstructor()
Tests creation and destruction of ObsErrorCovariances.