OOPS
InterpolationInterface.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2019 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 
9 #ifndef TEST_GENERIC_INTERPOLATIONINTERFACE_H_
10 #define TEST_GENERIC_INTERPOLATIONINTERFACE_H_
11 
12 #include <cmath>
13 #include <iostream>
14 #include <limits>
15 #include <memory>
16 #include <string>
17 #include <vector>
18 
19 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
20 
21 #include <boost/noncopyable.hpp>
22 
23 #include "atlas/array.h"
24 #include "atlas/field.h"
25 #include "atlas/functionspace.h"
26 #include "atlas/functionspace/PointCloud.h"
27 #include "atlas/grid.h"
28 #include "atlas/mesh.h"
29 #include "atlas/meshgenerator.h"
30 #include "atlas/option.h"
31 #include "atlas/util/Point.h"
32 #include "eckit/config/LocalConfiguration.h"
33 #include "eckit/mpi/Comm.h"
34 #include "eckit/testing/Test.h"
35 
37 #include "oops/mpi/mpi.h"
38 #include "oops/runs/Test.h"
39 #include "oops/util/Logger.h"
40 #include "oops/util/Random.h"
41 #include "test/TestEnvironment.h"
42 
43 using atlas::array::make_view;
44 using atlas::option::halo;
45 using atlas::option::levels;
46 using atlas::option::name;
47 
48 namespace test {
49 
50 // -----------------------------------------------------------------------------
51 /*! smooth function for testing interpolation */
52 double testfunc(double lon, double lat, std::size_t jlev = 1,
53  std::size_t nlev = 1) {
54  double deg2rad = M_PI / 180.0L;
55  double zz = static_cast<double>(jlev+1)/static_cast<double>(nlev);
56  return (sin(deg2rad*lon)*cos(deg2rad*lat) + zz)/1.5;
57 }
58 
59 // -----------------------------------------------------------------------------
60 /*! Test C++ interface to interpolation implementations
61  *
62 */
63 
65  const eckit::Configuration &topConf = ::test::TestEnvironment::config();
66  typedef oops::InterpolatorBase Interpolator_;
67  typedef oops::InterpolatorFactory InterpolatorFactory_;
68 
69  // Skip this test if it's not specified in the yaml
70  if (!topConf.has("test interpolation interface"))
71  return;
72 
73  eckit::LocalConfiguration config = topConf.getSubConfiguration("test interpolation interface");
74 
75  // Default to atlas interpolator
76  std::string intname = "atlas";
77  if (config.has("interpolator"))
78  intname = config.getString("interpolator");
79 
80  oops::Log::info() << "\n----------------------------------------------------"
81  << "\nStarting test of oops interface for interpolator "
82  << intname
83  << "\n----------------------------------------------------"
84  << std::endl;
85 
86  // For testing use a regular Octahedral lat/lon grid
87  atlas::Grid grid1("O64");
88 
89  // use 3 vertical levels for testing unless otherwise specified
90  std::size_t nlev = 3;
91  if (config.has("nlevels"))
92  nlev = static_cast<std::size_t>(config.getInt("nlevels"));
93  else
94  config.set("nlevels", nlev);
95 
96  // Generate mesh and functionspace for input grid
97  atlas::MeshGenerator meshgen("structured");
98  atlas::Mesh mesh1 = meshgen.generate(grid1);
99 
100  // Some interpolators like bump will want to set up their own communication
101  // patterns so the input field should be set up with no halo.
102  // Others that use atlas mesh will want atlas to generate the halo.
103 
104  atlas::functionspace::NodeColumns fs1;
105  if (config.has("nhalo")) {
106  int nhalo = config.getInt("nhalo");
107  fs1 = atlas::functionspace::NodeColumns(mesh1, levels(nlev) | halo(nhalo));
108  } else {
109  fs1 = atlas::functionspace::NodeColumns(mesh1, levels(nlev));
110  }
111 
112  // Generate random point cloud as output grid
113  const std::size_t N = static_cast<size_t>(config.getInt("nrandom"));
114  unsigned int seed = static_cast<unsigned int>(config.getInt("seed"));
115 
116  // Distribute output longitudes over MPI tasks
117  unsigned int check_no_obs = 0;
118  if (config.has("check no obs")) {
119  check_no_obs = static_cast<unsigned int>(config.getInt("check no obs"));
120  }
121  size_t ntasks = oops::mpi::world().size();
122  if (check_no_obs == 1) {
123  ntasks = ntasks-1;
124  }
125  size_t myrank = oops::mpi::world().rank();
126 
127  // number of output points on this mpi task
128  size_t myN = 0;
129  if (myrank < ntasks) {
130  myN = N/ntasks;
131  if (myrank < (N % ntasks)) myN += 1;
132  }
133 
134  // longitude range assigned to this mpi task
135  double mylon_min = myrank * 360.0/static_cast<double>(ntasks);
136  double mylon_max = (myrank+1) * 360.0/static_cast<double>(ntasks);
137 
138  // task-dependent random seed
139  unsigned int myseed = (myrank+1)*seed;
140 
141  util::UniformDistribution<double> lon(myN, mylon_min, mylon_max, myseed);
142  util::UniformDistribution<double> lat(myN, -88.0, 88.0);
143  lon.sort();
144 
145  atlas::PointXY point;
146  std::vector<atlas::PointXY> gridpoints;
147  for (std::size_t jj=0; jj < myN; ++jj) {
148  point.assign(lon[jj], lat[jj]);
149  gridpoints.push_back(point);
150  }
151 
152  atlas::functionspace::PointCloud fs2(gridpoints);
153 
154  std::unique_ptr<Interpolator_>
155  interpolator(InterpolatorFactory_::create(config, fs1, fs2));
156 
157  // Test print method
158  oops::Log::info() << "Interpolator created:\n" << *interpolator << std::endl;
159 
160  // Next - define the input fields
161  atlas::Field field1 = fs1.createField<double>(name("testfield"));
162 
163  // now fill in the input field with a smooth test function
164  auto lonlat = make_view<double, 2>(fs1.nodes().lonlat());
165  auto infield = make_view<double, 2>(field1);
166 
167  for (size_t jnode = 0; jnode < static_cast<size_t>(fs1.nodes().size()); ++jnode) {
168  for (size_t jlev = 0; jlev < static_cast<size_t>(fs1.levels()); ++jlev)
169  infield(jnode, jlev) = testfunc(lonlat(jnode, 0), lonlat(jnode, 1), jlev, nlev);
170  }
171 
172  // Store the input and output fields in a FieldSet to check the interface
173  // You can pass the interpolator an empty output field set and it will
174  // allocate the output fields, compute them, and add them to the field set
175  atlas::FieldSet infields, outfields;
176  infields.add(field1);
177 
178  oops::Log::info() << "\n----------------------------------------------------"
179  << "\nTesting " << intname << " interpolation"
180  << "\n----------------------------------------------------"
181  << std::endl;
182 
183  // apply interpolation
184  interpolator->apply(infields, outfields);
185 
186  // get test tolerance
187  const double tolerance = config.getDouble("tolerance");
188 
189  // check result from interpolation
190  auto lonlat2 = make_view<double, 2>(fs2.lonlat());
191  auto outfield = make_view<double, 2>(outfields.field("testfield"));
192 
193  std::size_t jlev = 2; // level for checking result
194  if (jlev > nlev-1)
195  jlev = nlev-1;
196 
197  for (size_t jnode = 0; jnode < static_cast<size_t>(fs2.size()); ++jnode) {
198  EXPECT(oops::is_close(outfield(jnode, jlev),
199  testfunc(lonlat2(jnode, 0), lonlat2(jnode, 1), jlev, nlev), tolerance));
200  }
201 
202  oops::Log::info() << "\n----------------------------------------------------"
203  << "\nRepeat for single field"
204  << "\n----------------------------------------------------"
205  << std::endl;
206 
207  // define output field
208  atlas::Field field2 = fs2.createField<double>(name("testoutput") |
209  levels(field1.levels()));
210 
211  // apply interpolation
212  interpolator->apply(field1, field2);
213 
214  // check result from interpolation
215  auto outfield2 = make_view<double, 2>(field2);
216 
217  for (size_t jnode = 0; jnode < static_cast<size_t>(fs2.size()); ++jnode) {
218  EXPECT(oops::is_close(outfield2(jnode, jlev),
219  testfunc(lonlat2(jnode, 0), lonlat2(jnode, 1), jlev, nlev), tolerance));
220  }
221 
222  // --------------------------------------------------
223  /// Now test the adjoint. But, skip this test for interpolators like atlas
224  /// that have not yet implemented the adjoint
225 
226  bool adjoint_test = true;
227  if (config.has("adjoint_test"))
228  adjoint_test = config.getBool("adjoint_test");
229 
230  if (!adjoint_test) {
231  oops::Log::info() << "\n----------------------------------------------------"
232  << "\nSkipping adjoint test for interpolator " << intname
233  << "\n----------------------------------------------------"
234  << std::endl;
235  } else {
236  oops::Log::info() << "\n----------------------------------------------------"
237  << "\nTesting adjoint for interpolator " << intname
238  << "\n----------------------------------------------------"
239  << std::endl;
240 
241  // define random field on output grid
242  size_t nout = fs2.size()*nlev;
243 
244  util::UniformDistribution<double> x(nout, 0.0, 1.0);
245 
246  atlas::Field field2_adcheck = fs2.createField<double>(name("adcheck")|levels(nlev));
247  auto adcheck2 = make_view<double, 2>(field2_adcheck);
248 
249  size_t idx = 0;
250  for (size_t jnode = 0; jnode < static_cast<size_t>(fs2.size()); ++jnode) {
251  for (size_t jlev = 0; jlev < static_cast<size_t>(nlev); ++jlev) {
252  adcheck2(jnode, jlev) = x[idx];
253  ++idx;
254  }
255  }
256  atlas::FieldSet adcheck_grid2;
257  adcheck_grid2.add(field2_adcheck);
258 
259  // Define empty FieldSet for the result (on grid1)
260  atlas::FieldSet adcheck_grid1;
261 
262  // Apply interpolator adjoint
263  interpolator->apply_ad(adcheck_grid2, adcheck_grid1);
264 
265  // Check adjoint
266 
267  // we need to implement a general dot product for atlas Fields.
268  // For now, do it manually.
269 
270  double dot1 = 0;
271  double mydot1 = 0;
272  atlas::Field field1_adcheck = adcheck_grid1.field("adcheck");
273  auto adcheck1 = make_view<double, 2>(field1_adcheck);
274  atlas::mesh::IsGhostNode is_ghost(fs1.nodes());
275  for (size_t jnode = 0; jnode < static_cast<size_t>(fs1.size()); ++jnode) {
276  if (is_ghost(jnode) == 0) {
277  for (size_t jlev = 0; jlev < nlev; ++jlev)
278  mydot1 += infield(jnode, jlev)*adcheck1(jnode, jlev);
279  }
280  }
281  oops::mpi::world().allReduce(mydot1, dot1, eckit::mpi::sum());
282 
283  double dot2 = 0;
284  double mydot2 = 0;
285  for (size_t jnode = 0; jnode < static_cast<size_t>(fs2.size()); ++jnode) {
286  for (size_t jlev = 0; jlev < nlev; ++jlev)
287  mydot2 += outfield(jnode, jlev)*adcheck2(jnode, jlev);
288  }
289  oops::mpi::world().allReduce(mydot2, dot2, eckit::mpi::sum());
290 
291  EXPECT(oops::is_close(dot1, dot2, tolerance));
292  }
293 
294  // --------------------------------------------------
295  // Now test writing and reading the weights to a file.
296  // Not all interpolators have this capability so only do it if requested
297 
298  bool readwrite_test = false;
299  if (config.has("readwrite_test"))
300  readwrite_test = config.getBool("readwrite_test");
301 
302  if (!readwrite_test) {
303  oops::Log::info() << "\n----------------------------------------------------"
304  << "\nSkipping read/write test for interpolator " << intname
305  << "\n----------------------------------------------------"
306  << std::endl;
307  } else {
308  oops::Log::info() << "\n----------------------------------------------------"
309  << "\nTesting read/write for interpolator " << intname
310  << "\n----------------------------------------------------"
311  << std::endl;
312 
313  if (interpolator->write(config) != 0)
314  throw eckit::NotImplemented("Write method not implemented", Here());
315 
316  // read interpolator back as a new object
317  config.set("read_from_file", true);
318  std::string infile = config.getString("outfile");
319  config.set("infile", infile);
320 
321  std::unique_ptr<Interpolator_>
322  interpolator_read(InterpolatorFactory_::create(config, fs1, fs2));
323 
324  // test by applying this interpolator to our previous infield
325  atlas::Field field2_read = fs2.createField<double>(name("testoutput") |
326  levels(field1.levels()));
327 
328  interpolator->apply(field1, field2_read);
329  auto outfield2_read = make_view<double, 2>(field2_read);
330 
331  for (size_t jnode = 0; jnode < static_cast<size_t>(fs2.size()); ++jnode) {
332  EXPECT(oops::is_close(outfield2_read(jnode, jlev),
333  testfunc(lonlat2(jnode, 0), lonlat2(jnode, 1), jlev, nlev), tolerance));
334  }
335  }
336 
337  oops::Log::info() << "\n----------------------------------------------------"
338  << "\nFinishing test of oops interface for interpolator "
339  << intname
340  << "\n----------------------------------------------------"
341  << std::endl;
342 }
343 
344 // -----------------------------------------------------------------------------
345 /// Oops interpolation interface test
346 ///
348  public:
351 
352  private:
353  std::string testid() const override {return "test::InterpolationInterface";}
354 
355  void register_tests() const override {
356  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
357 
358  ts.emplace_back(CASE("generic/InterpolationInterface/testInterpolation")
359  { testInterpolation(); });
360  }
361 
362  void clear() const override {}
363 };
364 
365 // -----------------------------------------------------------------------------
366 } // namespace test
367 
368 #endif // TEST_GENERIC_INTERPOLATIONINTERFACE_H_
Base class for Generic interpolation.
InterpolatorFactory: Factory for creating Interpolator objects.
void register_tests() const override
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
double testfunc(double lon, double lat, std::size_t jlev=1, std::size_t nlev=1)
smooth function for testing interpolation
void testInterpolation()
Test C++ interface to interpolation implementations.
CASE("test_linearmodelparameterswrapper_valid_name")