/* * (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." "\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 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\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 "; char* err = (char*)calloc(1, sizeof(msg) + 3); snprintf(err, 1024, "%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; } const char* grib_options_get_args(const 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); snprintf(err, 1024, "%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 (char*)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("
-%c %s | \n", grib_options[i].id[0], grib_options_get_args(grib_options[i].id)); printf("|
%s | ", grib_options_get_help(grib_options[i].id)); printf("|