IODA
ObsVectorPackEigen.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2021- UCAR.
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  */
7 
8 #ifndef TEST_IODA_OBSVECTORPACKEIGEN_H_
9 #define TEST_IODA_OBSVECTORPACKEIGEN_H_
10 
11 #include <Eigen/Dense>
12 
13 #include <string>
14 #include <vector>
15 
16 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
17 
18 #include "eckit/config/LocalConfiguration.h"
19 #include "eckit/testing/Test.h"
20 
21 #include "oops/mpi/mpi.h"
22 #include "oops/runs/Test.h"
23 #include "oops/test/TestEnvironment.h"
24 #include "oops/util/Logger.h"
25 
26 #include "ioda/ObsDataVector.h"
27 #include "ioda/ObsSpace.h"
28 #include "ioda/ObsVector.h"
29 
30 namespace ioda {
31 namespace test {
32 
33 // -----------------------------------------------------------------------------
34 
35 /// \brief tests ObsVector::packEigen, packEigenSize methods and mask methods
36 /// \details Tests that:
37 /// - number of local masked obs returned by ObsVector::packEigenSize is the same
38 /// as reference in yaml (reference local masked nobs);
39 /// - norm of Eigen::VectorXd returned by ObsVector::packEigen is close to the
40 /// reference specified in yaml (reference local masked norm);
41 /// - norm of a random vector with mask applied is different from the same vector
42 /// before mask application;
43 /// - norm of a random vector with mask(ObsDataVector<int>) applied is the same as
44 /// norm of the same vector with mask(ObsVector).
45 void testPackEigen() {
46  std::vector<eckit::LocalConfiguration> conf;
47  ::test::TestEnvironment::config().get("observations", conf);
48  util::DateTime bgn((::test::TestEnvironment::config().getString("window begin")));
49  util::DateTime end((::test::TestEnvironment::config().getString("window end")));
50 
51  for (std::size_t jj = 0; jj < conf.size(); ++jj) {
52  eckit::LocalConfiguration obsconf(conf[jj], "obs space");
54  obsparams.validateAndDeserialize(obsconf);
55  ioda::ObsSpace obsdb(obsparams, oops::mpi::world(), bgn, end, oops::mpi::myself());
56 
57  const size_t rank = obsdb.distribution()->rank();
58  ioda::ObsVector obsvec(obsdb, "ObsValue");
59 
60  // test packEigenSize
61  const std::string maskname = conf[jj].getString("mask variable");
62  ioda::ObsDataVector<int> mask(obsdb, obsdb.obsvariables(), maskname);
63  ioda::ObsVector maskvector(obsdb);
64  maskvector.mask(mask);
65  const size_t size = obsvec.packEigenSize(maskvector);
66  const std::vector<size_t> ref_sizes =
67  conf[jj].getUnsignedVector("reference local masked nobs");
68  EXPECT_EQUAL(size, ref_sizes[rank]);
69 
70  // test packEigen
71  Eigen::VectorXd packed = obsvec.packEigen(maskvector);
72  const std::vector<double> ref_norms =
73  conf[jj].getDoubleVector("reference local masked norm");
74  EXPECT(oops::is_close(packed.norm(), ref_norms[rank], 1.e-5));
75 
76  // test mask
77  ioda::ObsVector vec1(obsdb);
78  vec1.random();
79  ioda::ObsVector vec2 = vec1;
80  vec1.mask(mask);
81  EXPECT_NOT_EQUAL(vec1.rms(), vec2.rms());
82  vec2.mask(maskvector);
83  EXPECT_EQUAL(vec1.rms(), vec2.rms());
84  }
85 }
86 
87 // -----------------------------------------------------------------------------
88 
89 class ObsVectorPackEigen : public oops::Test {
90  public:
91  ObsVectorPackEigen() = default;
92  virtual ~ObsVectorPackEigen() = default;
93 
94  private:
95  std::string testid() const override {return "test::ObsVector<ioda::IodaTrait>";}
96 
97  void register_tests() const override {
98  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
99 
100  ts.emplace_back(CASE("ioda/ObsVectorPackEigen/testPackEigen")
101  { testPackEigen(); });
102  }
103 
104  void clear() const override {}
105 };
106 
107 // =============================================================================
108 
109 } // namespace test
110 } // namespace ioda
111 
112 #endif // TEST_IODA_OBSVECTORPACKEIGEN_H_
ObsDataVector<DATATYPE> handles vectors of data of type DATATYPE in observation space.
ObsVector class to handle vectors in observation space for IODA.
Definition: src/ObsVector.h:34
void mask(const ObsDataVector< int > &mask)
Set this ObsVector values to missing where mask is non-zero.
Definition: ObsVector.cc:281
double rms() const
Definition: ObsVector.cc:186
Eigen::VectorXd packEigen(const ObsVector &mask) const
Definition: ObsVector.cc:245
size_t packEigenSize(const ObsVector &mask) const
Definition: ObsVector.cc:235
std::string testid() const override
void register_tests() const override
virtual ~ObsVectorPackEigen()=default
CASE("Derived variable, unit conversion, and exception checking methods")
void testPackEigen()
tests ObsVector::packEigen, packEigenSize methods and mask methods