mirror of https://github.com/ecmwf/eccodes.git
510 lines
18 KiB
C++
510 lines
18 KiB
C++
/*
|
|
* (C) Copyright 2005- ECMWF.
|
|
*
|
|
* This software is licensed under the terms of the Apache Licence Version 2.0
|
|
* which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
|
|
*
|
|
* In applying this licence, ECMWF does not waive the privileges and immunities granted to it by
|
|
* virtue of its status as an intergovernmental organisation nor does it submit to any jurisdiction.
|
|
*/
|
|
|
|
#include "grib_tools.h"
|
|
|
|
extern char* optarg;
|
|
extern int optind;
|
|
|
|
#ifdef ECCODES_ON_WINDOWS
|
|
#include "wingetopt.h"
|
|
#endif
|
|
|
|
static const char* names[] = { "parameter", "vertical", "geography", "data", "mars", "local" };
|
|
static int names_count = 6;
|
|
|
|
/* id,args,help */
|
|
static grib_options_help grib_options_help_list[] = {
|
|
{ "a", 0, "Dump aliases.\n" },
|
|
{ "b:", "key,key,...",
|
|
"\n\t\tAll the keys in this list are skipped in the comparison. Bit-by-bit compare on.\n" },
|
|
{ "B:", "'order by' directive",
|
|
"\n\t\tOrder by. The output will be ordered according to the 'order by' directive."
|
|
"\n\t\tExample: \"step:i asc, centre desc\" (step numeric ascending and centre descending)"
|
|
"\n\t\tDefault sort mode is 'asc'\n"
|
|
},
|
|
{ "c:", "key[:i|d|s|n],key[:i|d|s|n],...",
|
|
"\n\t\tOnly the listed keys or namespaces (:n) are compared. The optional letter after the colon is used "
|
|
"\n\t\tto force the type in the comparison: i->integer, d->float, s->string, n->namespace."
|
|
"\n\t\tSee -a option. Incompatible with -H option.\n" },
|
|
{ "d:", "value",
|
|
"\n\t\tSet all the data values to \"value\".\n" },
|
|
{ "e:", "tolerance", "\n\t\tOnly values whose difference is more than tolerance are considered different.\n" },
|
|
{ "f", 0, "Force. Force the execution not to fail on error.\n" },
|
|
{ "F:", "format", "\n\t\tC style format for floating-point values.\n" },
|
|
{ "g", 0, "Copy GTS header. \n" },
|
|
{ "G", 0, "GRIBEX compatibility mode.\n" },
|
|
{ "i:", "index",
|
|
"\n\t\tData value corresponding to the given index is printed.\n" },
|
|
{ "j", 0, "JSON mode (JavaScript Object Notation).\n" },
|
|
{ "l:", "latitude,longitude[,MODE,file]",
|
|
"\n\t\tValue close to the point of a latitude,longitude (nearest neighbour)."
|
|
"\n\t\tAllowed values for MODE are:"
|
|
"\n\t\t 4 (4 values in the nearest points are printed) Default"
|
|
"\n\t\t 1 (the value at the nearest point is printed)"
|
|
"\n\t\tfile (file is used as mask. The closer point with mask value>=0.5 is printed)\n" },
|
|
{ "n:", "namespace",
|
|
"\n\t\tAll the keys belonging to the given namespace are printed.\n" },
|
|
{ "m", 0, "Mars keys are printed.\n" },
|
|
{ "o:", "output_file",
|
|
"\n\t\tOutput is written to output_file."
|
|
"\n\t\tIf an output file is required and -o is not used, the"
|
|
" output is written to 'filter.out'\n" },
|
|
{ "p:", "key[:{s|d|i}],key[:{s|d|i}],...",
|
|
"\n\t\tDeclaration of keys to print."
|
|
"\n\t\tFor each key a string (key:s), a double (key:d) or an integer (key:i)"
|
|
"\n\t\ttype can be requested. Default type is string.\n" },
|
|
{ "q", 0, "Quiet.\n" },
|
|
{ "r", 0,
|
|
"Repack data. Sometimes after setting some keys involving properties"
|
|
"\n\t\tof the packing algorithm a repacking of data is needed."
|
|
"\n\t\tThis repacking is performed setting this -r option.\n" },
|
|
{ "s:", "key[:{s|d|i}]=value,key[:{s|d|i}]=value,...",
|
|
"\n\t\tKey/values to set."
|
|
"\n\t\tFor each key a string (key:s), a double (key:d) or an integer (key:i)"
|
|
"\n\t\ttype can be defined. By default the native type is set.\n" },
|
|
{ "t", 0, "Print type information.\n" },
|
|
{ "w:", "key[:{s|d|i}]{=|!=}value,key[:{s|d|i}]{=|!=}value,...",
|
|
"\n\t\tWhere clause."
|
|
"\n\t\tMessages are processed only if they match all the"
|
|
" key/value constraints."
|
|
"\n\t\tA valid constraint is of type key=value or key!=value."
|
|
"\n\t\tFor each key a string (key:s), a double (key:d) or"
|
|
" an integer (key:i)\n\t\ttype can be specified. Default type is string."
|
|
"\n\t\tIn the value you can also use the forward-slash character '/' to specify an OR condition (i.e. a logical disjunction)"
|
|
"\n\t\tNote: only one -w clause is allowed.\n" },
|
|
{ "v", 0, "Verbose.\n" },
|
|
{ "7", 0, "Does not fail when the message has wrong length\n" },
|
|
{ "A:", "absolute error\n",
|
|
"\tCompare floating-point values using the absolute error as tolerance.\n\t\tDefault is absolute error=0\n" },
|
|
{ "C", 0, "C code mode. A C code program generating the message is dumped.\n" },
|
|
{ "D", 0, "Debug mode.\n" },
|
|
{ "H", 0, "Print octet content in hexadecimal format.\n" },
|
|
{ "M", 0, "Multi-field support off. Turn off support for multiple fields in a single GRIB message.\n" },
|
|
{ "O", 0, "Octet mode. WMO documentation style dump.\n" },
|
|
{ "P:", "key[:{s|d|i}],key[:{s|d|i}],...",
|
|
"\n\t\tAs -p adding the declared keys to the default list.\n" },
|
|
{ "R:", "key1=relative_error1,key2=relative_error2,...\n",
|
|
"\tCompare floating-point values using the relative error as tolerance."
|
|
"\n\t\tkey1=relative_error1 will compare key1 using relative_error1."
|
|
"\n\t\tall=relative_error will compare all the floating-point keys using relative_error. Default all=0.\n" },
|
|
{ "S", 0,
|
|
"Strict. Only messages matching all the constraints are copied to"
|
|
"\n\t\tthe output file\n" },
|
|
{ "T:", "T | B | M | A", "Message type. T->GTS, B->BUFR, M->METAR (Experimental), A->Any (Experimental).\n\t\t\t\tThe input file is interpreted according to the message type.\n" },
|
|
{ "V", 0, "Version.\n" },
|
|
{ "W:", "width", "\n\t\tMinimum width of each column in output. Default is 10.\n" },
|
|
{ "X:", "offset", "\n\t\tInput file offset in bytes. Processing of the input file will start from the given offset.\n" },
|
|
{ "x", 0, "Fast parsing option, only headers are loaded.\n" },
|
|
{ "k:", "key1,key2,...",
|
|
"\n\t\tSpecify a list of keys to index on. By default the input files are indexed on the MARS keys."
|
|
"\n\t\tFor each key a string (key:s) or a double (key:d) or an integer (key:i)"
|
|
"\n\t\ttype can be requested.\n" },
|
|
{ "h", 0, "Display this help text and exit.\n" }
|
|
};
|
|
|
|
static int grib_options_help_count = sizeof(grib_options_help_list) / sizeof(grib_options_help);
|
|
|
|
|
|
void usage(void)
|
|
{
|
|
int i = 0;
|
|
printf("\nNAME \t%s\n\n", tool_name);
|
|
printf("DESCRIPTION\n\t%s\n\n", tool_description);
|
|
printf("USAGE \n\t%s %s\n\n", tool_name, tool_usage);
|
|
printf("OPTIONS\n");
|
|
for (i = 0; i < grib_options_count; i++) {
|
|
if (grib_options[i].command_line) {
|
|
printf("\t-%c %s\t%s", grib_options[i].id[0],
|
|
grib_options_get_args(grib_options[i].id),
|
|
grib_options_get_help(grib_options[i].id));
|
|
}
|
|
}
|
|
printf("\n");
|
|
if (tool_online_doc)
|
|
printf("SEE ALSO\n\tFull documentation and examples at:\n\t<%s>\n\n", tool_online_doc);
|
|
printf("\n");
|
|
exit(1);
|
|
}
|
|
|
|
char* grib_options_get_option(const char* id)
|
|
{
|
|
int i = 0;
|
|
for (i = 0; i < grib_options_count; i++) {
|
|
if (!strcmp(id, grib_options[i].id))
|
|
return grib_options[i].value;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
int grib_options_command_line(const char* id)
|
|
{
|
|
int i = 0;
|
|
for (i = 0; i < grib_options_count; i++) {
|
|
if (!strcmp(id, grib_options[i].id))
|
|
return grib_options[i].command_line;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int grib_options_on(const char* id)
|
|
{
|
|
int i = 0;
|
|
for (i = 0; i < grib_options_count; i++) {
|
|
if (!strcmp(id, grib_options[i].id))
|
|
return grib_options[i].on;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int grib_get_runtime_options(int argc, char** argv, grib_runtime_options* options)
|
|
{
|
|
int i = 0, c = 0;
|
|
char* optstr = (char*)calloc(1, 2 * grib_options_count * sizeof(char));
|
|
|
|
if (!optstr)
|
|
return GRIB_OUT_OF_MEMORY;
|
|
|
|
for (i = 0; i < grib_options_count; i++)
|
|
if (grib_options[i].command_line)
|
|
strncat(optstr, grib_options[i].id, 2);
|
|
|
|
|
|
while ((c = getopt(argc, argv, optstr)) != -1) {
|
|
if (c == '?')
|
|
usage();
|
|
i = 0;
|
|
while (i < grib_options_count && grib_options[i].id[0] != c)
|
|
i++;
|
|
grib_options[i].on = 1;
|
|
if (grib_options[i].id[1] == ':')
|
|
grib_options[i].value = optarg;
|
|
}
|
|
free(optstr);
|
|
return 0;
|
|
}
|
|
|
|
int grib_process_runtime_options(grib_context* context, int argc, char** argv, grib_runtime_options* options)
|
|
{
|
|
int i = 0, ret = 0;
|
|
int has_output = 0;
|
|
int has_input_extra = 0, nfiles = 0;
|
|
char *karg = NULL, *warg = NULL, *sarg = NULL, *barg = NULL;
|
|
|
|
if (grib_options_on("h")) {
|
|
usage();
|
|
}
|
|
|
|
if (grib_options_on("V")) {
|
|
printf("\necCodes Version ");
|
|
grib_print_api_version(stdout);
|
|
printf("\n\n");
|
|
exit(0);
|
|
}
|
|
|
|
if (grib_options_on("B:"))
|
|
options->orderby = grib_options_get_option("B:");
|
|
|
|
if (grib_options_on("x"))
|
|
options->headers_only = 1;
|
|
else
|
|
options->headers_only = 0;
|
|
|
|
if (grib_options_on("T:")) {
|
|
char* x = grib_options_get_option("T:");
|
|
if (*x == 'T')
|
|
options->mode = MODE_GTS;
|
|
else if (*x == 'B')
|
|
options->mode = MODE_BUFR;
|
|
else if (*x == 'M')
|
|
options->mode = MODE_METAR;
|
|
else if (*x == 'F')
|
|
options->mode = MODE_TAF;
|
|
else if (*x == 'A')
|
|
options->mode = MODE_ANY;
|
|
else
|
|
options->mode = MODE_GRIB;
|
|
}
|
|
|
|
if (grib_options_on("F:"))
|
|
options->format = grib_options_get_option("F:");
|
|
else
|
|
options->format = strdup("%g");
|
|
|
|
if (grib_options_on("i:")) {
|
|
options->index_on = 1;
|
|
options->index = atoi(grib_options_get_option("i:"));
|
|
}
|
|
|
|
if (grib_options_on("l:"))
|
|
options->latlon = grib_options_get_option("l:");
|
|
|
|
if (grib_options_on("j"))
|
|
options->json_output = 1;
|
|
else
|
|
options->json_output = 0;
|
|
|
|
if (grib_options_on("X:"))
|
|
options->infile_offset = atol(grib_options_get_option("X:"));
|
|
|
|
#ifndef ECCODES_ON_WINDOWS
|
|
/* Check at compile time to ensure our file offset is at least 64 bits */
|
|
static_assert(sizeof(options->infile_offset) >= 8);
|
|
#endif
|
|
|
|
has_output = grib_options_on("U");
|
|
has_input_extra = grib_options_on("I");
|
|
options->repack = grib_options_on("r");
|
|
options->gts = grib_options_on("g");
|
|
|
|
if (grib_options_on("d:")) {
|
|
char* endPtr = NULL; /* for error handling */
|
|
const char* optionStr = grib_options_get_option("d:");
|
|
options->constant = strtod(optionStr, &endPtr);
|
|
if (*endPtr) {
|
|
fprintf(stderr, "%s: Invalid number for -d option: '%s'\n", tool_name, optionStr);
|
|
exit(1);
|
|
}
|
|
options->repack = 1;
|
|
}
|
|
|
|
if (grib_options_on("G"))
|
|
grib_gribex_mode_on(context);
|
|
|
|
nfiles = argc - optind;
|
|
if (nfiles < (1 + has_output + has_input_extra) && !options->infile)
|
|
usage();
|
|
|
|
if (has_input_extra) {
|
|
options->infile_extra = (grib_tools_file*)calloc(1, sizeof(grib_tools_file));
|
|
options->infile_extra->name = argv[optind];
|
|
}
|
|
|
|
if (!options->infile) {
|
|
for (i = optind + has_input_extra; i < argc - has_output; i++) {
|
|
grib_tools_file* p = NULL;
|
|
grib_tools_file* infile = (grib_tools_file*)calloc(1, sizeof(grib_tools_file));
|
|
infile->name = argv[i];
|
|
if (!options->infile)
|
|
options->infile = infile;
|
|
else {
|
|
p = options->infile;
|
|
while (p->next)
|
|
p = p->next;
|
|
p->next = infile;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (has_output) {
|
|
options->outfile = (grib_tools_file*)calloc(1, sizeof(grib_tools_file));
|
|
options->outfile->name = strdup(argv[argc - 1]);
|
|
}
|
|
|
|
if (grib_options_on("o:")) {
|
|
options->outfile = (grib_tools_file*)calloc(1, sizeof(grib_tools_file));
|
|
options->outfile->name = grib_options_get_option("o:");
|
|
}
|
|
|
|
options->print_number = grib_options_on("N");
|
|
options->print_header = grib_options_on("H");
|
|
options->verbose = grib_options_on("v");
|
|
|
|
if (grib_options_on("q") && grib_options_command_line("q"))
|
|
options->verbose = 0;
|
|
|
|
options->fail = !grib_options_on("f");
|
|
if (grib_options_get_option("W:"))
|
|
options->default_print_width = atoi(grib_options_get_option("W:"));
|
|
|
|
if (grib_options_on("n:"))
|
|
options->name_space = grib_options_get_option("n:");
|
|
|
|
if (grib_options_on("m"))
|
|
options->name_space = strdup("mars");
|
|
|
|
|
|
if (grib_options_on("P:"))
|
|
karg = grib_options_get_option("P:");
|
|
else if (grib_options_on("p:")) {
|
|
karg = grib_options_get_option("p:");
|
|
options->name_space = NULL;
|
|
}
|
|
|
|
options->requested_print_keys_count = MAX_KEYS;
|
|
ret = parse_keyval_string(tool_name, karg, 0, GRIB_TYPE_UNDEFINED,
|
|
options->requested_print_keys, &(options->requested_print_keys_count));
|
|
if (ret == GRIB_INVALID_ARGUMENT)
|
|
usage();
|
|
GRIB_CHECK_NOLINE(ret, 0);
|
|
|
|
options->strict = grib_options_on("S");
|
|
|
|
if (grib_options_on("M"))
|
|
grib_multi_support_off(context);
|
|
else
|
|
grib_multi_support_on(context);
|
|
|
|
if (grib_options_on("g"))
|
|
grib_gts_header_on(context);
|
|
else
|
|
grib_gts_header_off(context);
|
|
|
|
if (grib_options_on("V")) {
|
|
printf("\necCodes Version ");
|
|
grib_print_api_version(stdout);
|
|
printf("\n\n");
|
|
}
|
|
|
|
if (grib_options_on("s:")) {
|
|
sarg = grib_options_get_option("s:");
|
|
options->set_values_count = MAX_KEYS;
|
|
ret = parse_keyval_string(tool_name, sarg, 1, GRIB_TYPE_UNDEFINED, options->set_values, &(options->set_values_count));
|
|
if (ret == GRIB_INVALID_ARGUMENT)
|
|
usage();
|
|
}
|
|
|
|
if (grib_options_on("b:")) {
|
|
barg = grib_options_get_option("b:");
|
|
options->set_values_count = MAX_KEYS;
|
|
ret = parse_keyval_string(tool_name, barg, 0, GRIB_TYPE_STRING, options->set_values, &(options->set_values_count));
|
|
if (ret == GRIB_INVALID_ARGUMENT)
|
|
usage();
|
|
}
|
|
|
|
if (grib_options_on("c:")) {
|
|
sarg = grib_options_get_option("c:");
|
|
options->compare_count = MAX_KEYS;
|
|
ret = parse_keyval_string(tool_name, sarg, 0, GRIB_TYPE_UNDEFINED, options->compare,
|
|
&(options->compare_count));
|
|
if (ret == GRIB_INVALID_ARGUMENT)
|
|
usage();
|
|
}
|
|
if (grib_options_on("e")) {
|
|
for (i = 0; i < names_count; i++) {
|
|
options->compare[i + options->compare_count].name = names[i];
|
|
options->compare[i + options->compare_count].type = CODES_NAMESPACE;
|
|
}
|
|
options->compare_count += names_count;
|
|
}
|
|
|
|
warg = grib_options_get_option("w:");
|
|
|
|
options->constraints_count = MAX_KEYS;
|
|
ret = parse_keyval_string(tool_name, warg, 1, GRIB_TYPE_UNDEFINED,
|
|
options->constraints, &(options->constraints_count));
|
|
if (ret == GRIB_INVALID_ARGUMENT)
|
|
usage();
|
|
|
|
return GRIB_SUCCESS;
|
|
}
|
|
|
|
const char* grib_options_get_help(const char* id)
|
|
{
|
|
int i = 0;
|
|
char msg[] = "ERROR: help not found for option ";
|
|
const size_t msize = sizeof(msg) + 3;
|
|
char* err = (char*)calloc(1, msize);
|
|
snprintf(err, msize, "%s%c\n", msg, *id);
|
|
for (i = 0; i < grib_options_count; i++) {
|
|
if (!strcmp(id, grib_options[i].id)) {
|
|
if (grib_options[i].help != NULL)
|
|
return grib_options[i].help;
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
for (i = 0; i < grib_options_help_count; i++) {
|
|
if (!strcmp(id, grib_options_help_list[i].id)) {
|
|
return grib_options_help_list[i].help != NULL ? grib_options_help_list[i].help : err;
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
const char* grib_options_get_args(const char* id)
|
|
{
|
|
int i = 0;
|
|
char empty[] = "";
|
|
char msg[] = "ERROR: help not found for option -";
|
|
const size_t msize = sizeof(msg) + 3;
|
|
char* err = NULL;
|
|
if (id[1] != ':')
|
|
return strdup(empty);
|
|
err = (char*)calloc(1, msize);
|
|
snprintf(err, msize, "%s%c\n", msg, *id);
|
|
for (i = 0; i < grib_options_count; i++) {
|
|
if (!strcmp(id, grib_options[i].id)) {
|
|
if (grib_options[i].args != NULL) {
|
|
free(err);
|
|
return grib_options[i].args;
|
|
}
|
|
else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < grib_options_help_count; i++) {
|
|
if (!strcmp(id, grib_options_help_list[i].id)) {
|
|
if (grib_options_help_list[i].args != NULL) {
|
|
free(err);
|
|
return grib_options_help_list[i].args;
|
|
}
|
|
else {
|
|
return err;
|
|
}
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
void usage_doxygen(void)
|
|
{
|
|
int i = 0;
|
|
printf("/*! \\page %s %s\n", tool_name, tool_name);
|
|
printf("\\section DESCRIPTION \n %s\n\n", tool_description);
|
|
printf("\\section USAGE \n %s \n %s\n\n", tool_name, tool_usage);
|
|
printf("\\section OPTIONS \n");
|
|
for (i = 0; i < grib_options_count; i++) {
|
|
if (grib_options[i].command_line) {
|
|
printf("-%c %s \\n",
|
|
grib_options[i].id[0],
|
|
grib_options_get_args(grib_options[i].id));
|
|
printf(" %s \\n \\n ",
|
|
grib_options_get_help(grib_options[i].id));
|
|
}
|
|
}
|
|
exit(1);
|
|
}
|
|
|
|
// void usage_doxygen(void) {
|
|
// int i=0;
|
|
// printf("/*! \\page %s %s\n",tool_name,tool_name);
|
|
// printf("\\section DESCRIPTION \n%s\n\n",tool_description);
|
|
// printf("\\section USAGE \n%s \n%s\n\n",tool_name,tool_usage);
|
|
// printf("\\section OPTIONS\n");
|
|
// printf("<table frame=void border=0>\n");
|
|
// for (i=0;i<grib_options_count;i++) {
|
|
// if (grib_options[i].command_line) {
|
|
// printf("<tr>\n");
|
|
// printf("<td colspan=2>-%c %s</td>\n",
|
|
// grib_options[i].id[0],
|
|
// grib_options_get_args(grib_options[i].id));
|
|
// printf("</tr><tr>\n");
|
|
// printf("<td width=20></td><td>%s</td>",
|
|
// grib_options_get_help(grib_options[i].id));
|
|
// printf("</tr><tr><td></td></tr>\n");
|
|
// }
|
|
// }
|
|
// printf("</table>\n");
|
|
// exit(1);
|
|
// }
|