OOPS
test/interface/ObsLocalization.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_INTERFACE_OBSLOCALIZATION_H_
9 #define TEST_INTERFACE_OBSLOCALIZATION_H_
10 
11 #include <cmath>
12 #include <memory>
13 #include <string>
14 #include <vector>
15 
16 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
17 
18 #include <boost/noncopyable.hpp>
19 
20 #include "eckit/config/LocalConfiguration.h"
21 #include "eckit/testing/Test.h"
22 #include "oops/base/Geometry.h"
24 #include "oops/base/ObsVector.h"
26 #include "oops/mpi/mpi.h"
27 #include "oops/runs/Test.h"
29 #include "test/TestEnvironment.h"
30 
31 namespace test {
32 
33 /// \brief Tests ObsLocalization::computeLocalization method.
34 /// \details Tests that for obs localization around specified in yaml Geometry points:
35 /// 1. number of local obs matches reference ("reference local nobs")
36 /// 2. obs localization applied to a vector of ones makes rms(obsvec) < 1 or
37 /// doesn't change the vector (depending on the "localization reduces values" option)
38 /// Reference gridpoints are specified in yaml as "reference gridpoints.lons" and
39 /// "reference gridpoints.lats". They don't have to be exactly equal to the lon/lat
40 /// of the Geometry gridpoints, but should be no further than 1.e-5 distance away.
41 template <typename MODEL, typename OBS> void testObsLocalization() {
42  typedef ObsTestsFixture<OBS> Test_;
43  typedef oops::Geometry<MODEL> Geometry_;
44  typedef oops::GeometryIterator<MODEL> GeometryIterator_;
45  typedef oops::ObsLocalizationBase<MODEL, OBS> ObsLocalization_;
46  typedef oops::ObsSpace<OBS> ObsSpace_;
47  typedef oops::ObsVector<OBS> ObsVector_;
48 
49  const eckit::LocalConfiguration geometryConfig(TestEnvironment::config(), "geometry");
50  Geometry_ geometry(geometryConfig, oops::mpi::world());
51 
52  // loop over all obs spaces
53  for (size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
54  const ObsSpace_ & obspace = Test_::obspace()[jj];
55  // initialize obs-space localization
56  eckit::LocalConfiguration locconf(Test_::config(jj), "obs localization");
57 
58  // read reference local nobs values and reference gridpoints
59  const std::vector<double> lons = locconf.getDoubleVector("reference gridpoints.lons");
60  const std::vector<double> lats = locconf.getDoubleVector("reference gridpoints.lats");
61  const std::vector<size_t> nobs_local_ref = locconf.getUnsignedVector("reference local nobs");
62  ASSERT(lons.size() == lats.size());
63  ASSERT(lons.size() == nobs_local_ref.size());
64  ASSERT(lons.size() > 0);
65  std::vector<eckit::geometry::Point2> reference_points;
66  for (size_t jpoint = 0; jpoint < lons.size(); ++jpoint) {
67  reference_points.emplace_back(lons[jpoint], lats[jpoint]);
68  }
69  std::unique_ptr<ObsLocalization_> obsloc =
71  oops::Log::test() << "Testing obs-space localization: " << *obsloc << std::endl;
72 
73  ObsVector_ locvector(obspace);
74  ObsVector_ obsvector(obspace);
75 
76  size_t total_tested = 0;
77  std::vector<size_t> nobs_local(nobs_local_ref.size(), 0);
78  std::vector<double> locvector_rms(nobs_local_ref.size(), 0);
79  // loop over geometry points
80  for (GeometryIterator_ ii = geometry.begin(); ii != geometry.end(); ++ii) {
81  // debug print to help decide which points to specify for reference
82  // set OOPS_DEBUG environment variable to -1 to see prints from all MPI tasks
83  if (locconf.getBool("print iterator", false)) {
84  oops::Log::debug() << "Iterating over " << std::setprecision(9) << ii << ": "
85  << *ii << std::endl;
86  }
87  // check if we need to test at this location (if there are any points in the
88  // reference point list within 1e-5 of this locationn)
89  const auto & it = std::find_if(reference_points.begin(), reference_points.end(),
90  [ii] (const eckit::geometry::Point2 & point) {return point.distance(*ii) < 1e-5;});
91  if (it != reference_points.end()) {
92  total_tested++;
93  size_t index = it - reference_points.begin();
94  locvector.ones();
95  obsloc->computeLocalization(ii, locvector);
96  oops::Log::test() << "Obs localization with geometry iterator: " << ii << ": "
97  << *ii << std::endl;
98  oops::Log::test() << "Localization values: " << locvector << std::endl;
99  oops::Log::test() << "Local vector nobs and reference: " << locvector.nobs() << ", "
100  << nobs_local_ref[index] << std::endl;
101  oops::Log::debug() << "Local vector stats lat,lon,nobs,rms: " << *ii << ", "
102  << locvector.nobs() << ", " << locvector.rms() << std::endl;
103 
104  // save number of local obs to be tested later
105  nobs_local[index] = locvector.nobs();
106 
107  // apply localization to a vector of ones
108  obsvector.ones();
109  obsvector *= locvector;
110  oops::Log::test() << "Localization applied to local ObsVector of ones: " <<
111  obsvector << std::endl;
112  // save localized vector rms to be tested later
113  locvector_rms[index] = obsvector.rms();
114  }
115  }
116  // gather number of tested observations, nobs_local and locvector_rms
117  // (only one MPI task owns one gridpoint)
118  Test_::comm().allReduceInPlace(total_tested, eckit::mpi::sum());
119  Test_::comm().allReduceInPlace(nobs_local.begin(), nobs_local.end(), eckit::mpi::sum());
120  Test_::comm().allReduceInPlace(locvector_rms.begin(), locvector_rms.end(), eckit::mpi::sum());
121  // check that we tested all gridpoints
122  EXPECT_EQUAL(total_tested, lons.size());
123  // Test that computed number of local obs is the same as reference
124  EXPECT_EQUAL(nobs_local_ref, nobs_local);
125  // check value of the rms is close to reference
126  const std::vector<double> ref_locvector_rms = locconf.getDoubleVector("reference rms");
127  ASSERT(lons.size() == ref_locvector_rms.size());
128  oops::Log::debug() << "reference RMS" << ref_locvector_rms << std::endl;
129  oops::Log::debug() << "computed RMS" << locvector_rms << std::endl;
130  for (size_t jpoint = 0; jpoint < nobs_local.size(); ++jpoint) {
131  if (nobs_local[jpoint] > 0) {
132  EXPECT(std::abs(locvector_rms[jpoint]-ref_locvector_rms[jpoint]) < 1.e-5);
133  }
134  }
135  }
136 }
137 
138 // -----------------------------------------------------------------------------
139 
140 template <typename MODEL, typename OBS> class ObsLocalization : public oops::Test {
142 
143  public:
144  ObsLocalization() = default;
145  virtual ~ObsLocalization() = default;
146 
147  private:
148  std::string testid() const override {return "test::ObsLocalization<" + MODEL::name() + ","
149  + OBS::name() + ">";}
150 
151  void register_tests() const override {
152  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
153 
154  ts.emplace_back(CASE("interface/ObsLocalization/testObsLocalization")
155  { testObsLocalization<MODEL, OBS>(); });
156  }
157 
158  void clear() const override {
159  Test_::reset();
160  }
161 };
162 
163 // -----------------------------------------------------------------------------
164 
165 } // namespace test
166 
167 #endif // TEST_INTERFACE_OBSLOCALIZATION_H_
Geometry class used in oops; subclass of interface class interface::Geometry.
static std::unique_ptr< ObsLocalizationBase< MODEL, OBS > > create(const eckit::Configuration &, const ObsSpace_ &)
ObsVector class used in oops; subclass of interface class interface::ObsVector.
virtual ~ObsLocalization()=default
std::string testid() const override
static const eckit::Configuration & config()
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
Definition: oops/mpi/mpi.cc:84
void testObsLocalization()
Tests ObsLocalization::computeLocalization method.
CASE("test_linearmodelparameterswrapper_valid_name")