9 #ifndef TEST_GENERIC_INTERPOLATIONINTERFACE_H_
10 #define TEST_GENERIC_INTERPOLATIONINTERFACE_H_
19 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
21 #include <boost/noncopyable.hpp>
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"
39 #include "oops/util/Logger.h"
40 #include "oops/util/Random.h"
43 using atlas::array::make_view;
44 using atlas::option::halo;
45 using atlas::option::levels;
46 using atlas::option::name;
52 double testfunc(
double lon,
double lat, std::size_t jlev = 1,
53 std::size_t nlev = 1) {
55 double zz =
static_cast<double>(jlev+1)/
static_cast<double>(nlev);
70 if (!topConf.has(
"test interpolation interface"))
73 eckit::LocalConfiguration config = topConf.getSubConfiguration(
"test interpolation interface");
76 std::string intname =
"atlas";
77 if (config.has(
"interpolator"))
78 intname = config.getString(
"interpolator");
80 oops::Log::info() <<
"\n----------------------------------------------------"
81 <<
"\nStarting test of oops interface for interpolator "
83 <<
"\n----------------------------------------------------"
87 atlas::Grid grid1(
"O64");
91 if (config.has(
"nlevels"))
92 nlev =
static_cast<std::size_t
>(config.getInt(
"nlevels"));
94 config.set(
"nlevels", nlev);
97 atlas::MeshGenerator meshgen(
"structured");
98 atlas::Mesh mesh1 = meshgen.generate(grid1);
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));
109 fs1 = atlas::functionspace::NodeColumns(mesh1, levels(nlev));
113 const std::size_t N =
static_cast<size_t>(config.getInt(
"nrandom"));
114 unsigned int seed =
static_cast<unsigned int>(config.getInt(
"seed"));
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"));
122 if (check_no_obs == 1) {
129 if (myrank < ntasks) {
131 if (myrank < (N % ntasks)) myN += 1;
135 double mylon_min = myrank * 360.0/
static_cast<double>(ntasks);
136 double mylon_max = (myrank+1) * 360.0/
static_cast<double>(ntasks);
139 unsigned int myseed = (myrank+1)*seed;
141 util::UniformDistribution<double> lon(myN, mylon_min, mylon_max, myseed);
142 util::UniformDistribution<double> lat(myN, -88.0, 88.0);
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);
152 atlas::functionspace::PointCloud fs2(gridpoints);
154 std::unique_ptr<Interpolator_>
155 interpolator(InterpolatorFactory_::create(config, fs1, fs2));
158 oops::Log::info() <<
"Interpolator created:\n" << *interpolator << std::endl;
161 atlas::Field field1 = fs1.createField<
double>(name(
"testfield"));
164 auto lonlat = make_view<double, 2>(fs1.nodes().lonlat());
165 auto infield = make_view<double, 2>(field1);
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);
175 atlas::FieldSet infields, outfields;
176 infields.add(field1);
178 oops::Log::info() <<
"\n----------------------------------------------------"
179 <<
"\nTesting " << intname <<
" interpolation"
180 <<
"\n----------------------------------------------------"
184 interpolator->apply(infields, outfields);
187 const double tolerance = config.getDouble(
"tolerance");
190 auto lonlat2 = make_view<double, 2>(fs2.lonlat());
191 auto outfield = make_view<double, 2>(outfields.field(
"testfield"));
193 std::size_t jlev = 2;
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));
202 oops::Log::info() <<
"\n----------------------------------------------------"
203 <<
"\nRepeat for single field"
204 <<
"\n----------------------------------------------------"
208 atlas::Field field2 = fs2.createField<
double>(name(
"testoutput") |
209 levels(field1.levels()));
212 interpolator->apply(field1, field2);
215 auto outfield2 = make_view<double, 2>(field2);
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));
226 bool adjoint_test =
true;
227 if (config.has(
"adjoint_test"))
228 adjoint_test = config.getBool(
"adjoint_test");
231 oops::Log::info() <<
"\n----------------------------------------------------"
232 <<
"\nSkipping adjoint test for interpolator " << intname
233 <<
"\n----------------------------------------------------"
236 oops::Log::info() <<
"\n----------------------------------------------------"
237 <<
"\nTesting adjoint for interpolator " << intname
238 <<
"\n----------------------------------------------------"
242 size_t nout = fs2.size()*nlev;
244 util::UniformDistribution<double> x(nout, 0.0, 1.0);
246 atlas::Field field2_adcheck = fs2.createField<
double>(name(
"adcheck")|levels(nlev));
247 auto adcheck2 = make_view<double, 2>(field2_adcheck);
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];
256 atlas::FieldSet adcheck_grid2;
257 adcheck_grid2.add(field2_adcheck);
260 atlas::FieldSet adcheck_grid1;
263 interpolator->apply_ad(adcheck_grid2, adcheck_grid1);
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);
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);
291 EXPECT(oops::is_close(dot1, dot2, tolerance));
298 bool readwrite_test =
false;
299 if (config.has(
"readwrite_test"))
300 readwrite_test = config.getBool(
"readwrite_test");
302 if (!readwrite_test) {
303 oops::Log::info() <<
"\n----------------------------------------------------"
304 <<
"\nSkipping read/write test for interpolator " << intname
305 <<
"\n----------------------------------------------------"
308 oops::Log::info() <<
"\n----------------------------------------------------"
309 <<
"\nTesting read/write for interpolator " << intname
310 <<
"\n----------------------------------------------------"
313 if (interpolator->write(config) != 0)
314 throw eckit::NotImplemented(
"Write method not implemented", Here());
317 config.set(
"read_from_file",
true);
318 std::string infile = config.getString(
"outfile");
319 config.set(
"infile", infile);
321 std::unique_ptr<Interpolator_>
322 interpolator_read(InterpolatorFactory_::create(config, fs1, fs2));
325 atlas::Field field2_read = fs2.createField<
double>(name(
"testoutput") |
326 levels(field1.levels()));
328 interpolator->apply(field1, field2_read);
329 auto outfield2_read = make_view<double, 2>(field2_read);
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));
337 oops::Log::info() <<
"\n----------------------------------------------------"
338 <<
"\nFinishing test of oops interface for interpolator "
340 <<
"\n----------------------------------------------------"
353 std::string
testid()
const override {
return "test::InterpolationInterface";}
356 std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
358 ts.emplace_back(
CASE(
"generic/InterpolationInterface/testInterpolation")
Base class for Generic interpolation.
InterpolatorFactory: Factory for creating Interpolator objects.
void register_tests() const override
virtual ~InterpolationInterface()
std::string testid() const override
void clear() const override
static const eckit::Configuration & config()
real(8), parameter deg2rad
const eckit::mpi::Comm & world()
Default communicator with all MPI tasks (ie MPI_COMM_WORLD)
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")