Reading and decoding bulletins
#include <wreport/bulletin.h>
void read_bufr_raw(const Options& opts, const char* fname, RawHandler& handler)
{
FILE* in = fopen(fname, "rb");
if (in == NULL)
try {
std::string raw_data;
off_t offset;
handler.handle_raw_bufr(raw_data, fname, offset);
fclose(in);
} catch (...) {
fclose(in);
throw;
}
}
void read_crex_raw(const Options& opts, const char* fname, RawHandler& handler)
{
FILE* in = fopen(fname, "rt");
if (in == NULL)
try {
string raw_data;
off_t offset;
handler.handle_raw_crex(raw_data, fname, offset);
fclose(in);
} catch (...) {
fclose(in);
throw;
}
}
Creating bulletins
#include <wreport/bulletin.h>
#include <cstring>
void do_makebuoy()
{
unique_ptr<BufrBulletin> bulletin(BufrBulletin::create());
bulletin->edition_number = 4;
bulletin->data_category = 1;
bulletin->data_subcategory = 21;
bulletin->data_subcategory_local = 255;
bulletin->rep_year = 2011;
bulletin->rep_month = 10;
bulletin->rep_day = 3;
bulletin->rep_hour = 17;
bulletin->rep_minute = 0;
bulletin->rep_second = 0;
bulletin->originating_centre = 98;
bulletin->originating_subcentre = 0;
bulletin->master_table_version_number = 14;
bulletin->master_table_version_number_local = 0;
bulletin->compression = false;
bulletin->optional_section = "test";
bulletin->datadesc.push_back(
WR_VAR(3, 8, 3));
bulletin->load_tables();
Subset& s = bulletin->obtain_subset(0);
s.store_variable_i(
WR_VAR(0, 1, 5), 65602);
s.store_variable_d(
WR_VAR(0, 1, 12), 12.0);
s.store_variable_d(
WR_VAR(0, 1, 13), 0.2);
s.store_variable_i(
WR_VAR(0, 2, 1), 0);
s.store_variable_i(
WR_VAR(0, 4, 1), 2011);
s.store_variable_i(
WR_VAR(0, 4, 2), 10);
s.store_variable_i(
WR_VAR(0, 4, 3), 3);
s.store_variable_i(
WR_VAR(0, 4, 4), 17);
s.store_variable_i(
WR_VAR(0, 4, 5), 0);
s.store_variable_d(
WR_VAR(0, 5, 2), 59.03);
s.store_variable_d(
WR_VAR(0, 6, 2), -2.99);
s.store_variable_d(
WR_VAR(0, 10, 4), 99520.0);
s.store_variable_d(
WR_VAR(0, 10, 51), 99520.0);
s.store_variable_d(
WR_VAR(0, 10, 61), 310);
s.store_variable_i(
WR_VAR(0, 10, 63), 7);
s.store_variable_undef(
WR_VAR(0, 11, 11));
s.store_variable_undef(
WR_VAR(0, 11, 12));
s.store_variable_d(
WR_VAR(0, 12, 4), 278.5);
s.store_variable_undef(
WR_VAR(0, 12, 6));
s.store_variable_undef(
WR_VAR(0, 13, 3));
s.store_variable_undef(
WR_VAR(0, 20, 1));
s.store_variable_undef(
WR_VAR(0, 20, 3));
s.store_variable_undef(
WR_VAR(0, 20, 4));
s.store_variable_undef(
WR_VAR(0, 20, 5));
s.store_variable_undef(
WR_VAR(0, 20, 10));
s.store_variable_undef(
WR_VAR(0, 8, 2));
s.store_variable_undef(
WR_VAR(0, 20, 11));
s.store_variable_undef(
WR_VAR(0, 20, 13));
s.store_variable_undef(
WR_VAR(0, 20, 12));
s.store_variable_undef(
WR_VAR(0, 20, 12));
s.store_variable_undef(
WR_VAR(0, 20, 12));
s.store_variable_d(
WR_VAR(0, 22, 42), 280.8);
string encoded = bulletin->encode();
if (fwrite(encoded.data(), encoded.size(), 1, stdout) != 1)
perror("cannot write BUFR to standard output");
}
Printing bulletin contents
#include <wreport/bulletin.h>
#include <wreport/bulletin/dds-scanfeatures.h>
#include <cstring>
struct PrintContents : public BulletinFullHandler
{
FILE* out;
PrintContents(FILE* out=stderr) : out(out) {}
{
}
};
struct PrintTrace : public BulletinFullHandler
{
FILE* out;
PrintTrace(FILE* out=stderr) : out(out) {}
void handle_raw_bufr(const std::string& raw_data, const char* fname, long offset) override
{
try {
handle(*bulletin);
} catch (std::exception& e) {
fprintf(stderr, "%s:%ld:%s\n", fname, offset, e.what());
}
}
};
struct PrintStructure : public BulletinFullHandler
{
FILE* out;
PrintStructure(FILE* out=stderr) : out(out) {}
{
}
};
struct PrintDDS : public BulletinHeadHandler
{
FILE* out;
PrintDDS(FILE* out=stderr) : out(out) {}
{
}
};
struct PrintTables : public BulletinHeadHandler
{
FILE* out;
bool header_printed;
PrintTables(FILE* out=stderr) : out(out), header_printed(false) {}
{
if (const BufrBulletin* m = dynamic_cast<const BufrBulletin*>(&b))
{
if (!header_printed)
{
fprintf(out,
"%-*s\tOffset\tCentre\tSubc.\tMaster\tLocal\n", (
int)b.
fname.size(),
"Filename");
header_printed = true;
}
fprintf(out, "%s\t%zd\t%d\t%d\t%d\t%d\n",
m->originating_centre, m->originating_subcentre,
m->master_table_version_number, m->master_table_version_number_local);
}
else if (const CrexBulletin* m = dynamic_cast<const CrexBulletin*>(&b))
{
if (!header_printed)
{
fprintf(out, "Filename\tOffset\tMaster\tEdition\tTable\n");
header_printed = true;
}
fprintf(out, "%s\t%zd\t%d\t%d\t%d\n",
m->master_table_number, m->edition_number, m->master_table_version_number);
}
else
{
fprintf(out, "%s\t%zd\tunknown message type\n",
}
}
};
struct PrintFeatures : public BulletinHeadHandler
{
FILE* out;
PrintFeatures(FILE* out=stderr) : out(out) {}
{
scan.run();
bool first = true;
for (const auto& f: scan.features)
if (first)
{
fprintf(out, "%s", f.c_str());
first = false;
} else
fprintf(out, ",%s", f.c_str());
fprintf(out, "\n");
}
};
Printing library configuration
#include <cstdlib>
void do_info()
{
printf("Tables search paths (tried in order):\n");
printf("Extra tables directory: %s (env var WREPORT_EXTRA_TABLES)\n", getenv("WREPORT_EXTRA_TABLES"));
printf("System tables directory: %s (env var WREPORT_TABLES)\n", getenv("WREPORT_TABLES"));
printf("Compiled-in default tables directory: %s\n", TABLE_DIR);
}
Iterating bulletin contents
struct PrintVars : public BulletinFullHandler
{
FILE* out;
const std::vector<wreport::Varcode>& codes;
PrintVars(const std::vector<wreport::Varcode>& codes, FILE* out=stdout)
: out(out), codes(codes) {}
{
for (size_t i = 0; i < subset.size(); ++i)
if (subset[i].code() == code)
return &subset[i];
return NULL;
}
{
for (
size_t sset = 0; sset < b.
subsets.size(); ++sset)
{
fprintf(out,
"%s:%zd:", b.
fname.c_str(), sset + 1);
for (size_t i = 0; i < codes.size(); ++i)
{
const Var* var = find_varcode(b.
subsets[sset], codes[i]);
if (var)
{
string formatted = var->format();
fprintf(out, "\t%s", formatted.c_str());
}
}
putc('\n', out);
}
}
};