OOPS
test/interface/ObsAuxIncrement.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_OBSAUXINCREMENT_H_
12 #define TEST_INTERFACE_OBSAUXINCREMENT_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/LocalConfiguration.h"
25 #include "eckit/testing/Test.h"
29 #include "oops/runs/Test.h"
30 #include "oops/util/DateTime.h"
31 #include "oops/util/dot_product.h"
33 #include "test/TestEnvironment.h"
34 
35 namespace test {
36 
37 // =============================================================================
38 
39 template <typename OBS> class ObsAuxIncrementFixture : private boost::noncopyable {
42 
43  public:
44  static const Covariance_ & covariance(const size_t ii) {return *getInstance().covar_.at(ii);}
45 
46  private:
48  static ObsAuxIncrementFixture<OBS> theObsAuxIncrementFixture;
49  return theObsAuxIncrementFixture;
50  }
51 
53  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
54  eckit::LocalConfiguration biasconf =
55  Test_::config(jj).getSubConfiguration("obs bias");
56  typename Covariance_::Parameters_ biasparams;
57  biasparams.validateAndDeserialize(biasconf);
58  std::shared_ptr<Covariance_> tmp(new Covariance_(Test_::obspace()[jj], biasparams));
59  covar_.push_back(tmp);
60  }
61  }
62 
63  ~ObsAuxIncrementFixture<OBS>() {}
64 
65  std::vector<std::shared_ptr<Covariance_> > covar_;
66 };
67 
68 // =============================================================================
69 /// \brief test constructor and print method
70 template <typename OBS> void testObsAuxIncrementConstructor() {
71  typedef ObsTestsFixture<OBS> Test_;
72  typedef oops::ObsAuxIncrement<OBS> AuxIncr_;
73 
74  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
75  eckit::LocalConfiguration biasconf = Test_::config(jj).getSubConfiguration("obs bias");
76  typename AuxIncr_::Parameters_ biasparams;
77  biasparams.validateAndDeserialize(biasconf);
78  AuxIncr_ dx(Test_::obspace()[jj], biasparams);
79  oops::Log::test() << "Printing zero ObsAuxIncrement: " << dx << std::endl;
80  EXPECT(dx.norm() == 0.0);
81  }
82 }
83 
84 // -----------------------------------------------------------------------------
85 /// Tests copy-constructor (with option of allocating, but not copying the data)
86 template <typename OBS> void testObsAuxIncrementCopyConstructor() {
87  typedef ObsTestsFixture<OBS> Test_;
88  typedef oops::ObsAuxIncrement<OBS> AuxIncr_;
89 
90  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
91  if (Test_::config(jj).has("obs bias")) {
92  eckit::LocalConfiguration biasconf = Test_::config(jj).getSubConfiguration("obs bias");
93  typename AuxIncr_::Parameters_ biasparams;
94  biasparams.validateAndDeserialize(biasconf);
95  AuxIncr_ dx1(Test_::obspace()[jj], biasparams);
97  oops::Log::test() << "Printing random ObsAuxIncrement: " << dx1 << std::endl;
98  /// Test that creating new increment without copying data works
99  AuxIncr_ dxempty(dx1, false);
100  EXPECT_EQUAL(dxempty.norm(), 0.0);
101  /// Test that creating new increment with copying data works
102  AuxIncr_ dx2(dx1);
103  EXPECT(dx2.norm() > 0.0);
104  EXPECT(dx2.norm() == dx1.norm());
105 
106  /// Test that the copy is equal to the original
107  dx2 -= dx1;
108  EXPECT(dx2.norm() == 0.0);
109  }
110  }
111 }
112 
113 // -----------------------------------------------------------------------------
114 
115 template <typename OBS> void testObsAuxIncrementTriangle() {
116  typedef ObsTestsFixture<OBS> Test_;
117  typedef ObsAuxIncrementFixture<OBS> AuxTest_;
118  typedef oops::ObsAuxIncrement<OBS> AuxIncr_;
119 
120  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
121  if (Test_::config(jj).has("obs bias")) {
122  eckit::LocalConfiguration biasconf = Test_::config(jj).getSubConfiguration("obs bias");
123  typename AuxIncr_::Parameters_ biasparams;
124  biasparams.validateAndDeserialize(biasconf);
125  AuxIncr_ dx1(Test_::obspace()[jj], biasparams);
126  AuxTest_::covariance(jj).randomize(dx1);
127  AuxIncr_ dx2(Test_::obspace()[jj], biasparams);
129 
130 // test triangle inequality
131  double dot1 = dx1.norm();
132  EXPECT(dot1 > 0.0);
133 
134  double dot2 = dx2.norm();
135  EXPECT(dot2 > 0.0);
136 
137  dx2 += dx1;
138  double dot3 = dx2.norm();
139  EXPECT(dot3 > 0.0);
140 
141  EXPECT(dot3 <= dot1 + dot2);
142  }
143  }
144 }
145 
146 // -----------------------------------------------------------------------------
147 
148 template <typename OBS> void testObsAuxIncrementOpPlusEq() {
149  typedef ObsTestsFixture<OBS> Test_;
150  typedef ObsAuxIncrementFixture<OBS> AuxTest_;
151  typedef oops::ObsAuxIncrement<OBS> AuxIncr_;
152 
153  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
154  eckit::LocalConfiguration biasconf = Test_::config(jj).getSubConfiguration("obs bias");
155  typename AuxIncr_::Parameters_ biasparams;
156  biasparams.validateAndDeserialize(biasconf);
157  AuxIncr_ dx1(Test_::obspace()[jj], biasparams);
158  AuxTest_::covariance(jj).randomize(dx1);
159  AuxIncr_ dx2(dx1);
160 
161 // test *= and +=
162  dx2 += dx1;
163  dx1 *= 2.0;
164 
165  dx2 -= dx1;
166  EXPECT(dx2.norm() < 1e-8);
167  }
168 }
169 
170 // -----------------------------------------------------------------------------
171 
172 template <typename OBS> void testObsAuxIncrementDotProduct() {
173  typedef ObsTestsFixture<OBS> Test_;
174  typedef ObsAuxIncrementFixture<OBS> AuxTest_;
175  typedef oops::ObsAuxIncrement<OBS> AuxIncr_;
176 
177  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
178  eckit::LocalConfiguration biasconf = Test_::config(jj).getSubConfiguration("obs bias");
179  typename AuxIncr_::Parameters_ biasparams;
180  biasparams.validateAndDeserialize(biasconf);
181  AuxIncr_ dx1(Test_::obspace()[jj], biasparams);
182  AuxTest_::covariance(jj).randomize(dx1);
183  AuxIncr_ dx2(Test_::obspace()[jj], biasparams);
184  AuxTest_::covariance(jj).randomize(dx2);
185 
186 // test symmetry of dot product
187  double zz1 = dot_product(dx1, dx2);
188  double zz2 = dot_product(dx2, dx1);
189 
190  EXPECT(zz1 == zz2);
191  }
192 }
193 
194 // -----------------------------------------------------------------------------
195 
196 template <typename OBS> void testObsAuxIncrementZero() {
197  typedef ObsTestsFixture<OBS> Test_;
198  typedef ObsAuxIncrementFixture<OBS> AuxTest_;
199  typedef oops::ObsAuxIncrement<OBS> AuxIncr_;
200 
201  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
202  eckit::LocalConfiguration biasconf = Test_::config(jj).getSubConfiguration("obs bias");
203  typename AuxIncr_::Parameters_ biasparams;
204  biasparams.validateAndDeserialize(biasconf);
205  AuxIncr_ dx(Test_::obspace()[jj], biasparams);
206  AuxTest_::covariance(jj).randomize(dx);
207  EXPECT(dx.norm() > 0.0);
208 
209 // test zero
210  dx->zero();
211  EXPECT(dx.norm() == 0.0);
212  }
213 }
214 
215 // -----------------------------------------------------------------------------
216 
217 template <typename OBS> void testObsAuxIncrementAxpy() {
218  typedef ObsTestsFixture<OBS> Test_;
219  typedef ObsAuxIncrementFixture<OBS> AuxTest_;
220  typedef oops::ObsAuxIncrement<OBS> AuxIncr_;
221 
222  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
223  eckit::LocalConfiguration biasconf = Test_::config(jj).getSubConfiguration("obs bias");
224  typename AuxIncr_::Parameters_ biasparams;
225  biasparams.validateAndDeserialize(biasconf);
226  AuxIncr_ dx1(Test_::obspace()[jj], biasparams);
227  AuxTest_::covariance(jj).randomize(dx1);
228 
229 // test axpy
230  AuxIncr_ dx2(dx1);
231  dx2.axpy(2.0, dx1);
232 
233  dx2 -= dx1;
234  dx2 -= dx1;
235  dx2 -= dx1;
236 
237  EXPECT(dx2.norm() < 1e-8);
238  }
239 }
240 
241 // =============================================================================
242 
243 template <typename OBS>
244 class ObsAuxIncrement : public oops::Test {
246 
247  public:
249  virtual ~ObsAuxIncrement() {}
250 
251  private:
252  std::string testid() const override {return "test::ObsAuxIncrement<" + OBS::name() + ">";}
253 
254  void register_tests() const override {
255  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
256 
257  ts.emplace_back(CASE("interface/ObsAuxIncrement/testObsAuxIncrementConstructor")
258  { testObsAuxIncrementConstructor<OBS>(); });
259  ts.emplace_back(CASE("interface/ObsAuxIncrement/testObsAuxIncrementCopyConstructor")
260  { testObsAuxIncrementCopyConstructor<OBS>(); });
261  ts.emplace_back(CASE("interface/ObsAuxIncrement/testObsAuxIncrementTriangle")
262  { testObsAuxIncrementTriangle<OBS>(); });
263  ts.emplace_back(CASE("interface/ObsAuxIncrement/testObsAuxIncrementOpPlusEq")
264  { testObsAuxIncrementOpPlusEq<OBS>(); });
265  ts.emplace_back(CASE("interface/ObsAuxIncrement/testObsAuxIncrementDotProduct")
266  { testObsAuxIncrementDotProduct<OBS>(); });
267  ts.emplace_back(CASE("interface/ObsAuxIncrement/testObsAuxIncrementAxpy")
268  { testObsAuxIncrementAxpy<OBS>(); });
269  }
270 
271  void clear() const override {
272  Test_::reset();
273  }
274 };
275 
276 // =============================================================================
277 
278 } // namespace test
279 
280 #endif // TEST_INTERFACE_OBSAUXINCREMENT_H_
Auxiliary error covariance related to observations, templated on <OBS>
ObsAuxCovariance_::Parameters_ Parameters_
void randomize(ObsAuxIncrement_ &) const
randomize the values in the ObsAuxIncrement
Auxiliary increment related to observations, templated on <OBS>
std::size_t size() const
Access.
Definition: ObsSpaces.h:61
oops::ObsAuxCovariance< OBS > Covariance_
static ObsAuxIncrementFixture< OBS > & getInstance()
static const Covariance_ & covariance(const size_t ii)
std::vector< std::shared_ptr< Covariance_ > > covar_
std::string testid() const override
static ObsSpaces_ & obspace()
accessor to a all obs spaces
static eckit::LocalConfiguration & config(size_t jj)
accessor to a jj-th obs type config
void testObsAuxIncrementCopyConstructor()
Tests copy-constructor (with option of allocating, but not copying the data)
void testObsAuxIncrementConstructor()
test constructor and print method
void testObsAuxIncrementOpPlusEq()
void testObsAuxIncrementTriangle()
void testObsAuxIncrementDotProduct()
CASE("test_linearmodelparameterswrapper_valid_name")