OOPS
test/interface/GetValues.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020 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_GETVALUES_H_
9 #define TEST_INTERFACE_GETVALUES_H_
10 
11 #include <algorithm>
12 #include <cmath>
13 #include <iomanip>
14 #include <iostream>
15 #include <limits>
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 #include <boost/ptr_container/ptr_vector.hpp>
24 
25 #include "eckit/config/LocalConfiguration.h"
26 #include "eckit/testing/Test.h"
27 #include "oops/base/Geometry.h"
28 #include "oops/base/State.h"
29 #include "oops/base/Variables.h"
31 #include "oops/interface/GeoVaLs.h"
34 #include "oops/mpi/mpi.h"
35 #include "oops/runs/Test.h"
36 #include "oops/util/DateTime.h"
37 #include "oops/util/dot_product.h"
38 #include "oops/util/Duration.h"
39 #include "test/TestEnvironment.h"
40 
41 namespace test {
42 
43 // =================================================================================================
44 
45 template <typename MODEL, typename OBS> class GetValuesFixture : private boost::noncopyable {
46  typedef eckit::LocalConfiguration LocalConfig_;
52  typedef util::DateTime DateTime_;
53 
54  public:
55  static const DateTime_ & timebeg() {return *getInstance().timebeg_;}
56  static const DateTime_ & timeend() {return *getInstance().timeend_;}
57  static const GeoVaLs_ & geovals() {return *getInstance().geovals_;}
58  static const Geometry_ & resol() {return *getInstance().resol_;}
59  static const GetValues_ & getvalues() {return *getInstance().getvalues_;}
60  static const LocalConfig_ & testconf() {return *getInstance().testconf_;}
61  static const Locations_ & locs() {return *getInstance().locs_;}
62  static const Variables_ & geovalvars() {return *getInstance().geovalvars_;}
63  static const std::vector<size_t> & geovalvarsizes() {return getInstance().geovalvarsizes_;}
64  static void reset() {
65  getInstance().getvalues_.reset();
66  getInstance().geovals_.reset();
67  getInstance().timeend_.reset();
68  getInstance().timebeg_.reset();
69  getInstance().locs_.reset();
70  getInstance().geovalvars_.reset();
71  getInstance().resol_.reset();
72  getInstance().testconf_.reset();
73  }
74 
75  private:
77  static GetValuesFixture<MODEL, OBS> theGetValuesFixture;
78  return theGetValuesFixture;
79  }
80 
82  testconf_.reset(new LocalConfig_(TestEnvironment::config(), "getvalues test"));
83 
84  // Geometry
85  const LocalConfig_ resolConfig(TestEnvironment::config(), "geometry");
86  resol_.reset(new Geometry_(resolConfig, oops::mpi::world()));
87 
88  // Variables
89  geovalvars_.reset(new Variables_(TestEnvironment::config(), "state variables"));
90  geovalvarsizes_ = resol_->variableSizes(*geovalvars_);
91 
92  // Locations
93  const LocalConfig_ locsConfig(TestEnvironment::config(), "locations");
94  locs_.reset(new Locations_(locsConfig, oops::mpi::world()));
95 
96  // Window times
97  timebeg_.reset(new DateTime_(locsConfig.getString("window begin")));
98  timeend_.reset(new DateTime_(locsConfig.getString("window end")));
99 
100  // GeoVaLs
102 
103  // GetValues
104  LocalConfig_ getvaluesConfig;
105  if (TestEnvironment::config().has("get values"))
106  getvaluesConfig = eckit::LocalConfiguration(TestEnvironment::config(), "get values");
107  getvalues_.reset(new GetValues_( *resol_, *locs_, getvaluesConfig)); }
108 
109  ~GetValuesFixture<MODEL, OBS>() {}
110 
111  std::unique_ptr<const DateTime_> timebeg_;
112  std::unique_ptr<const DateTime_> timeend_;
113  std::unique_ptr<const GeoVaLs_> geovals_;
114  std::unique_ptr<const Geometry_> resol_;
115  std::unique_ptr<const GetValues_> getvalues_;
116  std::unique_ptr<const LocalConfig_> testconf_;
117  std::unique_ptr<const Locations_> locs_;
118  std::unique_ptr<const Variables_> geovalvars_;
119  std::vector<size_t> geovalvarsizes_;
120 };
121 
122 // =================================================================================================
123 /// \brief tests constructor and print method
124 template <typename MODEL, typename OBS> void testGetValuesConstructor() {
125  typedef GetValuesFixture<MODEL, OBS> Test_;
126  typedef oops::GetValues<MODEL, OBS> GetValues_;
127 
128  std::unique_ptr<const GetValues_>
129  GetValues(new GetValues_(Test_::resol(), Test_::locs(), TestEnvironment::config()));
130 
131  EXPECT(GetValues.get());
132  oops::Log::test() << "Testing GetValues: " << *GetValues << std::endl;
133  GetValues.reset();
134  EXPECT(!GetValues.get());
135 }
136 
137 // -------------------------------------------------------------------------------------------------
138 
139 template <typename MODEL, typename OBS> void testGetValuesMultiWindow() {
140  typedef GetValuesFixture<MODEL, OBS> Test_;
141  typedef oops::GeoVaLs<OBS> GeoVaLs_;
142  typedef oops::State<MODEL> State_;
143 
144  const util::Duration windowlength = Test_::timeend() - Test_::timebeg();
145  const util::DateTime timemid = Test_::timebeg() + windowlength/2;
146 
147  const eckit::LocalConfiguration confgen(Test_::testconf(), "state generate");
148  const State_ xx(Test_::resol(), confgen);
149 
150  EXPECT(xx.norm() > 0.0);
151 
152  GeoVaLs_ gv1(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
153  GeoVaLs_ gv2(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
154 
155  // Compute all geovals together
156  Test_::getvalues().fillGeoVaLs(xx, Test_::timebeg(), Test_::timeend(), gv1);
157 
158  // Compute all geovals as two subwindows
159  Test_::getvalues().fillGeoVaLs(xx, Test_::timebeg(), timemid, gv2);
160  Test_::getvalues().fillGeoVaLs(xx, timemid, Test_::timeend(), gv2);
161 
162  EXPECT(gv1.rms() > 0.0);
163  EXPECT(gv2.rms() > 0.0);
164  EXPECT(gv1.rms() == gv2.rms());
165 }
166 
167 // -------------------------------------------------------------------------------------------------
168 
169 /*! \brief Interpolation test
170  *
171  * \details **testGetValuesInterpolation()** tests the creation of an
172  * analytic state for a given model. The conceptual steps are as follows:
173  * 1. Initialize the JEDI State object based on idealized analytic formulae
174  * 2. Interpolate the State variables onto selected "observation" locations
175  * using the getValues() method of the State object. The result is
176  * placed in a JEDI GeoVaLs object
177  * 3. Compute the correct solution by applying the analytic formulae directly
178  * at the observation locations.
179  * 4. Assess the accuracy of the interpolation by comparing the interpolated
180  * values from Step 2 with the exact values from Step 3
181  *
182  * The interpolated state values are compared to the analytic solution for
183  * a series of **locations** which includes values optionally specified by the
184  * user in the "state test" section of the config file in addition to a
185  * randomly-generated list of **nrandom** random locations. nrandom is also
186  * specified by the user in the "state test" section of the config file, as is the
187  * (nondimensional) tolerence level (**interpolation tolerance**) to be used for the tests.
188  *
189  * Relevant parameters in the **State* section of the config file include
190  *
191  * * **norm-gen** Normalization test for the generated State
192  * * **interpolation tolerance** tolerance for the interpolation test
193  *
194  * \date April, 2018: M. Miesch (JCSDA) adapted a preliminary version in the
195  * feature/interp branch
196  *
197  * \warning Since this model compares the interpolated state values to an exact analytic
198  * solution, it requires that the "analytic_init" option be implemented in the model and
199  * selected in the "state.state generate" section of the config file.
200  */
201 
202 template <typename MODEL, typename OBS> void testGetValuesInterpolation() {
203  typedef GetValuesFixture<MODEL, OBS> Test_;
204  typedef oops::AnalyticInit<OBS> AnalyticInit_;
205  typedef oops::State<MODEL> State_;
206  typedef oops::GeoVaLs<OBS> GeoVaLs_;
207 
208  const eckit::LocalConfiguration confgen(Test_::testconf(), "state generate");
209  const State_ xx(Test_::resol(), confgen);
210 
211  // Interpolation tolerance
212  double interp_tol = Test_::testconf().getDouble("interpolation tolerance");
213 
214  // Ceate a GeoVaLs object from locs and vars
215  GeoVaLs_ gval(Test_::locs(), Test_::geovalvars(), Test_::geovalvarsizes());
216 
217  EXPECT(xx.norm() > 0.0);
218 
219  // Execute the interpolation
220  Test_::getvalues().fillGeoVaLs(xx, Test_::timebeg(), Test_::timeend(), gval);
221 
222  EXPECT(gval.rms() > 0.0);
223  oops::Log::debug() << "RMS GeoVaLs: " << gval.rms() << std::endl;
224 
225  // Now create another GeoVaLs object that contains the exact analytic solutions.
226  GeoVaLs_ ref(gval);
227 
228  AnalyticInit_ init(confgen);
229  init.fillGeoVaLs(Test_::locs(), ref);
230 
231  EXPECT(ref.rms() > 0.0);
232  oops::Log::debug() << "RMS reference GeoVaLs: " << ref.rms() << std::endl;
233 
234  // Compute the difference between the interpolated and exact values
235  gval -= ref;
236 
237  // And check to see if the errors are within specified tolerance
238  oops::Log::test() << "Normalized rms of the difference: " << gval.normalizedrms(ref) << std::endl;
239  EXPECT(gval.normalizedrms(ref) < interp_tol);
240 }
241 
242 // =================================================================================================
243 
244 template <typename MODEL, typename OBS>
245 class GetValues : public oops::Test {
246  public:
249 
250  private:
251  std::string testid() const override {return "test::GetValues<" + MODEL::name() +
252  ", " + 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/GetValues/testGetValuesConstructor")
258  { testGetValuesConstructor<MODEL, OBS>(); });
259  ts.emplace_back(CASE("interface/GetValues/testGetValuesMultiWindow")
260  { testGetValuesMultiWindow<MODEL, OBS>(); });
261  ts.emplace_back(CASE("interface/GetValues/testGetValuesInterpolation")
262  { testGetValuesInterpolation<MODEL, OBS>(); });
263  }
264 
265  void clear() const override {}
266 };
267 
268 // =================================================================================================
269 
270 } // namespace test
271 
272 #endif // TEST_INTERFACE_GETVALUES_H_
Initializes GeoVaLs with analytic formula.
Geometry class used in oops; subclass of interface class interface::Geometry.
Gets values from model State to observation locations (fills GeoVaLs)
Locations of observations for observation operator.
State class used in oops; subclass of interface class interface::State.
static const GetValues_ & getvalues()
std::unique_ptr< const GeoVaLs_ > geovals_
std::unique_ptr< const LocalConfig_ > testconf_
static const Variables_ & geovalvars()
std::unique_ptr< const Geometry_ > resol_
static const Geometry_ & resol()
oops::Locations< OBS > Locations_
static const Locations_ & locs()
std::unique_ptr< const DateTime_ > timeend_
static const GeoVaLs_ & geovals()
std::vector< size_t > geovalvarsizes_
static const DateTime_ & timebeg()
static const DateTime_ & timeend()
oops::Geometry< MODEL > Geometry_
static const std::vector< size_t > & geovalvarsizes()
eckit::LocalConfiguration LocalConfig_
std::unique_ptr< const GetValues_ > getvalues_
std::unique_ptr< const Variables_ > geovalvars_
std::unique_ptr< const DateTime_ > timebeg_
oops::GetValues< MODEL, OBS > GetValues_
static const LocalConfig_ & testconf()
std::unique_ptr< const Locations_ > locs_
static GetValuesFixture< MODEL, OBS > & getInstance()
oops::GeoVaLs< OBS > GeoVaLs_
void register_tests() const override
std::string testid() const override
void clear() 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
logical function has(this, var)
void testGetValuesMultiWindow()
void testGetValuesInterpolation()
Interpolation test.
void testGetValuesConstructor()
tests constructor and print method
CASE("test_linearmodelparameterswrapper_valid_name")