OOPS
test/interface/ObsVector.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_OBSVECTOR_H_
12 #define TEST_INTERFACE_OBSVECTOR_H_
13 
14 #include <memory>
15 #include <string>
16 #include <utility>
17 #include <vector>
18 
19 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
20 
21 #include "eckit/testing/Test.h"
22 #include "oops/base/ObsVector.h"
23 #include "oops/base/Variables.h"
25 #include "oops/runs/Test.h"
26 #include "oops/util/dot_product.h"
28 #include "test/TestEnvironment.h"
29 
30 namespace test {
31 
32 // -----------------------------------------------------------------------------
33 /// \brief tests constructor and print method
34 template <typename OBS> void testConstructor() {
35  typedef ObsTestsFixture<OBS> Test_;
36  typedef oops::ObsVector<OBS> ObsVector_;
37 
38  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
39  std::unique_ptr<ObsVector_> ov(new ObsVector_(Test_::obspace()[jj]));
40  EXPECT(ov.get());
41  oops::Log::test() << "Printing zero ObsVector: " << *ov << std::endl;
42  ov.reset();
43  EXPECT(!ov.get());
44  }
45 }
46 
47 // -----------------------------------------------------------------------------
48 
49 template <typename OBS> void testCopyConstructor() {
50  typedef ObsTestsFixture<OBS> Test_;
51  typedef oops::ObsVector<OBS> ObsVector_;
52 
53  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
54  std::unique_ptr<ObsVector_> ov(new ObsVector_(Test_::obspace()[jj]));
55 
56  ov->random();
57  oops::Log::test() << "Printing random ObsVector: " << *ov << std::endl;
58 
59  std::unique_ptr<ObsVector_> other(new ObsVector_(*ov));
60  EXPECT(other.get());
61 
62  const double ov2 = dot_product(*ov, *ov);
63  const double other2 = dot_product(*other, *other);
64 
65  EXPECT(ov2 == other2);
66 
67  other.reset();
68  EXPECT(!other.get());
69 
70  EXPECT(ov.get());
71  }
72 }
73 
74 // -----------------------------------------------------------------------------
75 
76 /// Test the constructor taking a std::unique_ptr<OBS::ObsVector>.
77 template <typename OBS> void testWrappingConstructor() {
78  typedef ObsTestsFixture<OBS> Test_;
79  typedef oops::ObsVector<OBS> ObsVector_;
80 
81  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
82  ObsVector_ ov(Test_::obspace()[jj]);
83 
84  ov.random();
85  oops::Log::test() << "Printing random ObsVector: " << ov << std::endl;
86 
87  ObsVector_ other(std::make_unique<typename OBS::ObsVector_>(ov->obsvector()),
88  Test_::obspace()[jj]);
89 
90  const double ov2 = dot_product(ov, ov);
91  const double other2 = dot_product(other, other);
92 
93  EXPECT(ov2 == other2);
94  }
95 }
96 
97 // -----------------------------------------------------------------------------
98 /// Test that:
99 /// - dot_product of a random vector with self is non-zero
100 /// - dot_product of a random vector with a zero vector is zero
101 /// - dot_product of a zero vector with self is zero
102 /// - dot_product of a vector of ones with self is equal to nobs
103 template <typename OBS> void testNotZero() {
104  typedef ObsTestsFixture<OBS> Test_;
105  typedef oops::ObsVector<OBS> ObsVector_;
106  const double zero = 0.0;
107 
108  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
109  ObsVector_ ov1(Test_::obspace()[jj]);
110  ov1.random();
111 
112  const double zz = dot_product(ov1, ov1);
113  EXPECT(zz > zero);
114 
115  ObsVector_ ov2(ov1);
116  ov2.zero();
117 
118  EXPECT(dot_product(ov2, ov1) == zero);
119  EXPECT(dot_product(ov2, ov2) == zero);
120 
121  ObsVector_ ov3(ov1);
122  ov3.ones();
123 
124  EXPECT(dot_product(ov3, ov3) == ov3.nobs());
125  }
126 }
127 // -----------------------------------------------------------------------------
128 
129 template <typename OBS> void testLinearAlgebra() {
130  typedef ObsTestsFixture<OBS> Test_;
131  typedef oops::ObsVector<OBS> ObsVector_;
132  const double tolerance = 1.0e-8;
133  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
134  ObsVector_ ov1(Test_::obspace()[jj]);
135  ov1.random();
136 
137  // test *=, += and -=
138  ObsVector_ ov2(ov1);
139  ov2 += ov1;
140  ov1 *= 2.0;
141  ov2 -= ov1;
142  EXPECT(dot_product(ov2, ov2) < tolerance);
143 
144  // test =
145  ObsVector_ ov3(ov1);
146  ov2 = ov1;
147  ov2 -= ov3;
148  EXPECT(dot_product(ov2, ov2) < tolerance);
149 
150  // test *=(const ObsVector &) and /=(const ObsVector &)
151  ov2 = ov1;
152  ov2 *= ov1;
153  ov2 /= ov1;
154  ov2 -= ov1;
155  EXPECT(dot_product(ov2, ov2) < tolerance);
156 
157  // test axpy
158  ov2 = ov1;
159  ov3 = ov1;
160  ov2.axpy(2.0, ov1);
161  ov3 *= 3;
162  ov2 -= ov3;
163  EXPECT(dot_product(ov2, ov2) < tolerance);
164 
165  // test invert()
166  ov2 = ov1;
167  ov2.invert();
168  ov2 *= ov1;
169  EXPECT(std::abs(dot_product(ov2, ov2) - ov2.nobs()) < tolerance);
170  }
171 }
172 
173 // -----------------------------------------------------------------------------
174 template <typename OBS> void testReadWrite() {
175  typedef ObsTestsFixture<OBS> Test_;
176  typedef oops::ObsVector<OBS> ObsVector_;
177  const double tolerance = 1.0e-8;
178  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
179  ObsVector_ ov1(Test_::obspace()[jj]);
180  ov1.random();
181 
182  ov1.save("test");
183  ObsVector_ ov2(Test_::obspace()[jj], "test");
184  ov2 -= ov1;
185  EXPECT(dot_product(ov2, ov2) < tolerance);
186  }
187 }
188 // -----------------------------------------------------------------------------
189 /// \brief Tests ObsVector::mask, ObsVector::packEigen and
190 /// ObsVector::packEigenSize methods.
191 /// \details Tests that:
192 /// - mask of all zeros (nothing to mask) applied to ObsVector doesn't change
193 /// its size and content;
194 /// - mask of either all ones (if "mask variable" isn't specified in yaml), or
195 /// from the file applied to ObsVector changes its size.
196 /// - linear algebra operations with ObsVector that were masked out produce
197 /// ObsVectors that have the same number of obs masked out.
198 /// - size returned by packEigenSize is consistent with size of Eigen Vector
199 /// returned by packEigen, and is the same as reference value for each MPI
200 /// task.
201 template <typename OBS> void testMask() {
202  typedef ObsTestsFixture<OBS> Test_;
203  typedef oops::ObsDataVector<OBS, int> ObsDataVector_;
204  typedef oops::ObsSpace<OBS> ObsSpace_;
205  typedef oops::ObsVector<OBS> ObsVector_;
206 
207  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
208  const ObsSpace_ & obspace = Test_::obspace()[jj];
209 
210  ObsVector_ reference(obspace);
211  reference.random();
212  oops::Log::test() << "ObsVector before masking: " << reference << std::endl;
213 
214  const size_t nobs_all = Test_::config(jj).getInt("reference global nobs");
215  EXPECT_EQUAL(reference.nobs(), nobs_all);
216  EXPECT(nobs_all > 0);
217 
218  /// apply empty mask, check that vector is the same
219  ObsDataVector_ unsetmask(obspace, obspace.obsvariables());
220  unsetmask.zero();
221  ObsVector_ with_unsetmask(reference);
222  with_unsetmask.mask(unsetmask);
223  oops::Log::test() << "ObsVector masked with all-zero-mask: " << with_unsetmask << std::endl;
224  EXPECT_EQUAL(with_unsetmask.nobs(), nobs_all);
225  with_unsetmask -= reference;
226  EXPECT_EQUAL(with_unsetmask.rms(), 0.0);
227 
228  /// apply non-empty mask, check number of observations
229  std::string maskvarname;
230  /// by default (else statement below), apply mask that masks out everything
231  size_t nobs_after_mask = 0;
232  std::vector<size_t> nobs_after_mask_local(Test_::comm().size(), 0);
233  /// if mask variable is available, apply mask from file and read reference number of masked obs
234  if (Test_::config(jj).has("mask variable")) {
235  maskvarname = Test_::config(jj).getString("mask variable");
236  nobs_after_mask = Test_::config(jj).getUnsigned("reference global masked nobs");
237  nobs_after_mask_local = Test_::config(jj).getUnsignedVector("reference local masked nobs");
238  // check that specified mask masks out something
239  EXPECT_NOT_EQUAL(nobs_after_mask, nobs_all);
240  // check that "reference local masked nobs" are defined for all MPI tasks
241  EXPECT_EQUAL(Test_::comm().size(), nobs_after_mask_local.size());
242  /// if mask variable is unavailable, apply mask with all ones
243  } else {
244  // Hack for mask with ones: use ObsVector set to ones, write to ObsSpace,
245  // then read as ObsDataVector.
246  maskvarname = "set_mask";
247  ObsVector_ tmp(obspace);
248  tmp.ones();
249  tmp.save(maskvarname);
250  }
251  ObsDataVector_ mask(obspace, obspace.obsvariables(), maskvarname);
252  ObsVector_ with_mask(reference);
253  with_mask.mask(mask);
254  oops::Log::test() << "ObsVector masked with " << maskvarname << " mask: " <<
255  with_mask << std::endl;
256  EXPECT_EQUAL(with_mask.nobs(), nobs_after_mask);
257 
258  /// Test that various linear algebra operations with masked out ObsVector
259  /// produce masked out ObsVector
260  /// Test invert()
261  ObsVector_ test(with_mask);
262  test.invert();
263  EXPECT_EQUAL(test.nobs(), nobs_after_mask);
264 
265  /// Test *=(float)
266  test *= 2.0;
267  EXPECT_EQUAL(test.nobs(), nobs_after_mask);
268 
269  /// Test +=(ObsVector)
270  test.random();
271  EXPECT_EQUAL(test.nobs(), nobs_all);
272  test += with_mask;
273  EXPECT_EQUAL(test.nobs(), nobs_after_mask);
274  test.random();
275  EXPECT_EQUAL(test.nobs(), nobs_all);
276  with_mask += test;
277  EXPECT_EQUAL(with_mask.nobs(), nobs_after_mask);
278 
279  /// Test -=(ObsVector)
280  EXPECT_EQUAL(test.nobs(), nobs_all);
281  test -= with_mask;
282  EXPECT_EQUAL(test.nobs(), nobs_after_mask);
283  test.random();
284  EXPECT_EQUAL(test.nobs(), nobs_all);
285  with_mask -= test;
286  EXPECT_EQUAL(with_mask.nobs(), nobs_after_mask);
287 
288  /// Test *=(ObsVector)
289  EXPECT_EQUAL(test.nobs(), nobs_all);
290  test *= with_mask;
291  EXPECT_EQUAL(test.nobs(), nobs_after_mask);
292  test.random();
293  EXPECT_EQUAL(test.nobs(), nobs_all);
294  with_mask *= test;
295  EXPECT_EQUAL(with_mask.nobs(), nobs_after_mask);
296 
297  /// Test /=(ObsVector)
298  EXPECT_EQUAL(test.nobs(), nobs_all);
299  test /= with_mask;
300  EXPECT_EQUAL(test.nobs(), nobs_after_mask);
301  test.random();
302  EXPECT_EQUAL(test.nobs(), nobs_all);
303  with_mask /= test;
304  EXPECT_EQUAL(with_mask.nobs(), nobs_after_mask);
305 
306  /// Test axpy
307  EXPECT_EQUAL(test.nobs(), nobs_all);
308  test.axpy(2.0, with_mask);
309  EXPECT_EQUAL(test.nobs(), nobs_after_mask);
310  test.random();
311  EXPECT_EQUAL(test.nobs(), nobs_all);
312  with_mask.axpy(2.0, test);
313  EXPECT_EQUAL(with_mask.nobs(), nobs_after_mask);
314 
315  /// test packEigen
316  // create maskvec ObsVector - mask with missing values
317  ObsVector_ maskvec = test;
318  maskvec.ones();
319  maskvec.mask(mask);
320  // randomize the vector, and call packEigen with maskvec
321  test.random();
322  Eigen::VectorXd with_mask_vec = test.packEigen(maskvec);
323  // check that the size of returned Eigen Vector is consistent with size
324  // returned by packEigenSize()
325  EXPECT_EQUAL(with_mask_vec.size(), test.packEigenSize(maskvec));
326  oops::Log::debug() << "Local number of masked observations is: " <<
327  with_mask_vec.size() << std::endl;
328  // check that the size is consistent with reference for this MPI task
329  EXPECT_EQUAL(with_mask_vec.size(), nobs_after_mask_local[Test_::comm().rank()]);
330  }
331 }
332 // -----------------------------------------------------------------------------
333 
334 template <typename OBS>
335 class ObsVector : public oops::Test {
337 
338  public:
340  virtual ~ObsVector() {}
341 
342  private:
343  std::string testid() const override {return "test::ObsVector<" + OBS::name() + ">";}
344 
345  void register_tests() const override {
346  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
347 
348  ts.emplace_back(CASE("interface/ObsVector/testConstructor")
349  { testConstructor<OBS>(); });
350  ts.emplace_back(CASE("interface/ObsVector/testCopyConstructor")
351  { testCopyConstructor<OBS>(); });
352  ts.emplace_back(CASE("interface/ObsVector/testNotZero")
353  { testNotZero<OBS>(); });
354  ts.emplace_back(CASE("interface/ObsVector/testLinearAlgebra")
355  { testLinearAlgebra<OBS>(); });
356  ts.emplace_back(CASE("interface/ObsVector/testReadWrite")
357  { testReadWrite<OBS>(); });
358  ts.emplace_back(CASE("interface/ObsVector/testMask")
359  { testMask<OBS>(); });
360  }
361 
362  void clear() const override {
363  Test_::reset();
364  }
365 };
366 
367 // =============================================================================
368 
369 } // namespace test
370 
371 #endif // TEST_INTERFACE_OBSVECTOR_H_
ObsDataVector is a vector templated on data type, in the observation space.
ObsVector class used in oops; subclass of interface class interface::ObsVector.
std::string testid() const override
void clear() const override
ObsTestsFixture< OBS > Test_
void register_tests() const override
logical function has(this, var)
void testLinearAlgebra()
void testCopyConstructor()
void testMask()
Tests ObsVector::mask, ObsVector::packEigen and ObsVector::packEigenSize methods.
void testWrappingConstructor()
Test the constructor taking a std::unique_ptr<OBS::ObsVector>.
CASE("test_linearmodelparameterswrapper_valid_name")
void testConstructor()
Tests creation and destruction of ObsErrorCovariances.