eccodes/tools/grib_options.c

512 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.
*/
/*
* C Implementation: grib_options
*
*/
#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" }
};
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("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 */
COMPILE_TIME_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, "Invalid number for -d option: '%s'\n", 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 = GRIB_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);
}
#if 0
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);
}
#endif