OOPS
test/interface/ObsOperator.h
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2017-2018 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_OBSOPERATOR_H_
9 #define TEST_INTERFACE_OBSOPERATOR_H_
10 
11 #include <memory>
12 #include <string>
13 #include <vector>
14 
15 #define ECKIT_TESTING_SELF_REGISTER_CASES 0
16 
17 #include <boost/noncopyable.hpp>
18 
19 #include "eckit/config/LocalConfiguration.h"
20 #include "eckit/testing/Test.h"
21 #include "oops/base/ObsVector.h"
22 #include "oops/base/Variables.h"
23 #include "oops/interface/GeoVaLs.h"
28 #include "oops/runs/Test.h"
29 #include "oops/util/Expect.h"
31 #include "test/TestEnvironment.h"
32 
33 namespace test {
34 
35 const char *expectConstructorToThrow = "expect constructor to throw exception with message";
36 const char *expectSimulateObsToThrow = "expect simulateObs to throw exception with message";
37 
38 // -----------------------------------------------------------------------------
39 /// \brief tests constructor and print method
40 template <typename OBS> void testConstructor() {
41  typedef ObsTestsFixture<OBS> Test_;
42  typedef oops::ObsOperator<OBS> ObsOperator_;
43  typedef typename ObsOperator_::Parameters_ ObsOperatorParameters_;
44 
45  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
46  eckit::LocalConfiguration obsopconf(Test_::config(jj), "obs operator");
47  ObsOperatorParameters_ obsopparams;
48  obsopparams.validateAndDeserialize(obsopconf);
49 
50  if (!Test_::config(jj).has(expectConstructorToThrow)) {
51  std::unique_ptr<ObsOperator_> hop(new ObsOperator_(Test_::obspace()[jj], obsopparams));
52  EXPECT(hop.get());
53  oops::Log::test() << "Testing ObsOperator: " << *hop << std::endl;
54  hop.reset();
55  EXPECT(!hop.get());
56  } else {
57  // The constructor is expected to throw an exception containing the specified string.
58  const std::string expectedMessage = Test_::config(jj).getString(expectConstructorToThrow);
59  EXPECT_THROWS_MSG(ObsOperator_(Test_::obspace()[jj], obsopparams),
60  expectedMessage.c_str());
61  }
62  }
63 }
64 
65 // -----------------------------------------------------------------------------
66 
67 template <typename OBS> void testSimulateObs() {
68  typedef ObsTestsFixture<OBS> Test_;
69  typedef oops::GeoVaLs<OBS> GeoVaLs_;
70  typedef oops::ObsDiagnostics<OBS> ObsDiags_;
71  typedef oops::ObsAuxControl<OBS> ObsAuxCtrl_;
72  typedef oops::ObsOperator<OBS> ObsOperator_;
73  typedef typename ObsOperator_::Parameters_ ObsOperatorParameters_;
74  typedef oops::ObsVector<OBS> ObsVector_;
75 
76  for (std::size_t jj = 0; jj < Test_::obspace().size(); ++jj) {
77  const eckit::LocalConfiguration & conf = Test_::config(jj);
78  if (conf.has(expectConstructorToThrow))
79  continue;
80 
81  // initialize observation operator (set variables requested from the model,
82  // variables simulated by the observation operator, other init)
83  eckit::LocalConfiguration obsopconf(conf, "obs operator");
84  ObsOperatorParameters_ obsopparams;
85  obsopparams.validateAndDeserialize(obsopconf);
86  ObsOperator_ hop(Test_::obspace()[jj], obsopparams);
87 
88  // initialize bias correction
89  eckit::LocalConfiguration biasconf = conf.getSubConfiguration("obs bias");
90  typename ObsAuxCtrl_::Parameters_ biasparams;
91  biasparams.validateAndDeserialize(biasconf);
92  const ObsAuxCtrl_ ybias(Test_::obspace()[jj], biasparams);
93 
94  // read geovals from the file
95  eckit::LocalConfiguration gconf(conf, "geovals");
96  oops::Variables hopvars = hop.requiredVars();
97  hopvars += ybias.requiredVars();
98  const GeoVaLs_ gval(gconf, Test_::obspace()[jj], hopvars);
99 
100  // create obsvector to hold H(x)
101  ObsVector_ hofx(Test_::obspace()[jj]);
102 
103  // create obsvector to hold bias
104  ObsVector_ bias(Test_::obspace()[jj]);
105  bias.zero();
106 
107  // create diagnostics to hold HofX diags
108  oops::Variables diagvars;
109  diagvars += ybias.requiredHdiagnostics();
110  ObsDiags_ diags(Test_::obspace()[jj], hop.locations(), diagvars);
111 
112  // call H(x), save result in the output file as @hofx
113  if (Test_::config(jj).has(expectSimulateObsToThrow)) {
114  // The simulateObs method is expected to throw an exception
115  // containing the specified string.
116  const std::string expectedMessage =
117  Test_::config(jj).getString(expectSimulateObsToThrow);
118  EXPECT_THROWS_MSG(hop.simulateObs(gval, hofx, ybias, bias, diags),
119  expectedMessage.c_str());
120  continue;
121  } else {
122  hop.simulateObs(gval, hofx, ybias, bias, diags);
123  }
124  hofx.save("hofx");
125 
126  const double tol = conf.getDouble("tolerance");
127  if (conf.has("vector ref")) {
128  // if reference h(x) is saved in file as a vector, read from file
129  // and compare the norm of difference to zero
130  ObsVector_ obsref(Test_::obspace()[jj], conf.getString("vector ref"));
131  obsref -= hofx;
132  const double zz = obsref.rms();
133  oops::Log::test() << "Vector difference between reference and computed: " << obsref;
134  EXPECT(zz < 100*tol); // change tol from percent to actual value.
135  // tol used in is_close is relative
136  } else if (conf.has("norm ref")) {
137  // if reference h(x) is saved in file as a vector, read from file
138  // and compare the difference, normalised by the reference values to zero
139  ObsVector_ obsref(Test_::obspace()[jj], conf.getString("norm ref"));
140  obsref -= hofx;
141  obsref /= hofx;
142  const double zz = obsref.rms();
143  oops::Log::test() << "Normalised vector difference between reference and computed: "
144  << obsref;
145  EXPECT(zz < 100*tol); // change tol from percent to actual value.
146  // tol used in is_close is relative
147  } else {
148  // else compare h(x) norm to the norm from the config
149  const double zz = hofx.rms();
150  const double xx = conf.getDouble("rms ref");
151 
152  oops::Log::debug() << "zz: " << std::fixed << std::setprecision(8) << zz << std::endl;
153  oops::Log::debug() << "xx: " << std::fixed << std::setprecision(8) << xx << std::endl;
154 
155  EXPECT(oops::is_close(xx, zz, tol));
156  }
157  }
158 }
159 
160 // -----------------------------------------------------------------------------
161 
162 template <typename OBS>
163 class ObsOperator : public oops::Test {
165  public:
167  virtual ~ObsOperator() {}
168  private:
169  std::string testid() const override {return "test::ObsOperator<" + OBS::name() + ">";}
170 
171  void register_tests() const override {
172  std::vector<eckit::testing::Test>& ts = eckit::testing::specification();
173 
174  ts.emplace_back(CASE("interface/ObsOperator/testConstructor")
175  { testConstructor<OBS>(); });
176  ts.emplace_back(CASE("interface/ObsOperator/testSimulateObs")
177  { testSimulateObs<OBS>(); });
178  }
179 
180  void clear() const override {
181  Test_::reset();
182  }
183 };
184 
185 // =============================================================================
186 
187 } // namespace test
188 
189 #endif // TEST_INTERFACE_OBSOPERATOR_H_
Auxiliary state related to observations, templated on <OBS>
MODEL-agnostic part of nonlinear observation (forward) operator. The full nonlinear observation opera...
ObsVector class used in oops; subclass of interface class interface::ObsVector.
void clear() const override
void register_tests() const override
ObsTestsFixture< OBS > Test_
std::string testid() const override
logical function has(this, var)
const char * expectConstructorToThrow
const char * expectSimulateObsToThrow
CASE("test_linearmodelparameterswrapper_valid_name")
void testConstructor()
Tests creation and destruction of ObsErrorCovariances.