2016-04-04 10:06:37 +00:00
|
|
|
/*
|
2020-01-28 14:32:34 +00:00
|
|
|
* (C) Copyright 2005- ECMWF.
|
2016-04-04 10:06:37 +00:00
|
|
|
*
|
|
|
|
* 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"
|
|
|
|
|
2021-02-04 11:53:15 +00:00
|
|
|
grib_option grib_options[] = {
|
|
|
|
/* {id, args, help}, on, command_line, value*/
|
|
|
|
/*{"r",0,"Compare files in which the messages are not in the same order. This option is time expensive.\n",0,1,0},*/
|
|
|
|
{ "b:", 0, 0, 0, 1, 0 },
|
|
|
|
{ "d", 0, "Write different messages on files\n", 0, 1, 0 },
|
|
|
|
{ "T:", 0, 0, 1, 0, "T" }, /* GTS */
|
|
|
|
{ "c:", 0, 0, 0, 1, 0 },
|
|
|
|
{ "a", 0, "-c option modifier. The keys listed with the option -c will be added to the list of keys compared without -c.\n", 0, 1, 0 },
|
|
|
|
/*{"R:",0,0,0,1,0},*/
|
|
|
|
/*{"A:",0,0,0,1,0},*/
|
|
|
|
{ "w:", 0, 0, 0, 1, 0 },
|
2024-04-30 16:13:31 +00:00
|
|
|
{ "f", 0, "Forcefully compare, do not stop after first difference.\n", 0, 1, 0 },
|
2021-02-04 11:53:15 +00:00
|
|
|
{ "F", 0, 0, 1, 0, 0 },
|
|
|
|
{ "q", 0, 0, 1, 0, 0 },
|
|
|
|
{ "I", 0, 0, 1, 0, 0 },
|
|
|
|
{ "V", 0, 0, 0, 1, 0 },
|
|
|
|
{ "7", 0, 0, 0, 1, 0 },
|
|
|
|
{ "v", 0, 0, 0, 1, 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
int grib_options_count = sizeof(grib_options) / sizeof(grib_option);
|
|
|
|
|
|
|
|
const char* tool_description =
|
|
|
|
"Compare GTS messages contained in two files."
|
|
|
|
"\n\tIf some differences are found it fails returning an error code."
|
|
|
|
"\n\tDefault behaviour: bit-by-bit compare, same order in files.";
|
|
|
|
|
|
|
|
const char* tool_name = "gts_compare";
|
2023-01-04 21:16:24 +00:00
|
|
|
const char* tool_online_doc = NULL;
|
2023-08-11 20:57:13 +00:00
|
|
|
const char* tool_usage = "[options] file file";
|
2021-02-04 11:53:15 +00:00
|
|
|
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
GRIB_INLINE static int grib_inline_strcmp(const char* a, const char* b)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
if (*a != *b)
|
|
|
|
return 1;
|
|
|
|
while ((*a != 0 && *b != 0) && *(a) == *(b)) {
|
|
|
|
a++;
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
return (*a == 0 && *b == 0) ? 0 : 1;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
typedef double (*compare_double_proc)(double*, double*, double*);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
typedef struct grib_error grib_error;
|
2020-01-22 13:10:59 +00:00
|
|
|
struct grib_error
|
|
|
|
{
|
2016-04-04 10:06:37 +00:00
|
|
|
char* key;
|
|
|
|
int count;
|
|
|
|
grib_error* next;
|
|
|
|
};
|
|
|
|
|
|
|
|
grib_error* error_summary;
|
|
|
|
|
|
|
|
compare_double_proc compare_double;
|
2020-09-24 14:54:33 +00:00
|
|
|
grib_string_list* blocklist = 0;
|
2020-01-22 13:10:59 +00:00
|
|
|
|
|
|
|
static int compare_handles(grib_handle* h1, grib_handle* h2, grib_runtime_options* options);
|
2021-02-04 11:53:15 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
int error = 0;
|
|
|
|
int count = 0;
|
|
|
|
int force = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
double maxAbsoluteError = 1e-19;
|
2020-01-22 13:10:59 +00:00
|
|
|
int onlyListed = 1;
|
|
|
|
int morein1 = 0;
|
|
|
|
int morein2 = 0;
|
2022-04-06 12:22:24 +00:00
|
|
|
int listFromCommandLine = 0;
|
|
|
|
int verbose = 0;
|
|
|
|
double tolerance_factor = 1;
|
|
|
|
static int write_error = 0;
|
|
|
|
static int write_count = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2021-02-04 11:53:15 +00:00
|
|
|
grib_handle* global_handle = NULL;
|
|
|
|
int counter = 0;
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
static void write_message(grib_handle* h, const char* str)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
const void* m;
|
|
|
|
size_t s;
|
|
|
|
char fname[1024] = {0,};
|
|
|
|
FILE* fh = NULL;
|
|
|
|
|
|
|
|
grib_get_message(h, &m, &s);
|
2022-11-10 19:18:43 +00:00
|
|
|
snprintf(fname, 1024, "%s_%d.gts", str, write_count);
|
2020-01-22 13:10:59 +00:00
|
|
|
|
|
|
|
fh = fopen(fname, "w");
|
|
|
|
if (!fh) {
|
|
|
|
grib_context_log(h->context, (GRIB_LOG_ERROR) | (GRIB_LOG_PERROR),
|
|
|
|
"Error opening %s", fname);
|
2016-04-04 10:06:37 +00:00
|
|
|
exit(GRIB_IO_PROBLEM);
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (fwrite(m, 1, s, fh) != s) {
|
|
|
|
grib_context_log(h->context, (GRIB_LOG_ERROR) | (GRIB_LOG_PERROR),
|
|
|
|
"Error writing to %s", fname);
|
2016-04-04 10:06:37 +00:00
|
|
|
exit(GRIB_IO_PROBLEM);
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(fh);
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
static void write_messages(grib_handle* h1, grib_handle* h2)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!write_error)
|
|
|
|
return;
|
2016-04-04 10:06:37 +00:00
|
|
|
write_count++;
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
write_message(h1, "error1");
|
|
|
|
write_message(h2, "error2");
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2020-09-24 14:54:33 +00:00
|
|
|
static int blocklisted(const char* name)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-09-24 14:54:33 +00:00
|
|
|
grib_string_list* b = blocklist;
|
2020-02-12 11:13:03 +00:00
|
|
|
if (!name) return 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
while (b) {
|
2024-12-20 12:58:07 +00:00
|
|
|
ECCODES_ASSERT(b->value);
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!strcmp(name, b->value))
|
2016-04-04 10:06:37 +00:00
|
|
|
return 1;
|
2020-01-22 13:10:59 +00:00
|
|
|
b = b->next;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
int main(int argc, char* argv[])
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
return grib_tool(argc, argv);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int grib_tool_before_getopt(grib_runtime_options* options)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int grib_tool_init(grib_runtime_options* options)
|
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_context* context = grib_context_get_default();
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
options->strict = 1;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (grib_options_on("f"))
|
|
|
|
force = 1;
|
|
|
|
else
|
|
|
|
force = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (grib_options_on("d"))
|
|
|
|
write_error = 1;
|
|
|
|
else
|
|
|
|
write_error = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
verbose = grib_options_on("v");
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
listFromCommandLine = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
if (grib_options_on("c:") || grib_options_on("e"))
|
2020-01-22 13:10:59 +00:00
|
|
|
listFromCommandLine = 1;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (grib_options_on("a"))
|
|
|
|
onlyListed = 0;
|
|
|
|
else
|
|
|
|
onlyListed = 1;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
if (grib_options_on("a") && !grib_options_on("c:")) {
|
2024-03-23 19:19:08 +00:00
|
|
|
fprintf(stderr, "Error: -a option requires -c option. Please define a list of keys with the -c option.\n");
|
2016-04-04 10:06:37 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (grib_options_on("b:")) {
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_string_list* next = 0;
|
|
|
|
int i = 0;
|
2020-09-24 14:54:33 +00:00
|
|
|
blocklist = (grib_string_list*)grib_context_malloc_clear(context, sizeof(grib_string_list));
|
|
|
|
blocklist->value = grib_context_strdup(context, options->set_values[0].name);
|
|
|
|
next = blocklist;
|
2020-01-22 13:10:59 +00:00
|
|
|
for (i = 1; i < options->set_values_count; i++) {
|
|
|
|
next->next = (grib_string_list*)grib_context_malloc_clear(context, sizeof(grib_string_list));
|
|
|
|
next->next->value = grib_context_strdup(context, options->set_values[i].name);
|
|
|
|
next = next->next;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
2020-09-24 14:54:33 +00:00
|
|
|
context->blocklist = blocklist;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2023-08-11 20:57:13 +00:00
|
|
|
options->random = 0;
|
|
|
|
options->infile_extra->file = fopen(options->infile_extra->name, "r");
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2023-08-11 20:57:13 +00:00
|
|
|
if (!options->infile_extra->file) {
|
|
|
|
perror(options->infile_extra->name);
|
|
|
|
exit(1);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (grib_options_on("t:"))
|
2022-04-06 12:22:24 +00:00
|
|
|
tolerance_factor = atof(grib_options_get_option("t:"));
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
/* Check for 2nd file being a directory. If so, we assume user is comparing to a file */
|
|
|
|
/* with the same name as first file in that directory */
|
|
|
|
grib_tools_file* infile = options->infile; /* the 2nd file in comparison */
|
|
|
|
if (infile) {
|
2019-12-27 11:38:59 +00:00
|
|
|
if (path_is_directory(infile->name)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
/* Take the filename of the 1st file and append to dir */
|
|
|
|
char bufr[2048] = {0,};
|
|
|
|
/* options->infile_extra->name is the 1st file */
|
2022-11-10 19:18:43 +00:00
|
|
|
snprintf(bufr, 2048, "%s%c%s",
|
2016-04-04 10:06:37 +00:00
|
|
|
infile->name,
|
|
|
|
get_dir_separator_char(),
|
|
|
|
extract_filename(options->infile_extra->name));
|
|
|
|
infile->name = strdup(bufr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
int grib_tool_new_filename_action(grib_runtime_options* options, const char* file)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2020-01-22 13:10:59 +00:00
|
|
|
int grib_tool_new_file_action(grib_runtime_options* options, grib_tools_file* file)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void printInfo(grib_handle* h)
|
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
printf("== %d == DIFFERENCE == ", count);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2019-05-10 13:11:01 +00:00
|
|
|
static grib_handle* gts_handle_new_from_file_x(
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_context* c, FILE* f, int mode, int headers_only, int* err)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
return codes_handle_new_from_file(c, f, PRODUCT_GTS, err);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2016-04-04 10:42:48 +00:00
|
|
|
int grib_tool_new_handle_action(grib_runtime_options* options, grib_handle* h)
|
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
int err = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
count++;
|
|
|
|
|
2023-08-24 10:34:30 +00:00
|
|
|
global_handle = gts_handle_new_from_file_x(h->context, options->infile_extra->file, options->mode, 0, &err);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!global_handle || err != GRIB_SUCCESS) {
|
2016-04-04 10:06:37 +00:00
|
|
|
morein2++;
|
2020-01-22 13:10:59 +00:00
|
|
|
if (global_handle)
|
|
|
|
grib_handle_delete(global_handle);
|
2016-04-04 10:06:37 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (compare_handles(global_handle, h, options)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
error++;
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!force)
|
|
|
|
exit(1);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2016-04-13 18:32:18 +00:00
|
|
|
grib_handle_delete(global_handle);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int grib_tool_skip_handle(grib_runtime_options* options, grib_handle* h)
|
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
int err = 0;
|
2023-08-24 10:34:30 +00:00
|
|
|
if (!options->through_index) {
|
2020-01-22 13:10:59 +00:00
|
|
|
global_handle = gts_new_from_file(h->context, options->infile_extra->file, &err);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!global_handle || err != GRIB_SUCCESS)
|
2016-04-04 10:06:37 +00:00
|
|
|
morein2++;
|
|
|
|
|
2016-04-13 18:32:18 +00:00
|
|
|
grib_handle_delete(global_handle);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
grib_handle_delete(h);
|
|
|
|
count++;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int grib_tool_finalise_action(grib_runtime_options* options)
|
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_error* e = error_summary;
|
|
|
|
int err = 0;
|
|
|
|
grib_context* c = grib_context_get_default();
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
while ((global_handle = gts_new_from_file(c, options->infile_extra->file, &err))) {
|
2019-05-10 13:11:01 +00:00
|
|
|
morein1++;
|
2023-01-22 13:32:32 +00:00
|
|
|
grib_handle_delete(global_handle);
|
2019-05-10 13:11:01 +00:00
|
|
|
}
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
error += morein1 + morein2;
|
2016-04-04 10:06:37 +00:00
|
|
|
if (error) {
|
|
|
|
printf("\n## ERRORS SUMMARY #######\n");
|
|
|
|
}
|
2019-05-10 13:11:01 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (morein1 > 0) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printf("##\n## Different number of messages \n");
|
2020-01-22 13:10:59 +00:00
|
|
|
printf("## %d more messages in %s than in %s\n", morein1,
|
|
|
|
options->infile_extra->name, options->infile->name);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (morein2 > 0) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printf("##\n## Different number of messages \n");
|
2020-01-22 13:10:59 +00:00
|
|
|
printf("## %d more messages in %s than in %s\n", morein2,
|
|
|
|
options->infile->name, options->infile_extra->name);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (error) {
|
|
|
|
printf("##\n## Summary of different key values \n");
|
|
|
|
while (e) {
|
2020-01-22 13:10:59 +00:00
|
|
|
printf("## %s ( %d different )\n", e->key, e->count);
|
|
|
|
e = e->next;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
printf("##\n## %d different messages out of %d\n\n", error, count);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (error != 0)
|
|
|
|
exit(1);
|
2016-04-04 10:06:37 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
static void save_error(grib_context* c, const char* key)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_error* e = 0;
|
|
|
|
grib_error* next = 0;
|
|
|
|
int saved = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
if (!error_summary) {
|
2020-01-22 13:10:59 +00:00
|
|
|
error_summary = (grib_error*)grib_context_malloc_clear(c, sizeof(grib_error));
|
|
|
|
error_summary->count = 1;
|
|
|
|
error_summary->key = grib_context_strdup(c, key);
|
2016-04-04 10:06:37 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
e = error_summary;
|
|
|
|
next = e;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
while (next) {
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!strcmp(next->key, key)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
next->count++;
|
2020-01-22 13:10:59 +00:00
|
|
|
saved = 1;
|
2016-04-04 10:06:37 +00:00
|
|
|
break;
|
|
|
|
}
|
2020-01-22 13:10:59 +00:00
|
|
|
e = next;
|
|
|
|
next = next->next;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!saved) {
|
2020-01-22 13:10:59 +00:00
|
|
|
e->next = (grib_error*)grib_context_malloc_clear(c, sizeof(grib_error));
|
|
|
|
e->next->count = 1;
|
|
|
|
e->next->key = grib_context_strdup(c, key);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-06 21:35:38 +00:00
|
|
|
static int compare_values(const grib_runtime_options* options, grib_handle* h1, grib_handle* h2, const char* name, int type)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2024-04-09 11:03:22 +00:00
|
|
|
size_t len1 = 0, len2 = 0;
|
|
|
|
int err = 0, err1 = 0, err2 = 0;
|
2020-01-22 13:10:59 +00:00
|
|
|
int type1, type2;
|
2016-04-04 10:06:37 +00:00
|
|
|
int countdiff;
|
2024-04-09 11:03:22 +00:00
|
|
|
// int isMissing1 = 0, isMissing2 = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
char *sval1 = NULL, *sval2 = NULL;
|
2016-04-04 10:06:37 +00:00
|
|
|
long *lval1 = NULL, *lval2 = NULL;
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_context* c = h1->context;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
type1 = type;
|
|
|
|
type2 = type;
|
|
|
|
if (verbose)
|
|
|
|
printf(" comparing %s", name);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (type1 == GRIB_TYPE_UNDEFINED && (err = grib_get_native_type(h1, name, &type1)) != GRIB_SUCCESS) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get type of [%s] in 1st field: %s\n", name, grib_get_error_message(err));
|
2020-01-22 13:10:59 +00:00
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (type2 == GRIB_TYPE_UNDEFINED && (err = grib_get_native_type(h2, name, &type2)) != GRIB_SUCCESS) {
|
|
|
|
if (err == GRIB_NOT_FOUND) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printInfo(h1);
|
2020-01-22 13:10:59 +00:00
|
|
|
printf("[%s] not found in 2nd field\n", name);
|
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get type of [%s] in 2nd field: %s\n", name, grib_get_error_message(err));
|
2020-01-22 13:10:59 +00:00
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (type1 == GRIB_TYPE_LABEL)
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (type1 == GRIB_TYPE_SECTION)
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if ((err = grib_get_size(h1, name, &len1)) != GRIB_SUCCESS) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get size of [%s] in 1st field: %s\n", name, grib_get_error_message(err));
|
2020-01-22 13:10:59 +00:00
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if ((err = grib_get_size(h2, name, &len2)) != GRIB_SUCCESS) {
|
|
|
|
if (err == GRIB_NOT_FOUND) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printInfo(h1);
|
2020-01-22 13:10:59 +00:00
|
|
|
printf("[%s] not found in 2nd field\n", name);
|
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get size of [%s] in 2nd field: %s\n", name, grib_get_error_message(err));
|
2020-01-22 13:10:59 +00:00
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
return err;
|
|
|
|
}
|
2024-12-20 12:58:07 +00:00
|
|
|
ECCODES_ASSERT(options->mode == MODE_GTS);
|
2016-04-04 10:06:37 +00:00
|
|
|
/* TODO: Ignore missing values for keys in GTS. Not yet implemented */
|
2024-01-13 18:27:19 +00:00
|
|
|
//isMissing1 = ((grib_is_missing(h1, name, &err1) == 1) && (err1 == 0)) ? 1 : 0;
|
|
|
|
//isMissing2 = ((grib_is_missing(h2, name, &err2) == 1) && (err2 == 0)) ? 1 : 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
switch (type1) {
|
|
|
|
case GRIB_TYPE_STRING:
|
2024-01-06 20:49:08 +00:00
|
|
|
if (verbose) printf(" as string\n");
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_get_string_length(h1, name, &len1);
|
|
|
|
grib_get_string_length(h2, name, &len2);
|
|
|
|
sval1 = (char*)grib_context_malloc(h1->context, len1 * sizeof(char));
|
|
|
|
sval2 = (char*)grib_context_malloc(h2->context, len2 * sizeof(char));
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if ((err1 = grib_get_string(h1, name, sval1, &len1)) != GRIB_SUCCESS) {
|
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get string value of [%s] in 1st field: %s\n",
|
2020-01-22 13:10:59 +00:00
|
|
|
name, grib_get_error_message(err1));
|
|
|
|
save_error(c, name);
|
|
|
|
}
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if ((err2 = grib_get_string(h2, name, sval2, &len2)) != GRIB_SUCCESS) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get string value of [%s] in 2nd field: %s\n",
|
2020-01-22 13:10:59 +00:00
|
|
|
name, grib_get_error_message(err2));
|
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS) {
|
|
|
|
if (grib_inline_strcmp(sval1, sval2) != 0) {
|
|
|
|
printInfo(h1);
|
|
|
|
printf("string [%s]: [%s] != [%s]\n",
|
|
|
|
name, sval1, sval2);
|
|
|
|
err1 = GRIB_VALUE_MISMATCH;
|
|
|
|
save_error(c, name);
|
|
|
|
}
|
|
|
|
}
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_context_free(h1->context, sval1);
|
|
|
|
grib_context_free(h2->context, sval2);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (err1)
|
|
|
|
return err1;
|
|
|
|
if (err2)
|
|
|
|
return err2;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
break;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
case GRIB_TYPE_LONG:
|
2024-01-06 20:49:08 +00:00
|
|
|
if (verbose) printf(" as long\n");
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
lval1 = (long*)grib_context_malloc(h1->context, len1 * sizeof(long));
|
|
|
|
lval2 = (long*)grib_context_malloc(h2->context, len2 * sizeof(long));
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if ((err1 = grib_get_long_array(h1, name, lval1, &len1)) != GRIB_SUCCESS) {
|
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get long value of [%s] in 1st field: %s\n",
|
2020-01-22 13:10:59 +00:00
|
|
|
name, grib_get_error_message(err1));
|
|
|
|
save_error(c, name);
|
|
|
|
}
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if ((err2 = grib_get_long_array(h2, name, lval2, &len2)) != GRIB_SUCCESS) {
|
2016-04-04 10:06:37 +00:00
|
|
|
printInfo(h1);
|
2024-01-03 13:17:23 +00:00
|
|
|
printf("Error: cannot get long value of [%s] in 2nd field: %s\n",
|
2020-01-22 13:10:59 +00:00
|
|
|
name, grib_get_error_message(err2));
|
|
|
|
save_error(c, name);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 != len2) {
|
|
|
|
printInfo(h1);
|
|
|
|
printf("Different size for \"%s\" [%ld] [%ld]\n", name, (long)len1, (long)len2);
|
|
|
|
err1 = GRIB_INTERNAL_ERROR;
|
|
|
|
save_error(c, name);
|
|
|
|
}
|
|
|
|
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 == len2) {
|
2023-03-04 18:51:14 +00:00
|
|
|
size_t i;
|
2020-01-22 13:10:59 +00:00
|
|
|
countdiff = 0;
|
|
|
|
for (i = 0; i < len1; i++)
|
|
|
|
if (lval1[i] != lval2[i])
|
|
|
|
countdiff++;
|
|
|
|
|
|
|
|
if (countdiff) {
|
|
|
|
printInfo(h1);
|
|
|
|
save_error(c, name);
|
|
|
|
err1 = GRIB_VALUE_MISMATCH;
|
|
|
|
if (len1 == 1)
|
|
|
|
printf("long [%s]: [%ld] != [%ld]\n",
|
|
|
|
name, *lval1, *lval2);
|
|
|
|
else
|
|
|
|
printf("long [%s] %d out of %ld different\n",
|
|
|
|
name, countdiff, (long)len1);
|
|
|
|
}
|
|
|
|
}
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
grib_context_free(h1->context, lval1);
|
|
|
|
grib_context_free(h2->context, lval2);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (err1)
|
|
|
|
return err1;
|
|
|
|
if (err2)
|
|
|
|
return err2;
|
|
|
|
break;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
case GRIB_TYPE_BYTES:
|
2024-03-01 12:10:27 +00:00
|
|
|
// We do not want to compare the message itself
|
|
|
|
return 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
case GRIB_TYPE_LABEL:
|
|
|
|
break;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
default:
|
2024-01-06 20:49:08 +00:00
|
|
|
if (verbose) printf("\n");
|
2020-01-22 13:10:59 +00:00
|
|
|
printInfo(h1);
|
|
|
|
save_error(c, name);
|
2024-03-23 19:19:08 +00:00
|
|
|
fprintf(stderr, "Cannot compare [%s], unsupported type %d\n", name, type1);
|
2020-01-22 13:10:59 +00:00
|
|
|
return GRIB_UNABLE_TO_COMPARE_ACCESSORS;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return GRIB_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2024-05-01 13:29:46 +00:00
|
|
|
static int compare_all_dump_keys(grib_handle* h1, grib_handle* h2, grib_runtime_options* options, int* pErr)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
int ret = 0;
|
|
|
|
const char* name = NULL;
|
|
|
|
grib_keys_iterator* iter = NULL;
|
|
|
|
iter = grib_keys_iterator_new(h1, 0, NULL);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
|
|
|
if (!iter) {
|
2023-12-23 12:46:29 +00:00
|
|
|
fprintf(stderr, "Error: unable to get keys iterator\n");
|
2016-04-04 10:06:37 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
while (grib_keys_iterator_next(iter)) {
|
|
|
|
grib_accessor* xa = grib_keys_iterator_get_accessor(iter);
|
|
|
|
name = grib_keys_iterator_get_name(iter);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-09-24 14:54:33 +00:00
|
|
|
if (blocklisted(name))
|
2020-01-22 13:10:59 +00:00
|
|
|
continue;
|
2024-08-11 23:54:29 +00:00
|
|
|
if (xa == NULL || (xa->flags_ & GRIB_ACCESSOR_FLAG_DUMP) == 0)
|
2020-01-22 13:10:59 +00:00
|
|
|
continue;
|
|
|
|
if (compare_values(options, h1, h2, name, GRIB_TYPE_UNDEFINED)) {
|
2024-05-01 13:29:46 +00:00
|
|
|
(*pErr)++;
|
2020-01-22 13:10:59 +00:00
|
|
|
write_messages(h1, h2);
|
|
|
|
ret = 1;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
grib_keys_iterator_delete(iter);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
static int compare_handles(grib_handle* h1, grib_handle* h2, grib_runtime_options* options)
|
2016-04-04 10:06:37 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
int err = 0;
|
|
|
|
int i = 0;
|
|
|
|
grib_keys_iterator* iter = NULL;
|
|
|
|
const char* name = NULL;
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (listFromCommandLine && onlyListed) {
|
|
|
|
for (i = 0; i < options->compare_count; i++) {
|
2020-09-24 14:54:33 +00:00
|
|
|
if (blocklisted((char*)options->compare[i].name))
|
2020-01-22 13:10:59 +00:00
|
|
|
continue;
|
2023-08-23 21:19:07 +00:00
|
|
|
if (options->compare[i].type == CODES_NAMESPACE) {
|
2020-01-22 13:10:59 +00:00
|
|
|
iter = grib_keys_iterator_new(h1, 0, options->compare[i].name);
|
2016-04-04 10:06:37 +00:00
|
|
|
if (!iter) {
|
2023-12-23 12:46:29 +00:00
|
|
|
fprintf(stderr, "Error: unable to get iterator\n");
|
2016-04-04 10:06:37 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
2020-01-22 13:10:59 +00:00
|
|
|
while (grib_keys_iterator_next(iter)) {
|
|
|
|
name = grib_keys_iterator_get_name(iter);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-09-24 14:54:33 +00:00
|
|
|
if (blocklisted(name))
|
2020-01-22 13:10:59 +00:00
|
|
|
continue;
|
|
|
|
if (compare_values(options, h1, h2, name, GRIB_TYPE_UNDEFINED)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
err++;
|
2020-01-22 13:10:59 +00:00
|
|
|
write_messages(h1, h2);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
grib_keys_iterator_delete(iter);
|
2020-01-22 13:10:59 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (compare_values(options, h1, h2, options->compare[i].name, options->compare[i].type))
|
2016-04-04 10:06:37 +00:00
|
|
|
err++;
|
2020-01-22 13:10:59 +00:00
|
|
|
write_messages(h1, h2);
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|
|
|
|
}
|
2020-01-22 13:10:59 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
const void *msg1 = NULL, *msg2 = NULL;
|
|
|
|
size_t size1 = 0, size2 = 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
/* int ii=0; */
|
2020-01-22 13:10:59 +00:00
|
|
|
GRIB_CHECK_NOLINE(grib_get_message(h1, &msg1, &size1), 0);
|
|
|
|
GRIB_CHECK_NOLINE(grib_get_message(h2, &msg2, &size2), 0);
|
2024-03-17 14:31:55 +00:00
|
|
|
if ( size1 == size2 && (0 == memcmp(msg1, msg2, size1)) ) {
|
2016-04-04 10:06:37 +00:00
|
|
|
return 0;
|
2020-01-22 13:10:59 +00:00
|
|
|
}
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (listFromCommandLine) {
|
|
|
|
for (i = 0; i < options->compare_count; i++) {
|
2020-09-24 14:54:33 +00:00
|
|
|
if (blocklisted(name))
|
2020-01-22 13:10:59 +00:00
|
|
|
continue;
|
2023-08-23 21:19:07 +00:00
|
|
|
if (options->compare[i].type == CODES_NAMESPACE) {
|
2020-01-22 13:10:59 +00:00
|
|
|
iter = grib_keys_iterator_new(h1, 0, options->compare[i].name);
|
2016-04-04 10:06:37 +00:00
|
|
|
if (!iter) {
|
2023-12-23 12:46:29 +00:00
|
|
|
fprintf(stderr, "Error: unable to get iterator for %s\n", options->compare[i].name);
|
2016-04-04 10:06:37 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
2020-01-22 13:10:59 +00:00
|
|
|
while (grib_keys_iterator_next(iter)) {
|
|
|
|
name = grib_keys_iterator_get_name(iter);
|
2016-04-04 10:06:37 +00:00
|
|
|
|
2020-09-24 14:54:33 +00:00
|
|
|
if (blocklisted(name))
|
2020-01-22 13:10:59 +00:00
|
|
|
continue;
|
|
|
|
if (compare_values(options, h1, h2, name, GRIB_TYPE_UNDEFINED)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
err++;
|
2020-01-22 13:10:59 +00:00
|
|
|
write_messages(h1, h2);
|
|
|
|
if (compare_all_dump_keys(h1, h2, options, &err)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
err++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
grib_keys_iterator_delete(iter);
|
2020-01-22 13:10:59 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (compare_values(options, h1, h2, options->compare[i].name, options->compare[i].type)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
err++;
|
2020-01-22 13:10:59 +00:00
|
|
|
write_messages(h1, h2);
|
|
|
|
if (compare_all_dump_keys(h1, h2, options, &err)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
err++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-22 13:10:59 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (compare_all_dump_keys(h1, h2, options, &err)) {
|
2016-04-04 10:06:37 +00:00
|
|
|
err++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2016-08-30 12:45:02 +00:00
|
|
|
int grib_no_handle_action(grib_runtime_options* options, int err)
|
2016-04-04 10:42:48 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
fprintf(dump_file, "\t\t\"ERROR: unreadable message\"\n");
|
2016-08-30 12:45:02 +00:00
|
|
|
return 0;
|
2016-04-04 10:06:37 +00:00
|
|
|
}
|