mirror of https://github.com/ecmwf/eccodes.git
451 lines
16 KiB
C
451 lines
16 KiB
C
/*
|
|
* Copyright 2005-2018 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
|
|
|
|
const char* names[]={"parameter","vertical","geography","data","mars","local"};
|
|
int names_count=6;
|
|
|
|
/* id,args,help */
|
|
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\tOrder by example: \"step:i asc, centre desc\" (step numeric ascending and centre descending)\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."
|
|
"\n\t\tAllowed values for MODE are:"
|
|
"\n\t\t4 (4 values in the nearest points are printed) Default"
|
|
"\n\t\t1 (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 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 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 \"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"}
|
|
|
|
};
|
|
|
|
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",grib_tool_name);
|
|
printf("DESCRIPTION\n\t%s\n\n",grib_tool_description);
|
|
printf("USAGE \n\t%s %s\n\n",grib_tool_name,grib_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\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 == '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("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:")) {
|
|
options->constant=atof(grib_options_get_option("d:"));
|
|
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(grib_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(grib_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(grib_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(grib_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(grib_tool_name,warg,1,GRIB_TYPE_UNDEFINED,
|
|
options->constraints,&(options->constraints_count));
|
|
if (ret == GRIB_INVALID_ARGUMENT) usage();
|
|
|
|
return GRIB_SUCCESS;
|
|
}
|
|
|
|
char* grib_options_get_help(char* id)
|
|
{
|
|
int i=0;
|
|
char msg[]="ERROR: help not found for option ";
|
|
char* err=(char*)calloc(1,sizeof(msg)+3);
|
|
sprintf(err,"%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 ? (char*)grib_options_help_list[i].help : err;
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
char* grib_options_get_args(char* id)
|
|
{
|
|
int i=0;
|
|
char empty[]="";
|
|
char msg[]="ERROR: help not found for option -";
|
|
char* err=NULL;
|
|
if (id[1] != ':') return strdup(empty);
|
|
err=(char*)calloc(1,sizeof(msg)+3);
|
|
sprintf(err,"%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) 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)) {
|
|
return grib_options_help_list[i].args != NULL ? (char*)grib_options_help_list[i].args : err;
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
void usage_doxygen(void)
|
|
{
|
|
int i=0;
|
|
printf("/*! \\page %s %s\n",grib_tool_name,grib_tool_name);
|
|
printf("\\section DESCRIPTION \n %s\n\n",grib_tool_description);
|
|
printf("\\section USAGE \n %s \n %s\n\n",grib_tool_name,grib_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",grib_tool_name,grib_tool_name);
|
|
printf("\\section DESCRIPTION \n%s\n\n",grib_tool_description);
|
|
printf("\\section USAGE \n%s \n%s\n\n",grib_tool_name,grib_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
|
|
|