mirror of https://github.com/ecmwf/eccodes.git
ECC-1126: BUFR: ECMWF Local use section key 'ident' should be trimmed (Part 2)
This commit is contained in:
parent
8e8a5eb2b8
commit
ff95bd30bc
|
@ -204,6 +204,8 @@ static int bufr_decode_rdb_keys(const void* message, long offset_section2, codes
|
||||||
return GRIB_SUCCESS;
|
return GRIB_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define IDENT_LEN 9 /* 8 chars plus the final 0 terminator */
|
||||||
|
|
||||||
/* The ECMWF BUFR local use section */
|
/* The ECMWF BUFR local use section */
|
||||||
static int bufr_decode_extra_rdb_keys(const void* message, long offset_section2, codes_bufr_header* hdr)
|
static int bufr_decode_extra_rdb_keys(const void* message, long offset_section2, codes_bufr_header* hdr)
|
||||||
{
|
{
|
||||||
|
@ -261,8 +263,11 @@ static int bufr_decode_extra_rdb_keys(const void* message, long offset_section2,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
size_t i = 0, j = 0;
|
size_t i = 0;
|
||||||
long lValue = 0;
|
long lValue = 0;
|
||||||
|
char* pTemp = NULL;
|
||||||
|
char temp[IDENT_LEN] = {0,};
|
||||||
|
|
||||||
start = 72;
|
start = 72;
|
||||||
lValue = (long)grib_decode_unsigned_long(pKeyData, &start, 25);
|
lValue = (long)grib_decode_unsigned_long(pKeyData, &start, 25);
|
||||||
hdr->localLatitude = (lValue - 9000000.0) / 100000.0;
|
hdr->localLatitude = (lValue - 9000000.0) / 100000.0;
|
||||||
|
@ -270,17 +275,14 @@ static int bufr_decode_extra_rdb_keys(const void* message, long offset_section2,
|
||||||
lValue = (long)grib_decode_unsigned_long(pKeyData, &start, 26);
|
lValue = (long)grib_decode_unsigned_long(pKeyData, &start, 26);
|
||||||
hdr->localLongitude = (lValue - 18000000.0) / 100000.0;
|
hdr->localLongitude = (lValue - 18000000.0) / 100000.0;
|
||||||
|
|
||||||
/* interpret keyMore as a string */
|
/* interpret keyMore as a string. Copy to a temporary */
|
||||||
for (i = 0; i < 8; ++i) {
|
for (i = 0; i < 8; ++i) {
|
||||||
const char c = *pKeyMore;
|
temp[i] = *pKeyMore++;
|
||||||
//printf("Lookin at %c, i=%lu, j=%lu\n", c, i, j);
|
|
||||||
if (c != ' ') {
|
|
||||||
//printf(" not space so copy to %lu\n", j);
|
|
||||||
hdr->ident[j++] = c;
|
|
||||||
}
|
}
|
||||||
pKeyMore++;
|
temp[i] = '\0';
|
||||||
}
|
pTemp = temp;
|
||||||
hdr->ident[j] = '\0';
|
lrtrim(&pTemp, 1, 1); /* Trim left and right */
|
||||||
|
strncpy(hdr->ident, pTemp, 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
return GRIB_SUCCESS;
|
return GRIB_SUCCESS;
|
||||||
|
|
|
@ -18,8 +18,8 @@
|
||||||
IMPLEMENTS = unpack_string;pack_string
|
IMPLEMENTS = unpack_string;pack_string
|
||||||
IMPLEMENTS = init
|
IMPLEMENTS = init
|
||||||
MEMBERS= const char* input
|
MEMBERS= const char* input
|
||||||
MEMBERS= int trimleft
|
MEMBERS= int trim_left
|
||||||
MEMBERS= int trimright
|
MEMBERS= int trim_right
|
||||||
END_CLASS_DEF
|
END_CLASS_DEF
|
||||||
|
|
||||||
*/
|
*/
|
||||||
|
@ -46,8 +46,8 @@ typedef struct grib_accessor_trim
|
||||||
/* Members defined in ascii */
|
/* Members defined in ascii */
|
||||||
/* Members defined in trim */
|
/* Members defined in trim */
|
||||||
const char* input;
|
const char* input;
|
||||||
int trimleft;
|
int trim_left;
|
||||||
int trimright;
|
int trim_right;
|
||||||
} grib_accessor_trim;
|
} grib_accessor_trim;
|
||||||
|
|
||||||
extern grib_accessor_class* grib_accessor_class_ascii;
|
extern grib_accessor_class* grib_accessor_class_ascii;
|
||||||
|
@ -142,26 +142,10 @@ static void init(grib_accessor* a, const long l, grib_arguments* arg)
|
||||||
grib_handle* h = grib_handle_of_accessor(a);
|
grib_handle* h = grib_handle_of_accessor(a);
|
||||||
|
|
||||||
self->input = grib_arguments_get_name(h, arg, n++);
|
self->input = grib_arguments_get_name(h, arg, n++);
|
||||||
self->trimleft = grib_arguments_get_long(h, arg, n++);
|
self->trim_left = grib_arguments_get_long(h, arg, n++);
|
||||||
self->trimright= grib_arguments_get_long(h, arg, n++);
|
self->trim_right= grib_arguments_get_long(h, arg, n++);
|
||||||
DebugAssert(self->trimleft == 0 || self->trimleft == 1);
|
DebugAssert(self->trim_left == 0 || self->trim_left == 1);
|
||||||
DebugAssert(self->trimright == 0 || self->trimright == 1);
|
DebugAssert(self->trim_right == 0 || self->trim_right == 1);
|
||||||
}
|
|
||||||
|
|
||||||
static void trim(char** x)
|
|
||||||
{
|
|
||||||
char* p = 0;
|
|
||||||
while (**x == ' ' && **x != '\0')
|
|
||||||
(*x)++;
|
|
||||||
if (**x == '\0')
|
|
||||||
return;
|
|
||||||
p = (*x) + strlen(*x) - 1;
|
|
||||||
while (*p == ' ') {
|
|
||||||
*p = '\0';
|
|
||||||
p--;
|
|
||||||
}
|
|
||||||
if (*p == ' ')
|
|
||||||
*p = '\0';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int unpack_string(grib_accessor* a, char* val, size_t* len)
|
static int unpack_string(grib_accessor* a, char* val, size_t* len)
|
||||||
|
@ -176,13 +160,12 @@ static int unpack_string(grib_accessor* a, char* val, size_t* len)
|
||||||
|
|
||||||
err = grib_get_string(h, self->input, input, &size);
|
err = grib_get_string(h, self->input, input, &size);
|
||||||
if (err) return err;
|
if (err) return err;
|
||||||
DebugAssert(size < 9);
|
|
||||||
|
|
||||||
trim(&pInput);
|
lrtrim(&pInput, self->trim_left, self->trim_right);
|
||||||
sprintf(val, "%s", pInput);
|
sprintf(val, "%s", pInput);
|
||||||
size = strlen(val);
|
size = strlen(val);
|
||||||
*len = size + 1;
|
*len = size + 1;
|
||||||
return err;
|
return GRIB_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int pack_string(grib_accessor* a, const char* val, size_t* len)
|
static int pack_string(grib_accessor* a, const char* val, size_t* len)
|
||||||
|
@ -206,7 +189,7 @@ static int pack_string(grib_accessor* a, const char* val, size_t* len)
|
||||||
|
|
||||||
sprintf(buf, "%s", val);
|
sprintf(buf, "%s", val);
|
||||||
pBuf = buf;
|
pBuf = buf;
|
||||||
trim(&pBuf);
|
lrtrim(&pBuf, self->trim_left, self->trim_right);
|
||||||
|
|
||||||
return grib_pack_string(inputAccesstor, pBuf, len);
|
return grib_pack_string(inputAccesstor, pBuf, len);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1494,6 +1494,7 @@ int codes_bufr_header_get_string(codes_bufr_header* bh, const char* key, char* v
|
||||||
/* string_util.c */
|
/* string_util.c */
|
||||||
int strcmp_nocase(const char* s1, const char* s2);
|
int strcmp_nocase(const char* s1, const char* s2);
|
||||||
void rtrim(char* s);
|
void rtrim(char* s);
|
||||||
|
void lrtrim(char** x, int do_left, int do_right);
|
||||||
const char* extract_filename(const char* filepath);
|
const char* extract_filename(const char* filepath);
|
||||||
char** string_split(char* inputString, const char* delimiter);
|
char** string_split(char* inputString, const char* delimiter);
|
||||||
int string_to_long(const char* input, long* output);
|
int string_to_long(const char* input, long* output);
|
||||||
|
|
|
@ -460,7 +460,6 @@ static void dump_string(grib_dumper* d, grib_accessor* a, const char* comment)
|
||||||
/* ECC-356: Solution for the special local section key 'keyMore' and its alias 'ident' */
|
/* ECC-356: Solution for the special local section key 'keyMore' and its alias 'ident' */
|
||||||
int skip = 1;
|
int skip = 1;
|
||||||
if ((a->flags & GRIB_ACCESSOR_FLAG_HIDDEN) != 0) {
|
if ((a->flags & GRIB_ACCESSOR_FLAG_HIDDEN) != 0) {
|
||||||
//grib_handle* h = grib_handle_of_accessor(a);
|
|
||||||
if (strcmp(a->name, "identTrimmed") == 0) {
|
if (strcmp(a->name, "identTrimmed") == 0) {
|
||||||
skip = 0;
|
skip = 0;
|
||||||
acc_name = "ident";
|
acc_name = "ident";
|
||||||
|
|
|
@ -38,6 +38,26 @@ void rtrim(char* s)
|
||||||
s[len] = '\0';
|
s[len] = '\0';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void lrtrim(char** x, int do_left, int do_right)
|
||||||
|
{
|
||||||
|
DebugAssert(do_left || do_right);
|
||||||
|
if (do_left) {
|
||||||
|
while (isspace(**x) && **x != '\0')
|
||||||
|
(*x)++;
|
||||||
|
}
|
||||||
|
if (**x == '\0')
|
||||||
|
return;
|
||||||
|
if (do_right) {
|
||||||
|
char* p = (*x) + strlen(*x) - 1;
|
||||||
|
while (isspace(*p)) {
|
||||||
|
*p = '\0';
|
||||||
|
p--;
|
||||||
|
}
|
||||||
|
if (isspace(*p))
|
||||||
|
*p = '\0';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Return the component after final slash */
|
/* Return the component after final slash */
|
||||||
/* "/tmp/x" -> "x" */
|
/* "/tmp/x" -> "x" */
|
||||||
/* "/tmp/" -> "" */
|
/* "/tmp/" -> "" */
|
||||||
|
|
|
@ -1485,9 +1485,41 @@ static void test_concept_condition_strings()
|
||||||
grib_handle_delete(h);
|
grib_handle_delete(h);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void test_trimming()
|
||||||
|
{
|
||||||
|
char a[] = " Standing ";
|
||||||
|
char b[] = " Weeping ";
|
||||||
|
char c[] = " Silhouette ";
|
||||||
|
char d[] = " The Forest Of October ";
|
||||||
|
char e[] = "\t\n Apostle In Triumph \r ";
|
||||||
|
char* pA = a;
|
||||||
|
char* pB = b;
|
||||||
|
char* pC = c;
|
||||||
|
char* pD = d;
|
||||||
|
char* pE = e;
|
||||||
|
|
||||||
|
lrtrim(&pA, 0, 1); /*right only*/
|
||||||
|
assert( strcmp(pA, " Standing")==0 );
|
||||||
|
|
||||||
|
lrtrim(&pB, 1, 0); /*left only*/
|
||||||
|
assert( strcmp(pB, "Weeping ")==0 );
|
||||||
|
|
||||||
|
lrtrim(&pC, 1, 1); /*both ends*/
|
||||||
|
assert( strcmp(pC, "Silhouette")==0 );
|
||||||
|
|
||||||
|
lrtrim(&pD, 1, 1); /*make sure other spaces are not removed*/
|
||||||
|
assert( strcmp(pD, "The Forest Of October")==0 );
|
||||||
|
|
||||||
|
lrtrim(&pE, 1, 1); /* Other chars */
|
||||||
|
assert( strcmp(pE, "Apostle In Triumph")==0 );
|
||||||
|
}
|
||||||
|
|
||||||
int main(int argc, char** argv)
|
int main(int argc, char** argv)
|
||||||
{
|
{
|
||||||
/*printf("Doing unit tests. ecCodes version = %ld\n", grib_get_api_version());*/
|
/*printf("Doing unit tests. ecCodes version = %ld\n", grib_get_api_version());*/
|
||||||
|
|
||||||
|
test_trimming();
|
||||||
|
|
||||||
test_get_git_sha1();
|
test_get_git_sha1();
|
||||||
|
|
||||||
test_concept_condition_strings();
|
test_concept_condition_strings();
|
||||||
|
|
Loading…
Reference in New Issue