eccodes/tools/bufr_compare.cc

1325 lines
46 KiB
C++
Raw Permalink Normal View History

2014-07-06 13:18:40 +00:00
/*
2020-01-28 14:32:34 +00:00
* (C) Copyright 2005- ECMWF.
2014-07-06 13:18:40 +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 },
{ "2", 0, "Enable two-way comparison.\n", 0, 1, 0 },
{ "T:", 0, 0, 1, 0, "B" },
{ "c:", 0, 0, 0, 1, 0 },
{ "S:", "start", "First field to be processed.\n", 0, 1, 0 },
{ "E:", "end", "Last field to be processed.\n", 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 },
{ "H", 0, "Compare only message headers. Bit-by-bit compare on. Incompatible with -c option.\n", 0, 1, 0 },
{ "R:", 0, 0, 0, 1, 0 },
{ "A:", 0, 0, 0, 1, 0 },
{ "t:", "factor", "Compare data values using factor multiplied by the tolerance specified in options -R -A.\n", 0, 1, 0 },
{ "w:", 0, 0, 0, 1, 0 },
{ "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 },
{ "M", 0, 0, 1, 0, 0 },
{ "I", 0, 0, 1, 0, 0 },
{ "V", 0, 0, 0, 1, 0 },
{ "7", 0, 0, 0, 1, 0 },
2023-11-02 19:52:42 +00:00
{ "v", 0, 0, 0, 1, 0 },
{ "h", 0, 0, 0, 1, 0 },
2021-02-04 11:53:15 +00:00
};
int grib_options_count = sizeof(grib_options) / sizeof(grib_option);
const char* tool_description =
"Compare BUFR messages contained in two files."
"\n\tIf some differences are found it fails returning an error code."
"\n\tFloating-point values are compared exactly by default, different tolerances can be defined (see -A -R)."
"\n\tDefault behaviour: absolute error=0, bit-by-bit compare, same order in files.";
const char* tool_name = "bufr_compare";
const char* tool_online_doc = "https://confluence.ecmwf.int/display/ECC/bufr_compare";
const char* tool_usage = "[options] bufr_file1 bufr_file2";
2021-02-04 11:53:15 +00:00
2021-11-06 14:18:48 +00:00
typedef double (*compare_double_proc)(const double*, const double*, const double*);
2014-07-06 13:18:40 +00:00
typedef struct grib_error grib_error;
2020-01-22 13:10:59 +00:00
struct grib_error
{
char* key;
int count;
grib_error* next;
2014-07-06 13:18:40 +00:00
};
static grib_error* error_summary;
static compare_double_proc compare_double;
2020-01-22 13:10:59 +00:00
static double global_tolerance = 0;
2020-09-24 14:54:33 +00:00
static grib_string_list* blocklist = 0;
static grib_string_list* keys_list = NULL; /* Used to determine rank of key */
2020-01-22 13:10:59 +00:00
static int isLeafKey = 0; /* 0 if key is top-level, 1 if key has no children attributes */
static int compareAbsolute = 1;
2014-07-06 13:18:40 +00:00
2016-04-14 11:11:48 +00:00
static int compare_handles(grib_handle* handle1, grib_handle* handle2, grib_runtime_options* options);
static int compare_attribute(grib_handle* handle1, grib_handle* handle2, grib_runtime_options* options,
2020-01-22 13:10:59 +00:00
grib_accessor* a, const char* prefix, int* err);
2020-01-22 13:10:59 +00:00
static int error = 0;
static int count = 0;
static int force = 0;
2018-03-12 15:51:04 +00:00
/* ECC-651: Boolean 'two_way' set to 1 when '-2' option used */
2020-01-22 13:10:59 +00:00
static int two_way = 0;
/* Boolean 'handles_swapped' relevant in 'two_way' mode:
2018-03-12 15:51:04 +00:00
* 0 means: h1 is first file, h2 is second file
* 1 means: h1 is second file, h2 is first file
*/
2020-01-22 13:10:59 +00:00
static int handles_swapped = 0;
static double maxAbsoluteError = 1e-19;
2020-01-22 13:10:59 +00:00
static int onlyListed = 1;
static int headerMode = 0;
static int morein1 = 0;
static int morein2 = 0;
2022-04-06 12:22:24 +00:00
static int listFromCommandLine = 0;
static int verbose = 0;
static double tolerance_factor = 1;
static int write_error = 0;
static int write_count = 0;
2021-02-04 11:53:15 +00:00
static grib_handle* global_handle = NULL;
static int counter = 0;
static int start = -1;
static int end = -1;
2014-07-06 13:18:40 +00:00
/* Create the list of keys (global variable keys_list) */
static void new_keys_list()
{
grib_context* c = grib_context_get_default();
2020-01-22 13:10:59 +00:00
keys_list = (grib_string_list*)grib_context_malloc_clear(c, sizeof(grib_string_list));
if (!keys_list) {
2023-08-08 18:49:28 +00:00
fprintf(stderr, "%s: Failed to allocate memory for keys list", tool_name);
exit(1);
}
}
/* Free the keys list (global variable keys_list) */
static void release_keys_list()
{
2020-01-22 13:10:59 +00:00
grib_string_list* next = keys_list;
2020-02-19 16:39:06 +00:00
grib_string_list* cur = NULL;
2020-01-22 13:10:59 +00:00
grib_context* c = grib_context_get_default();
while (next) {
cur = next;
next = next->next;
grib_context_free(c, cur->value);
grib_context_free(c, cur);
}
}
2021-11-06 14:18:48 +00:00
GRIB_INLINE static double compare_double_absolute(const double* a, const double* b, const double* err)
{
2020-01-22 13:10:59 +00:00
double ret = 0;
double d = fabs(*a - *b);
if (d > *err) {
2020-01-22 13:10:59 +00:00
ret = d;
}
return ret;
/* return fabs(*a-*b) > *err ? fabs(*a-*b) : 0; */
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
static void write_message(grib_handle* h, const char* str)
{
2020-01-22 13:10:59 +00:00
const void* m;
size_t s;
char fname[1024] = {0,};
2018-11-01 12:55:47 +00:00
FILE* fh;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
grib_get_message(h, &m, &s);
snprintf(fname, sizeof(fname), "%s_%d.bufr", str, write_count);
2014-07-06 13:18:40 +00:00
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);
exit(GRIB_IO_PROBLEM);
}
2014-07-06 13:18:40 +00:00
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);
exit(GRIB_IO_PROBLEM);
}
2014-07-06 13:18:40 +00:00
fclose(fh);
2014-07-06 13:18:40 +00:00
}
2016-04-14 11:11:48 +00:00
static void write_messages(grib_handle* handle1, grib_handle* handle2)
{
2020-01-22 13:10:59 +00:00
if (!write_error)
return;
write_count++;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
write_message(handle1, "error1");
write_message(handle2, "error2");
2014-07-06 13:18:40 +00:00
}
2021-11-06 14:18:48 +00:00
static double compare_double_relative(const double* a, const double* b, const double* err)
{
double relativeError;
2020-01-22 13:10:59 +00:00
if (fabs(*a) <= maxAbsoluteError || fabs(*b) <= maxAbsoluteError)
relativeError = fabs(*a - *b);
else if (fabs(*b) > fabs(*a))
2020-01-22 13:10:59 +00:00
relativeError = fabs((*a - *b) / *b);
else
2020-01-22 13:10:59 +00:00
relativeError = fabs((*a - *b) / *a);
return relativeError > *err ? relativeError : 0;
2014-07-06 13:18:40 +00:00
}
2020-09-24 14:54:33 +00:00
static int blocklisted(const char* name)
{
2020-09-24 14:54:33 +00:00
grib_string_list* b = blocklist;
if (!name) return 0;
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))
return 1;
2020-01-22 13:10:59 +00:00
b = b->next;
}
return 0;
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
static double relative_error(double a, double b, double err)
{
double relativeError;
2020-01-22 13:10:59 +00:00
if (fabs(a) <= maxAbsoluteError || fabs(b) <= maxAbsoluteError)
relativeError = fabs(a - b);
else if (fabs(b) > fabs(a))
2020-01-22 13:10:59 +00:00
relativeError = fabs((a - b) / b);
else
2020-01-22 13:10:59 +00:00
relativeError = fabs((a - b) / a);
2020-01-22 13:10:59 +00:00
return relativeError;
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
int main(int argc, char* argv[])
{
2020-01-22 13:10:59 +00:00
return grib_tool(argc, argv);
}
2014-07-06 13:18:40 +00:00
int grib_tool_before_getopt(grib_runtime_options* options)
{
return 0;
2014-07-06 13:18:40 +00:00
}
int grib_tool_init(grib_runtime_options* options)
{
2024-03-01 12:10:27 +00:00
int ret = 0;
2020-01-22 13:10:59 +00:00
grib_context* context = grib_context_get_default();
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
options->strict = 1;
if (grib_options_on("S:"))
2020-01-22 13:10:59 +00:00
start = atoi(grib_options_get_option("S:"));
2014-07-06 13:18:40 +00:00
if (grib_options_on("E:"))
2020-01-22 13:10:59 +00:00
end = atoi(grib_options_get_option("E:"));
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (grib_options_on("f"))
force = 1;
else
force = 0;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (grib_options_on("2"))
two_way = 1;
else
two_way = 0;
2018-03-12 15:51:04 +00:00
2020-01-22 13:10:59 +00:00
if (grib_options_on("d"))
write_error = 1;
else
write_error = 0;
2014-07-06 13:18:40 +00:00
verbose = grib_options_on("v");
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
listFromCommandLine = 0;
if (grib_options_on("c:") || grib_options_on("e"))
2020-01-22 13:10:59 +00:00
listFromCommandLine = 1;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (grib_options_on("a"))
onlyListed = 0;
else
onlyListed = 1;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (grib_options_on("H"))
headerMode = 1;
else
headerMode = 0;
2014-07-06 13:18:40 +00:00
if (grib_options_on("H") && grib_options_on("c:")) {
2023-08-08 18:49:28 +00:00
fprintf(stderr, "%s: -H and -c options are incompatible. Choose one of the two please.\n",tool_name);
exit(1);
}
if (grib_options_on("a") && !grib_options_on("c:")) {
2023-08-08 18:49:28 +00:00
fprintf(stderr, "%s: -a option requires -c option. Please define a list of keys with the -c option.\n",tool_name);
exit(1);
2014-07-06 13:18:40 +00:00
}
if (grib_options_on("b:")) {
2020-01-22 13:10:59 +00:00
grib_string_list* next = 0;
int ii = 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 (ii = 1; ii < options->set_values_count; ii++) {
next->next = (grib_string_list*)grib_context_malloc_clear(context, sizeof(grib_string_list));
next->next->value = grib_context_strdup(context, options->set_values[ii].name);
next = next->next;
}
2020-09-24 14:54:33 +00:00
context->blocklist = blocklist;
2014-07-06 13:18:40 +00:00
}
/* Check 1st file is not a directory */
2020-07-17 14:37:57 +00:00
exit_if_input_is_directory(tool_name, options->infile_extra->name);
2023-08-22 16:26:42 +00:00
options->random = 0;
options->infile_extra->file = fopen(options->infile_extra->name, "r");
2023-08-22 16:26:42 +00:00
if (!options->infile_extra->file) {
perror(options->infile_extra->name);
exit(1);
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
global_tolerance = 0;
compare_double = &compare_double_absolute;
2014-07-06 13:18:40 +00:00
if (grib_options_on("R:")) {
2020-01-22 13:10:59 +00:00
global_tolerance = 0;
compare_double = &compare_double_relative;
compareAbsolute = 0;
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
if (grib_options_on("A:")) {
if (grib_options_on("R:")) {
maxAbsoluteError = atof(grib_options_get_option("A:"));
2020-01-22 13:10:59 +00:00
}
else {
compare_double = &compare_double_absolute;
global_tolerance = atof(grib_options_get_option("A:"));
}
}
if (grib_options_on("t:"))
2022-04-06 12:22:24 +00:00
tolerance_factor = atof(grib_options_get_option("t:"));
if (grib_options_on("R:")) {
2024-03-01 12:10:27 +00:00
char* sarg = grib_options_get_option("R:");
2020-01-22 13:10:59 +00:00
options->tolerance_count = MAX_KEYS;
2024-03-01 12:10:27 +00:00
ret = parse_keyval_string(tool_name, sarg, 1, GRIB_TYPE_DOUBLE, options->tolerance, &(options->tolerance_count));
if (ret == GRIB_INVALID_ARGUMENT) {
usage();
exit(1);
}
}
{
/* 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) {
if (path_is_directory(infile->name)) {
/* Take the filename of the 1st file and append to dir */
char bufr[2048] = {0,};
2024-08-11 23:54:29 +00:00
/* options->infile_extra->name_ is the 1st file */
snprintf(bufr, sizeof(bufr), "%s%c%s",
2015-07-27 12:27:03 +00:00
infile->name,
get_dir_separator_char(),
extract_filename(options->infile_extra->name));
infile->name = strdup(bufr);
}
}
}
2014-07-06 13:18:40 +00:00
/* Turn off GRIB multi-field support mode. Not relevant for BUFR */
grib_multi_support_off(grib_context_get_default());
return 0;
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
int grib_tool_new_filename_action(grib_runtime_options* options, const char* file)
2016-08-04 10:07:00 +00:00
{
return 0;
2014-07-06 13:18:40 +00:00
}
2016-08-04 10:07:00 +00:00
2020-01-22 13:10:59 +00:00
int grib_tool_new_file_action(grib_runtime_options* options, grib_tools_file* file)
2016-08-04 10:07:00 +00:00
{
2020-07-17 14:37:57 +00:00
exit_if_input_is_directory(tool_name, file->name);
return 0;
2014-07-06 13:18:40 +00:00
}
static void printInfo(grib_handle* h)
{
2020-01-22 13:10:59 +00:00
printf("== %d == DIFFERENCE == ", count);
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
static void print_index_key_values(grib_index* index, int icounter, const char* error_message)
{
2020-01-22 13:10:59 +00:00
grib_index_key* keys = index->keys;
printf("== %d == ", icounter);
if (error_message)
printf("%s == ", error_message);
while (keys) {
2020-01-22 13:10:59 +00:00
printf("%s=%s ", keys->name, keys->value);
keys = keys->next;
}
printf("\n");
2014-07-06 13:18:40 +00:00
}
2019-05-10 13:11:01 +00:00
static grib_handle* bufr_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)
{
2020-01-22 13:10:59 +00:00
return codes_handle_new_from_file(c, f, PRODUCT_BUFR, err);
2014-07-06 13:18:40 +00:00
}
2016-04-14 11:11: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;
count++;
if (options->through_index) {
2020-01-22 13:10:59 +00:00
grib_index* idx1 = options->index1;
verbose = 0;
counter++;
2020-01-22 13:10:59 +00:00
if (start > 0 && counter < start)
return 0;
if (end > 0 && counter > end) {
options->stop = 1;
return 0;
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
grib_index_search_same(idx1, h);
global_handle = codes_new_from_index(idx1, CODES_BUFR, &err);
if (options->verbose) {
2020-01-22 13:10:59 +00:00
off_t offset = 0;
char* filename = grib_get_field_file(options->index2, &offset);
printf("file1=\"%s\" ", filename);
filename = grib_get_field_file(options->index1, &offset);
printf("file2=\"%s\" \n", filename);
print_index_key_values(options->index1, counter, NULL);
}
2014-07-06 13:18:40 +00:00
2016-04-14 11:11:48 +00:00
if (!global_handle) {
if (!options->verbose)
2020-01-22 13:10:59 +00:00
print_index_key_values(idx1, counter, "NOT FOUND ");
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (!global_handle || err != GRIB_SUCCESS) {
morein1++;
2020-01-22 13:10:59 +00:00
if (global_handle)
grib_handle_delete(global_handle);
return 0;
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (compare_handles(h, global_handle, options)) {
error++;
2023-08-24 21:52:33 +00:00
write_messages(h, global_handle);
2020-01-22 13:10:59 +00:00
if (!force)
exit(1);
}
2014-07-06 13:18:40 +00:00
2016-04-14 11:11:48 +00:00
grib_handle_delete(global_handle);
2014-07-06 13:18:40 +00:00
return 0;
2020-01-22 13:10:59 +00:00
}
2023-08-22 16:26:42 +00:00
else {
2020-01-22 13:10:59 +00:00
global_handle = bufr_handle_new_from_file_x(h->context, options->infile_extra->file, options->mode, 0, &err);
2023-08-22 16:26:42 +00:00
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (!global_handle || err != GRIB_SUCCESS) {
morein2++;
2020-01-22 13:10:59 +00:00
if (global_handle)
grib_handle_delete(global_handle);
return 0;
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (compare_handles(global_handle, h, options)) {
error++;
2023-08-24 21:52:33 +00:00
write_messages(global_handle, h);
if (!two_way) {
/* If two_way mode: Don't exit yet. Show further differences */
2020-01-22 13:10:59 +00:00
if (!force)
exit(1);
}
}
2018-03-12 15:51:04 +00:00
if (two_way) {
/* ECC-651 and ECC-431 */
2018-03-12 15:51:04 +00:00
handles_swapped = 1;
2020-01-22 13:10:59 +00:00
if (verbose)
printf(" Swapping handles (two-way mode)\n");
if (compare_handles(h, global_handle, options)) {
2018-03-12 15:51:04 +00:00
error++;
2023-08-24 21:52:33 +00:00
write_messages(h, global_handle);
2020-01-22 13:10:59 +00:00
if (!force)
exit(1);
}
else {
if (error) {
/* Error from first pass */
2020-01-22 13:10:59 +00:00
if (!force)
exit(1);
}
2018-03-12 15:51:04 +00:00
}
}
2014-07-06 13:18:40 +00:00
2016-04-14 11:11:48 +00:00
grib_handle_delete(global_handle);
2014-07-06 13:18:40 +00:00
return 0;
2014-07-06 13:18:40 +00:00
}
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-22 16:26:42 +00:00
if (!options->through_index) {
2020-01-22 13:10:59 +00:00
global_handle = codes_bufr_handle_new_from_file(h->context, options->infile_extra->file, &err);
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (!global_handle || err != GRIB_SUCCESS)
morein2++;
2014-07-06 13:18:40 +00:00
2016-04-14 11:11:48 +00:00
grib_handle_delete(global_handle);
}
grib_handle_delete(h);
count++;
2014-07-06 13:18:40 +00:00
return 0;
2014-07-06 13:18:40 +00:00
}
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();
2014-07-06 13:18:40 +00:00
/*if (grib_options_on("w:")) return 0;*/
2020-01-22 13:10:59 +00:00
while ((global_handle = codes_bufr_handle_new_from_file(c, options->infile_extra->file, &err))) {
morein1++;
grib_handle_delete(global_handle);
}
2019-05-10 13:11:01 +00:00
2020-01-22 13:10:59 +00:00
error += morein1 + morein2;
2019-05-10 13:11:01 +00:00
if (error) {
printf("\n## ERRORS SUMMARY #######\n");
}
2020-01-22 13:10:59 +00:00
if (morein1 > 0) {
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);
}
2020-01-22 13:10:59 +00:00
if (morein2 > 0) {
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);
2014-07-06 13:18:40 +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;
}
2020-01-22 13:10:59 +00:00
printf("##\n## %d different messages out of %d\n\n", error, count);
}
if (options->through_index) {
grib_index_delete(options->index1);
grib_index_delete(options->index2);
}
release_keys_list();
2020-01-22 13:10:59 +00:00
if (error != 0)
exit(1);
return 0;
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
static void save_error(grib_context* c, const char* key)
{
2020-01-22 13:10:59 +00:00
grib_error* e = 0;
grib_error* next = 0;
int saved = 0;
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);
return;
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
e = error_summary;
next = e;
2014-07-06 13:18:40 +00:00
while (next) {
2020-01-22 13:10:59 +00:00
if (!strcmp(next->key, key)) {
next->count++;
2020-01-22 13:10:59 +00:00
saved = 1;
break;
}
2020-01-22 13:10:59 +00:00
e = next;
next = next->next;
2014-07-06 13:18:40 +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);
}
2014-07-06 13:18:40 +00:00
}
static char* double_as_string(grib_context* c, double v)
{
2020-01-22 13:10:59 +00:00
char* sval = (char*)grib_context_malloc_clear(c, sizeof(char) * 40);
if (v == GRIB_MISSING_DOUBLE)
snprintf(sval, 32, "MISSING");
2020-01-22 13:10:59 +00:00
else
snprintf(sval, 32, "%.20e", v);
return sval;
}
/* Return the part of the key name without the '#rank#' part */
static char* get_keyname_without_rank(const char* name)
{
2020-01-22 13:10:59 +00:00
const char* p = name;
char* pEnd;
2020-01-22 13:10:59 +00:00
char* ret = NULL;
2020-01-22 13:10:59 +00:00
if (*p == '#') {
strtol(++p, &pEnd, 10);
if (*pEnd != '#') {
2023-06-26 20:21:07 +00:00
DEBUG_ASSERT(!"Badly formed rank in key");
2020-01-22 13:10:59 +00:00
}
else {
/* Take everything after 2nd '#' */
2020-01-22 13:10:59 +00:00
grib_context* c = grib_context_get_default();
2018-12-20 13:30:43 +00:00
pEnd++;
2020-01-22 13:10:59 +00:00
ret = grib_context_strdup(c, pEnd);
}
}
return ret;
}
static int are_strings_both_missing(grib_handle* h1, grib_handle* h2, const char* key,
2021-10-02 20:28:31 +00:00
const char* s1, const char* s2,
size_t slen1, size_t slen2)
{
/* Note: one string could have all its bits=1 and the other empty */
int is_miss_1 = 0, is_miss_2 = 0;
if (strlen(s1) == 0) {
is_miss_1 = 1;
2021-10-02 20:28:31 +00:00
}
else {
grib_accessor* a1 = grib_find_accessor(h1, key);
2021-10-02 20:28:31 +00:00
is_miss_1 = grib_is_missing_string(a1, (unsigned char*)s1, slen1);
}
if (strlen(s2) == 0) {
is_miss_2 = 1;
2021-10-02 20:28:31 +00:00
}
else {
grib_accessor* a2 = grib_find_accessor(h2, key);
2021-10-02 20:28:31 +00:00
is_miss_2 = grib_is_missing_string(a2, (unsigned char*)s2, slen2);
}
if (is_miss_1 && is_miss_2) {
return 1; /* both classed as missing */
}
return 0;
}
2021-10-02 20:28:31 +00:00
static int strings_are_different(grib_handle* h1, grib_handle* h2, const char* key,
const char* s1, const char* s2,
size_t slen1, size_t slen2)
{
if (strcmp(s1, s2) == 0) {
2021-10-02 20:28:31 +00:00
return 0;
}
/* Strings are different. Now check if strings are 'missing'.
* Note: one string could have all its bits=1 and the other empty */
if (are_strings_both_missing(h1, h2, key, s1, s2, slen1, slen2)) {
return 0;
}
return 1;
}
2020-01-22 13:10:59 +00:00
static int compare_values(grib_runtime_options* options, grib_handle* handle1, grib_handle* handle2, const char* name, int type)
{
size_t len1 = 0;
size_t len2 = 0;
2020-01-22 13:10:59 +00:00
int err = 0, i = 0;
int err1;
int err2;
2020-01-22 13:10:59 +00:00
int type1, type2;
int countdiff;
2020-01-22 13:10:59 +00:00
int isMissing1 = 0, isMissing2 = 0;
char *sval1 = NULL, *sval2 = NULL;
char **svals1 = NULL, **svals2 = NULL;
double *dval1 = NULL, *dval2 = NULL;
long *lval1 = NULL, *lval2 = NULL;
2020-01-22 13:10:59 +00:00
double maxdiff = 0;
double value_tolerance = 0;
grib_context* c = handle1->context;
const char* first_str = (handles_swapped == 0 ? "1st" : "2nd");
const char* second_str = (handles_swapped == 0 ? "2nd" : "1st");
type1 = type;
type2 = type;
if (verbose && !handles_swapped)
printf(" comparing %s", name);
if (type1 == GRIB_TYPE_UNDEFINED && (err = grib_get_native_type(handle1, name, &type1)) != GRIB_SUCCESS) {
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get type of [%s] in %s field: %s\n", name, first_str, grib_get_error_message(err));
2020-01-22 13:10:59 +00:00
save_error(c, name);
return err;
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (type2 == GRIB_TYPE_UNDEFINED && (err = grib_get_native_type(handle2, name, &type2)) != GRIB_SUCCESS) {
if (err == GRIB_NOT_FOUND) {
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2020-01-22 13:10:59 +00:00
printf("[%s] not found in %s field\n", name, second_str);
save_error(c, name);
return err;
}
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get type of [%s] in %s field: %s\n", name, second_str, grib_get_error_message(err));
2020-01-22 13:10:59 +00:00
save_error(c, name);
return err;
2014-07-06 13:18:40 +00:00
}
2016-08-05 17:38:47 +00:00
/* if(type1 != type2) {
printInfo(handle1);
printf("Warning, [%s] has different types: 1st field: [%s], 2nd field: [%s]\n",
name,grib_get_type_name(type1),grib_get_type_name(type2));
return GRIB_TYPE_MISMATCH;
} */
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (type1 == GRIB_TYPE_LABEL)
return err;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (type1 == GRIB_TYPE_SECTION)
return err;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if ((err = grib_get_size(handle1, name, &len1)) != GRIB_SUCCESS) {
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get size of [%s] in %s field: %s\n", name, first_str, grib_get_error_message(err));
2020-01-22 13:10:59 +00:00
save_error(c, name);
return err;
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
if ((err = grib_get_size(handle2, name, &len2)) != GRIB_SUCCESS) {
if (err == GRIB_NOT_FOUND) {
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2020-01-22 13:10:59 +00:00
printf("[%s] not found in %s field\n", name, second_str);
save_error(c, name);
return err;
}
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get size of [%s] in %s field: %s\n", name, second_str, grib_get_error_message(err));
2020-01-22 13:10:59 +00:00
save_error(c, name);
return err;
}
2020-01-22 13:10:59 +00:00
2018-03-12 15:51:04 +00:00
if (handles_swapped) {
/* Comparing a second time with handles swapped. Do not compare keys common to both handles */
return GRIB_SUCCESS;
}
2016-08-05 17:38:47 +00:00
/* if(len1 != len2 && type1 != GRIB_TYPE_STRING) {
printInfo(handle1);
printf("[%s] has different size: 1st field: %ld, 2nd field: %ld\n",name,(long)len1,(long)len2);
save_error(c,name);
return GRIB_COUNT_MISMATCH;
} */
2014-07-06 13:18:40 +00:00
2024-12-20 12:58:07 +00:00
ECCODES_ASSERT(options->mode == MODE_BUFR);
/* TODO: Ignore missing values for keys in BUFR. Not yet implemented */
2024-01-13 20:58:44 +00:00
//isMissing1 = ((grib_is_missing(handle1, name, &err1) == 1) && (err1 == 0)) ? 1 : 0;
//isMissing2 = ((grib_is_missing(handle2, name, &err2) == 1) && (err2 == 0)) ? 1 : 0;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if ((isMissing1 == 1) && (isMissing2 == 1)) {
2024-01-13 20:58:44 +00:00
// if (verbose) printf(" is set to missing in both fields\n");
return GRIB_SUCCESS;
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (isMissing1 == 1) {
2024-01-13 20:58:44 +00:00
// if (verbose) printf(" is set to missing in %s field\n", first_str);
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2020-01-22 13:10:59 +00:00
printf("%s is set to missing in %s field is not missing in %s field\n", name, first_str, second_str);
err1 = GRIB_VALUE_MISMATCH;
2020-01-22 13:10:59 +00:00
save_error(c, name);
return GRIB_VALUE_MISMATCH;
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (isMissing2 == 1) {
2024-01-13 20:58:44 +00:00
// if (verbose) printf(" is set to missing in %s field\n", first_str);
2016-04-14 11:11:48 +00:00
printInfo(handle1);
2020-01-22 13:10:59 +00:00
printf("%s is set to missing in %s field is not missing in %s field\n", name, second_str, first_str);
err1 = GRIB_VALUE_MISMATCH;
2020-01-22 13:10:59 +00:00
save_error(c, name);
return GRIB_VALUE_MISMATCH;
}
2020-01-22 13:10:59 +00:00
switch (type1) {
case GRIB_TYPE_STRING:
if (verbose)
printf(" as string\n");
if (len1 == len2 && len1 == 1) {
/* Single string */
size_t slen1 = 0;
size_t slen2 = 0;
/* See ECC-710: It is very slow getting the key length this way */
/*grib_get_string_length(handle1,name,&slen1);*/
/*grib_get_string_length(handle2,name,&slen2);*/
slen1 = slen2 = 4096; /* Significantly faster to use an upper bound */
2021-10-02 20:28:31 +00:00
sval1 = (char*)grib_context_malloc(handle1->context, slen1 * sizeof(char));
sval2 = (char*)grib_context_malloc(handle2->context, slen2 * sizeof(char));
if ((err1 = grib_get_string(handle1, name, sval1, &slen1)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get string value of [%s] in %s field: %s\n",
2021-10-02 20:28:31 +00:00
name, first_str, grib_get_error_message(err1));
save_error(c, name);
}
if ((err2 = grib_get_string(handle2, name, sval2, &slen2)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get string value of [%s] in %s field: %s\n",
name, second_str, grib_get_error_message(err2));
save_error(c, name);
}
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS) {
2021-10-02 20:28:31 +00:00
if (strings_are_different(handle1, handle2, name, sval1, sval2, slen1, slen2)) {
printInfo(handle1);
printf("string [%s]: [%s] != [%s]\n", name, sval1, sval2);
err1 = GRIB_VALUE_MISMATCH;
save_error(c, name);
}
}
grib_context_free(handle1->context, sval1);
grib_context_free(handle2->context, sval2);
}
else {
/* Array of strings */
2021-10-14 12:39:05 +00:00
size_t ii = 0;
2021-10-02 20:28:31 +00:00
svals1 = (char**)grib_context_malloc_clear(handle1->context, len1 * sizeof(char*));
svals2 = (char**)grib_context_malloc_clear(handle2->context, len2 * sizeof(char*));
2014-07-06 13:18:40 +00:00
if ((err1 = grib_get_string_array(handle1, name, svals1, &len1)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get string value of [%s] in %s field: %s\n",
name, first_str, grib_get_error_message(err1));
save_error(c, name);
}
if ((err2 = grib_get_string_array(handle2, name, svals2, &len2)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get string value of [%s] in %s field: %s\n",
name, second_str, grib_get_error_message(err2));
save_error(c, name);
}
2014-07-06 13:18:40 +00:00
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 != len2) {
printInfo(handle1);
printf("Different size for \"%s\" [%zu] [%zu]\n", name, len1, len2);
err1 = GRIB_INTERNAL_ERROR;
save_error(c, name);
}
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 == len2) {
countdiff = 0;
for (ii = 0; ii < len1; ii++) {
2021-10-02 20:28:31 +00:00
if (strings_are_different(handle1, handle2, name, svals1[ii], svals2[ii], len1, len2)) {
countdiff++;
}
2020-01-22 13:10:59 +00:00
}
if (countdiff) {
2020-01-22 13:10:59 +00:00
printInfo(handle1);
save_error(c, name);
err1 = GRIB_VALUE_MISMATCH;
if (len1 == 1)
printf("string [%s]: [%s] != [%s]\n", name, *svals1, *svals2);
else
printf("string [%s] %d out of %zu different\n", name, countdiff, len1);
2020-01-22 13:10:59 +00:00
}
}
2021-10-14 12:39:05 +00:00
for (ii = 0; ii < len1; ++ii) grib_context_free(c, svals1[ii]);
for (ii = 0; ii < len2; ++ii) grib_context_free(c, svals2[ii]);
grib_context_free(handle1->context, svals1);
grib_context_free(handle1->context, svals2);
}
2020-01-22 13:10:59 +00:00
if (err1)
return err1;
if (err2)
return err2;
2020-01-22 13:10:59 +00:00
break;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
case GRIB_TYPE_LONG:
if (verbose)
printf(" as long\n");
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
lval1 = (long*)grib_context_malloc(handle1->context, len1 * sizeof(long));
lval2 = (long*)grib_context_malloc(handle2->context, len2 * sizeof(long));
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if ((err1 = grib_get_long_array(handle1, name, lval1, &len1)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get long value of [%s] in %s field: %s\n",
2020-01-22 13:10:59 +00:00
name, first_str, grib_get_error_message(err1));
save_error(c, name);
}
2020-01-22 13:10:59 +00:00
if ((err2 = grib_get_long_array(handle2, name, lval2, &len2)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get long value of [%s] in %s field: %s\n",
2020-01-22 13:10:59 +00:00
name, second_str, grib_get_error_message(err2));
save_error(c, name);
}
2020-01-22 13:10:59 +00:00
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 != len2) {
2016-04-14 11:11:48 +00:00
printInfo(handle1);
printf("Different size for \"%s\" [%zu] [%zu]\n", name, len1, len2);
2020-01-22 13:10:59 +00:00
err1 = GRIB_INTERNAL_ERROR;
save_error(c, name);
}
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 == len2) {
size_t ii;
2020-01-22 13:10:59 +00:00
countdiff = 0;
for (ii = 0; ii < len1; ii++)
if (lval1[ii] != lval2[ii])
countdiff++;
if (countdiff) {
printInfo(handle1);
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 %zu different\n", name, countdiff, len1);
2020-01-22 13:10:59 +00:00
}
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
grib_context_free(handle1->context, lval1);
grib_context_free(handle2->context, lval2);
if (err1)
return err1;
if (err2)
return err2;
break;
case GRIB_TYPE_DOUBLE:
if (verbose)
printf(" as double\n");
dval1 = (double*)grib_context_malloc(handle1->context, len1 * sizeof(double));
dval2 = (double*)grib_context_malloc(handle2->context, len2 * sizeof(double));
value_tolerance = global_tolerance;
if (!compareAbsolute) {
int all_specified = 0; /* =1 if relative comparison with "all" specified */
for (i = 0; i < options->tolerance_count; i++) {
if (!strcmp((options->tolerance[i]).name, "all")) {
all_specified = 1;
value_tolerance = (options->tolerance[i]).double_value;
break;
2020-01-22 13:10:59 +00:00
}
}
if (!all_specified) {
char* basename = NULL;
for (i = 0; i < options->tolerance_count; i++) {
if (strcmp((options->tolerance[i]).name, name) == 0) {
value_tolerance = (options->tolerance[i]).double_value;
break;
}
2020-01-22 13:10:59 +00:00
else {
/* Check if the key without its rank has a relative tolerance */
basename = get_keyname_without_rank(name);
if (basename && strcmp((options->tolerance[i]).name, basename) == 0) {
value_tolerance = (options->tolerance[i]).double_value;
break;
}
}
}
2020-01-22 13:10:59 +00:00
if (basename)
grib_context_free(handle1->context, basename);
}
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if ((err1 = grib_get_double_array(handle1, name, dval1, &len1)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get double value of [%s] in %s field: %s\n",
2020-01-22 13:10:59 +00:00
name, first_str, grib_get_error_message(err1));
save_error(c, name);
}
2020-01-22 13:10:59 +00:00
if ((err2 = grib_get_double_array(handle2, name, dval2, &len2)) != GRIB_SUCCESS) {
printInfo(handle1);
2024-01-03 13:17:23 +00:00
printf("Error: cannot get double value of [%s] in %s field: %s\n",
2020-01-22 13:10:59 +00:00
name, second_str, grib_get_error_message(err2));
save_error(c, name);
}
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 != len2) {
2016-04-14 11:11:48 +00:00
printInfo(handle1);
printf("Different size for \"%s\" [%zu] [%zu]\n", name, len1, len2);
2020-01-22 13:10:59 +00:00
save_error(c, name);
}
if (err1 == GRIB_SUCCESS && err2 == GRIB_SUCCESS && len1 == len2) {
2023-12-23 12:54:52 +00:00
int imaxdiff = 0;
2020-01-22 13:10:59 +00:00
double diff;
2021-02-04 11:53:15 +00:00
double *pv1, *pv2;
2020-01-22 13:10:59 +00:00
maxdiff = 0;
countdiff = 0;
pv1 = dval1;
pv2 = dval2;
value_tolerance *= tolerance_factor;
if (verbose)
printf(" (%d values) tolerance=%g\n", (int)len1, value_tolerance);
for (i = 0; i < len1; i++) {
if ((diff = compare_double(pv1++, pv2++, &value_tolerance)) != 0) {
countdiff++;
if (maxdiff < diff) {
maxdiff = diff;
imaxdiff = i;
}
err1 = GRIB_VALUE_MISMATCH;
}
2020-01-22 13:10:59 +00:00
}
if (countdiff) {
printInfo(handle1);
save_error(c, name);
if (len1 > 1) {
printf("double [%s]: %d out of %zu different\n", name, countdiff, len1);
2020-01-22 13:10:59 +00:00
if (dval1[imaxdiff] != GRIB_MISSING_DOUBLE && dval2[imaxdiff] != GRIB_MISSING_DOUBLE) {
if (compareAbsolute)
printf(" max");
printf(" absolute diff. = %.16e,", fabs(dval1[imaxdiff] - dval2[imaxdiff]));
if (!compareAbsolute)
printf(" max");
printf(" relative diff. = %g", relative_error(dval1[imaxdiff], dval2[imaxdiff], value_tolerance));
printf("\n\tmax diff. element %d: %.20e %.20e",
imaxdiff, dval1[imaxdiff], dval2[imaxdiff]);
printf("\n\ttolerance=%.16e", value_tolerance);
}
else {
/* One or both values are missing */
char* svalA = double_as_string(c, dval1[imaxdiff]);
char* svalB = double_as_string(c, dval2[imaxdiff]);
printf("\tdiff. element %d: %s %s", imaxdiff, svalA, svalB);
grib_context_free(c, svalA);
grib_context_free(c, svalB);
}
printf("\n");
}
2020-01-22 13:10:59 +00:00
else {
if (dval1[0] != GRIB_MISSING_DOUBLE && dval2[0] != GRIB_MISSING_DOUBLE) {
printf("double [%s]: [%.20e] != [%.20e]\n", name, dval1[0], dval2[0]);
printf("\tabsolute diff. = %g,", fabs(dval1[0] - dval2[0]));
printf(" relative diff. = %g\n", relative_error(dval1[0], dval2[0], value_tolerance));
printf("\ttolerance=%g\n", value_tolerance);
}
else {
/* One or both values are missing */
char* svalA = double_as_string(c, dval1[0]);
char* svalB = double_as_string(c, dval2[0]);
printf("double [%s]: [%s] != [%s]\n", name, svalA, svalB);
grib_context_free(c, svalA);
grib_context_free(c, svalB);
}
}
}
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
grib_context_free(handle1->context, dval1);
grib_context_free(handle2->context, dval2);
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
if (err1)
return err1;
if (err2)
return err2;
break;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
case GRIB_TYPE_BYTES:
2024-03-01 12:10:27 +00:00
return 0; // Not in BUFR
2020-01-22 13:10:59 +00:00
break;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
case GRIB_TYPE_LABEL:
break;
2014-07-06 13:18:40 +00:00
2020-01-22 13:10:59 +00:00
default:
2024-03-01 12:10:27 +00:00
if (verbose) printf("\n");
2020-01-22 13:10:59 +00:00
printInfo(handle1);
save_error(c, name);
printf("Cannot compare [%s], unsupported type %d\n", name, type1);
return GRIB_UNABLE_TO_COMPARE_ACCESSORS;
}
2014-07-06 13:18:40 +00:00
return GRIB_SUCCESS;
2014-07-06 13:18:40 +00:00
}
static int compare_attributes(grib_handle* handle1, grib_handle* handle2, grib_runtime_options* options,
2020-01-22 13:10:59 +00:00
grib_accessor* a, const char* prefix, int* err)
{
2020-01-22 13:10:59 +00:00
int i = 0, ret = 0;
2024-08-11 23:54:29 +00:00
while (i < MAX_ACCESSOR_ATTRIBUTES && a->attributes_[i]) {
/*long native_type = 0;*/
grib_accessor* aa = NULL;
2024-08-11 23:54:29 +00:00
if ((a->attributes_[i]->flags_ & GRIB_ACCESSOR_FLAG_DUMP) == 0) {
++i; /* next attribute if accessor is not for dumping */
continue;
}
2024-08-11 23:54:29 +00:00
if ((a->attributes_[i]->flags_ & GRIB_ACCESSOR_FLAG_READ_ONLY) != 0) {
++i; /* next attribute if accessor is read-only */
continue;
}
2024-08-11 23:54:29 +00:00
aa = a->attributes_[i];
/*native_type = grib_accessor_get_native_type(aa); TODO: read only check? */
2024-08-11 23:54:29 +00:00
isLeafKey = aa->attributes_[0] == NULL ? 1 : 0; /* update global variable */
if (compare_attribute(handle1, handle2, options, aa, prefix, err)) {
2018-11-01 12:55:47 +00:00
(*err)++;
ret = 1;
}
++i; /* next attribute */
}
return ret;
}
2016-11-29 18:12:23 +00:00
static int compare_attribute(grib_handle* handle1, grib_handle* handle2, grib_runtime_options* options,
2020-01-22 13:10:59 +00:00
grib_accessor* a, const char* prefix, int* err)
{
2020-01-22 13:10:59 +00:00
int ret = 0;
grib_context* c = handle1->context;
2024-08-11 23:54:29 +00:00
const size_t fullnameMaxLen = strlen(a->name_) + strlen(prefix) + 5;
char* fullname = (char*)grib_context_malloc_clear(c, sizeof(char) * fullnameMaxLen);
2024-08-11 23:54:29 +00:00
snprintf(fullname, fullnameMaxLen, "%s->%s", prefix, a->name_);
if (compare_values(options, handle1, handle2, fullname, GRIB_TYPE_UNDEFINED)) {
2018-11-01 12:55:47 +00:00
(*err)++;
2020-01-22 13:10:59 +00:00
ret = 1;
}
/* Recurse if this key has children */
if (!isLeafKey) {
if (compare_attributes(handle1, handle2, options, a, fullname, err)) {
2020-01-22 13:10:59 +00:00
ret = 1;
}
}
grib_context_free(c, fullname);
return ret;
}
2020-01-22 13:10:59 +00:00
static int compare_all_dump_keys(grib_handle* handle1, grib_handle* handle2, grib_runtime_options* options, int* pErr)
2016-08-05 17:38:47 +00:00
{
2020-01-22 13:10:59 +00:00
int ret = 0;
const char* name = NULL;
grib_keys_iterator* iter = NULL;
grib_context* context = handle1->context;
2016-09-21 17:08:24 +00:00
if (!headerMode) {
/* See ECC-333: By setting unpack we get ALL the bufr keys. */
/* In headerMode we want just the header ones */
2020-01-22 13:10:59 +00:00
grib_set_long(handle1, "skipExtraKeyAttributes", 1); /* See ECC-745 */
ret = grib_set_long(handle1, "unpack", 1);
2016-09-21 17:08:24 +00:00
if (ret != GRIB_SUCCESS) {
grib_context_log(context, GRIB_LOG_ERROR, "Failed to unpack 1st message: %s", grib_get_error_message(ret));
exit(1);
}
2020-01-22 13:10:59 +00:00
grib_set_long(handle2, "skipExtraKeyAttributes", 1); /* See ECC-745 */
ret = grib_set_long(handle2, "unpack", 1);
2016-09-21 17:08:24 +00:00
if (ret != GRIB_SUCCESS) {
grib_context_log(context, GRIB_LOG_ERROR, "Failed to unpack 2nd message: %s", grib_get_error_message(ret));
exit(1);
}
}
2020-01-22 13:10:59 +00:00
iter = grib_keys_iterator_new(handle1, 0, NULL);
2014-07-06 13:18:40 +00:00
if (!iter) {
2022-12-27 13:19:33 +00:00
grib_context_log(context, GRIB_LOG_ERROR, "unable to create the BUFR keys iterator");
exit(1);
2014-07-06 13:18:40 +00:00
}
release_keys_list(); /* The keys list is used to determine the rank */
new_keys_list();
2020-01-22 13:10:59 +00:00
while (grib_keys_iterator_next(iter)) {
int rank = 0;
int dofree = 0;
char* prefix = NULL;
grib_accessor* xa = grib_keys_iterator_get_accessor(iter);
isLeafKey = 0; /* clear global variable for each key */
2020-01-22 13:10:59 +00:00
name = grib_keys_iterator_get_name(iter);
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;
/* Get full name of key, e.g. '#2#windSpeed' or 'blockNumber' */
2024-08-11 23:54:29 +00:00
rank = compute_bufr_key_rank(handle1, keys_list, xa->name_);
if (rank != 0) {
2024-08-11 23:54:29 +00:00
const size_t prefixMaxLen = strlen(xa->name_) + 10;
prefix = (char*)grib_context_malloc_clear(context, sizeof(char) * prefixMaxLen);
dofree = 1;
2024-08-11 23:54:29 +00:00
snprintf(prefix, prefixMaxLen, "#%d#%s", rank, xa->name_);
2020-01-22 13:10:59 +00:00
}
else {
2024-08-11 23:54:29 +00:00
prefix = (char*)xa->name_;
}
2020-09-24 14:54:33 +00:00
if (blocklisted(prefix)) {
2020-04-17 17:09:52 +00:00
if (dofree) grib_context_free(context, prefix);
2020-01-22 13:10:59 +00:00
continue;
2020-04-17 17:09:52 +00:00
}
/* Compare the key itself */
if (compare_values(options, handle1, handle2, prefix, GRIB_TYPE_UNDEFINED)) {
2018-11-01 12:55:47 +00:00
(*pErr)++;
2020-01-22 13:10:59 +00:00
ret = 1;
}
/* Now compare the key attributes (if any) */
2018-11-01 12:55:47 +00:00
if (compare_attributes(handle1, handle2, options, xa, prefix, pErr)) {
(*pErr)++;
2020-01-22 13:10:59 +00:00
ret = 1;
}
2020-01-22 13:10:59 +00:00
if (dofree)
grib_context_free(context, prefix);
2014-07-06 13:18:40 +00:00
}
grib_keys_iterator_delete(iter);
2018-11-01 11:50:00 +00:00
2014-07-06 13:18:40 +00:00
return ret;
}
2016-04-14 11:11:48 +00:00
static int compare_handles(grib_handle* handle1, grib_handle* handle2, grib_runtime_options* options)
2014-07-06 13:18:40 +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;
2014-07-06 13:18:40 +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(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(handle1, 0, options->compare[i].name);
if (!iter) {
2023-12-23 13:05:12 +00:00
grib_context_log(handle1->context, GRIB_LOG_ERROR, "unable to create the BUFR keys iterator");
exit(1);
}
2020-01-22 13:10:59 +00:00
while (grib_keys_iterator_next(iter)) {
name = grib_keys_iterator_get_name(iter);
2020-09-24 14:54:33 +00:00
if (blocklisted(name))
2020-01-22 13:10:59 +00:00
continue;
if (compare_values(options, handle1, handle2, name, GRIB_TYPE_UNDEFINED)) {
err++;
}
}
grib_keys_iterator_delete(iter);
2020-01-22 13:10:59 +00:00
}
else {
if (compare_values(options, handle1, handle2, options->compare[i].name, options->compare[i].type))
err++;
}
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
}
else {
const void *msg1 = NULL, *msg2 = NULL;
size_t size1 = 0, size2 = 0;
GRIB_CHECK_NOLINE(grib_get_message(handle1, &msg1, &size1), 0);
GRIB_CHECK_NOLINE(grib_get_message(handle2, &msg2, &size2), 0);
2024-03-17 14:31:55 +00:00
if (size1 == size2 && 0 == memcmp(msg1, msg2, size1)) {
return 0;
2020-01-22 13:10:59 +00:00
}
2023-06-21 09:45:16 +00:00
#if defined(BUFR_COMPARE_BYTES)
2015-09-03 14:49:24 +00:00
else {
int lcount=count,ii;
if (options->current_infile) lcount=options->current_infile->filter_handle_count;
if (size1 != size2) {
printf("#%d different size: %d!=%d\n",lcount,(int)size1,(int)size2);
}
if (memcmp_ret) {
unsigned char *m1=(unsigned char*)msg1;
unsigned char *m2=(unsigned char*)msg2;
printf("=== list of different bytes for message %d\n",lcount);
for (ii=0;ii<size1;ii++) {
if (memcmp(m1,m2,1)) {
printf(" %d 0x%.2X != 0x%.2X\n",ii,*m1,*m2);
}
m1++; m2++;
2014-07-06 13:18:40 +00:00
}
}
return err;
2014-07-06 13:18:40 +00:00
}
#endif
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(handle1, 0, options->compare[i].name);
if (!iter) {
2016-08-05 17:38:47 +00:00
grib_context_log(handle1->context, GRIB_LOG_ERROR,
2023-12-23 13:05:12 +00:00
"unable to create the BUFR keys iterator for %s", options->compare[i].name);
exit(1);
}
2020-01-22 13:10:59 +00:00
while (grib_keys_iterator_next(iter)) {
name = grib_keys_iterator_get_name(iter);
2020-09-24 14:54:33 +00:00
if (blocklisted(name))
2020-01-22 13:10:59 +00:00
continue;
if (compare_values(options, handle1, handle2, name, GRIB_TYPE_UNDEFINED)) {
err++;
2020-01-22 13:10:59 +00:00
if (compare_all_dump_keys(handle1, handle2, options, &err)) {
2015-02-04 09:30:29 +00:00
err++;
}
}
}
grib_keys_iterator_delete(iter);
2020-01-22 13:10:59 +00:00
}
else {
if (compare_values(options, handle1, handle2, options->compare[i].name, options->compare[i].type)) {
err++;
2020-01-22 13:10:59 +00:00
if (compare_all_dump_keys(handle1, handle2, options, &err)) {
2015-02-04 09:30:29 +00:00
err++;
}
}
}
2014-07-06 13:18:40 +00:00
}
2020-01-22 13:10:59 +00:00
}
else {
if (compare_all_dump_keys(handle1, handle2, options, &err)) {
2015-02-04 09:30:29 +00:00
err++;
}
2014-07-06 13:18:40 +00:00
}
}
return err;
2014-07-06 13:18:40 +00:00
}
int grib_no_handle_action(grib_runtime_options* options, int err)
2016-04-14 11:11:48 +00:00
{
2020-01-22 13:10:59 +00:00
fprintf(dump_file, "\t\t\"ERROR: unreadable message\"\n");
2016-04-14 11:11:48 +00:00
return 0;
}