23 #include "eckit/config/LocalConfiguration.h"
24 #include "eckit/exception/Exceptions.h"
28 #include "oops/util/abor1_cpp.h"
29 #include "oops/util/DateTime.h"
30 #include "oops/util/Duration.h"
31 #include "oops/util/Logger.h"
32 #include "oops/util/missingValues.h"
33 #include "oops/util/Random.h"
34 #include "oops/util/stringFunctions.h"
36 namespace sf = util::stringfunctions;
43 const util::DateTime & bgn,
const util::DateTime & end,
44 const eckit::mpi::Comm & timeComm)
45 :
oops::ObsSpaceBase(params, comm, bgn, end), winbgn_(bgn), winend_(end), comm_(timeComm),
48 oops::Log::trace() <<
"ObsTable::ObsTable starting" << std::endl;
49 if (params.
obsdatain.value() != boost::none) {
54 if (params.
generate.value() != boost::none) {
57 if (params.
obsdataout.value() != boost::none) {
59 sf::swapNameMember(params.toConfiguration(),
nameOut_);
61 oops::Log::trace() <<
"ObsTable::ObsTable created nobs = " <<
nobs() << std::endl;
67 oops::Log::trace() <<
"ObsTable::ObsTable destructed" << std::endl;
84 void ObsTable::putdb(
const std::string & col,
const std::vector<int> & vec)
const {
85 std::vector<double> tmp(vec.size());
86 int intmiss = util::missingValue(
int());
87 double doublemiss = util::missingValue(
double());
88 for (
size_t jobs = 0; jobs < vec.size(); ++jobs) {
89 if (vec[jobs] == intmiss) {
90 tmp[jobs] = doublemiss;
92 tmp[jobs] =
static_cast<double>(vec[jobs]);
95 this->
putdb(col, tmp);
101 std::vector<double> tmp(vec.size());
102 float floatmiss = util::missingValue(
float());
103 double doublemiss = util::missingValue(
double());
104 for (
size_t jobs = 0; jobs < vec.size(); ++jobs) {
105 if (vec[jobs] == floatmiss) {
106 tmp[jobs] = doublemiss;
108 tmp[jobs] =
static_cast<double>(vec[jobs]);
111 this->
putdb(col, tmp);
116 void ObsTable::putdb(
const std::string & col,
const std::vector<double> & vec)
const {
117 ASSERT(vec.size() ==
nobs());
119 oops::Log::info() <<
"ObsTable::putdb over-writing " << col << std::endl;
122 data_.insert(std::pair<std::string, std::vector<double> >(col, vec));
129 std::vector<double> tmp;
130 this->
getdb(col, tmp);
131 int intmiss = util::missingValue(
int());
132 double doublemiss = util::missingValue(
double());
134 for (
size_t jobs = 0; jobs <
nobs(); ++jobs) {
135 if (tmp[jobs] == doublemiss) {
138 vec[jobs] = lround(tmp[jobs]);
146 std::vector<double> tmp;
147 this->
getdb(col, tmp);
148 float floatmiss = util::missingValue(
float());
149 double doublemiss = util::missingValue(
double());
151 for (
size_t jobs = 0; jobs <
nobs(); ++jobs) {
152 if (tmp[jobs] == doublemiss) {
153 vec[jobs] = floatmiss;
155 vec[jobs] =
static_cast<float>(tmp[jobs]);
163 std::map<std::string, std::vector<double> >::const_iterator ic =
data_.find(col);
164 if (ic ==
data_.end()) {
165 oops::Log::error() <<
"ObsTable::getdb " << col <<
" not found." << std::endl;
166 ABORT(
"ObsTable::getdb column not found");
169 for (
unsigned int jobs = 0; jobs <
nobs(); ++jobs) {
170 vec[jobs] = ic->second[jobs];
177 oops::Log::trace() <<
"ObsTable::generateDistribution starting" << std::endl;
179 const util::Duration &first = params.
begin;
181 if (params.
end.value() != boost::none) {
182 last = *params.
end.value();
187 util::Duration step(first);
188 while (step <= last) {
193 const unsigned int nobs_locations = params.
obsDensity;
194 const unsigned int nobs = nobs_locations*nobstimes;
195 double dx = 1.0/
static_cast<double>(nobs_locations);
200 unsigned int iobs = 0;
203 while (step <= last) {
205 for (
unsigned int jobs = 0; jobs < nobs_locations; ++jobs) {
206 double xpos = jobs*dx;
213 ASSERT(iobs ==
nobs);
217 std::vector<double> obserr(
nobs);
218 for (
unsigned int jj = 0; jj <
nobs; ++jj) {
221 this->
putdb(
"ObsError", obserr);
223 oops::Log::trace() <<
"ObsTable::generateDistribution done, nobs= " <<
nobs << std::endl;
229 util::NormalDistribution<double> x(data.size(), 0.0, 1.0,
getSeed());
230 for (
size_t jj = 0; jj < data.size(); ++jj) data[jj] = x[jj];
238 oops::Log::trace() <<
"ObsTable::ot_read starting" << std::endl;
239 std::ifstream fin(filename.c_str());
240 if (!fin.is_open()) ABORT(
"ObsTable::otOpen: Error opening file: " + filename);
245 std::vector<std::string> colnames;
246 for (
int jc = 0; jc < ncol; ++jc) {
249 colnames.push_back(col);
254 std::vector<double> newcol;
255 for (
int jc = 0; jc < ncol; ++jc) {
256 ASSERT(
data_.find(colnames[jc]) ==
data_.end());
257 data_.insert(std::pair<std::string, std::vector<double> >(colnames[jc], newcol));
261 for (
int jobs = 0; jobs <
nobs; ++jobs) {
267 util::DateTime ttt(sss);
270 if (inside)
times_.push_back(ttt);
274 for (std::map<std::string, std::vector<double> >::iterator jo =
data_.begin();
275 jo !=
data_.end(); ++jo) {
278 if (inside) jo->second.push_back(
val);
283 oops::Log::trace() <<
"ObsTable::ot_read done" << std::endl;
289 oops::Log::trace() <<
"ObsTable::otWrite writing " << filename << std::endl;
291 const size_t ioproc = 0;
294 if (
comm_.size() > 1)
comm_.allReduceInPlace(
nobs, eckit::mpi::Operation::SUM);
296 std::vector<util::DateTime> timebuff(
nobs);
299 std::vector<double> locbuff(
nobs);
302 std::vector<double> datasend(
times_.size() *
data_.size());
304 for (
size_t jobs = 0; jobs <
times_.size(); ++jobs) {
305 for (
auto jo =
data_.begin(); jo !=
data_.end(); ++jo) {
306 datasend[iobs] = jo->second[jobs];
310 std::vector<double> databuff(
data_.size() *
nobs);
313 if (
comm_.rank() == ioproc) {
314 std::ofstream fout(filename.c_str());
315 if (!fout.is_open()) ABORT(
"ObsTable::otWrite: Error opening file: " + filename);
317 int ncol =
data_.size();
318 fout << ncol << std::endl;
320 for (
auto jo =
data_.begin(); jo !=
data_.end(); ++jo)
321 fout << jo->first << std::endl;
323 fout <<
nobs << std::endl;
326 for (
int jobs = 0; jobs <
nobs; ++jobs) {
328 fout <<
" " << timebuff[jobs];
329 fout <<
" " << locbuff[jobs];
330 for (
int jcol = 0; jcol < ncol; ++jcol) {
331 fout <<
" " << databuff[iii];
340 oops::Log::trace() <<
"ObsTable::otWrite done" << std::endl;
354 os <<
"ObsTable: assimilation window = " <<
winbgn_ <<
" to " <<
winend_ << std::endl;
355 os <<
"ObsTable: file in = " <<
nameIn_ <<
", file out = " <<
nameOut_;
Options controlling generation of artificial observations.
oops::RequiredParameter< util::Duration > begin
oops::RequiredParameter< util::Duration > obsFrequency
oops::RequiredParameter< int > obsDensity
Number of observations to generate in each time slot.
oops::OptionalParameter< util::Duration > end
oops::RequiredParameter< double > obsError
Iterator over all observations.
void random(std::vector< double > &) const
std::vector< double > locations_
void print(std::ostream &) const
bool has(const std::string &col) const
void getdb(const std::string &, std::vector< int > &) const
unsigned int nobs() const
void otWrite(const std::string &) const
const eckit::mpi::Comm & comm_
ObsIterator begin() const
iterator to the first observation
const util::DateTime winbgn_
void generateDistribution(const ObsGenerateParameters ¶ms)
const util::DateTime winend_
std::map< std::string, std::vector< double > > data_
ObsIterator end() const
iterator to the observation past-the-last
std::vector< util::DateTime > times_
void otOpen(const std::string &)
void putdb(const std::string &, const std::vector< int > &) const
ObsTable(const Parameters_ &, const eckit::mpi::Comm &, const util::DateTime &, const util::DateTime &, const eckit::mpi::Comm &)
Configuration parameters for the L95 model's ObsSpace.
oops::OptionalParameter< std::string > obsdatain
File from which to load observations.
oops::OptionalParameter< std::string > obsdataout
File to which to save observations and analysis.
oops::OptionalParameter< ObsGenerateParameters > generate
Options controlling generation of artificial observations.
The namespace for the L95 model.
void gather(const eckit::mpi::Comm &comm, const std::vector< double > &send, std::vector< double > &recv, const size_t root)
The namespace for the main oops code.