2016-11-25 14:46:14 +00:00
|
|
|
/*
|
2020-01-28 14:32:34 +00:00
|
|
|
* (C) Copyright 2005- ECMWF.
|
2016-11-25 14:46:14 +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_api_internal.h"
|
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Compare two strings ignoring case.
|
|
|
|
// strcasecmp is not in the C standard. However, it's defined by
|
|
|
|
// 4.4BSD, POSIX.1-2001. So we use our own
|
2020-01-22 13:10:59 +00:00
|
|
|
int strcmp_nocase(const char* s1, const char* s2)
|
2016-11-25 14:46:14 +00:00
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
const unsigned char *us1 = (const unsigned char*)s1,
|
|
|
|
*us2 = (const unsigned char*)s2;
|
2016-12-06 15:58:55 +00:00
|
|
|
|
|
|
|
while (tolower(*us1) == tolower(*us2++)) {
|
2016-11-25 14:46:14 +00:00
|
|
|
if (*us1++ == '\0')
|
|
|
|
return (0);
|
2016-12-06 15:58:55 +00:00
|
|
|
}
|
|
|
|
return (tolower(*us1) - tolower(*--us2));
|
2016-11-25 14:46:14 +00:00
|
|
|
}
|
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Strip whitespace from the end of a string
|
2022-06-10 19:51:39 +00:00
|
|
|
void string_rtrim(char* s)
|
2016-11-25 14:46:14 +00:00
|
|
|
{
|
|
|
|
size_t len = 0;
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!s)
|
|
|
|
return;
|
2016-11-25 14:46:14 +00:00
|
|
|
len = strlen(s);
|
|
|
|
while (len > 0 && isspace((unsigned char)s[len - 1]))
|
|
|
|
len--;
|
|
|
|
s[len] = '\0';
|
|
|
|
}
|
|
|
|
|
2022-06-10 19:51:39 +00:00
|
|
|
void string_lrtrim(char** x, int do_left, int do_right)
|
2020-06-26 21:18:08 +00:00
|
|
|
{
|
2023-06-26 20:21:07 +00:00
|
|
|
DEBUG_ASSERT(do_left || do_right);
|
2020-06-26 21:18:08 +00:00
|
|
|
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';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Return the component after final slash
|
|
|
|
// "/tmp/x" -> "x"
|
|
|
|
// "/tmp/" -> ""
|
2016-11-25 14:46:14 +00:00
|
|
|
const char* extract_filename(const char* filepath)
|
|
|
|
{
|
2023-08-05 17:43:26 +00:00
|
|
|
// Note: Windows users could pass in fwd slashes!
|
|
|
|
// so have to check both separators
|
2019-12-11 14:16:19 +00:00
|
|
|
const char* s = strrchr(filepath, '/');
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!s)
|
|
|
|
s = strrchr(filepath, '\\');
|
|
|
|
if (!s)
|
|
|
|
return filepath;
|
|
|
|
else
|
|
|
|
return s + 1;
|
2016-11-25 14:46:14 +00:00
|
|
|
}
|
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Returns an array of strings the last of which is NULL.
|
|
|
|
// Note: The delimiter here is a 'string' but must be ONE character!
|
|
|
|
// Splitting with several delimiters is not supported.
|
2016-11-25 14:46:14 +00:00
|
|
|
char** string_split(char* inputString, const char* delimiter)
|
|
|
|
{
|
2020-01-22 13:10:59 +00:00
|
|
|
char** result = NULL;
|
|
|
|
char* p = inputString;
|
2016-11-25 14:46:14 +00:00
|
|
|
char* lastDelimiter = NULL;
|
2020-01-22 13:10:59 +00:00
|
|
|
char* aToken = NULL;
|
2022-10-25 20:37:45 +00:00
|
|
|
char* lasts = NULL;
|
2020-01-22 13:10:59 +00:00
|
|
|
size_t numTokens = 0;
|
|
|
|
size_t strLength = 0;
|
|
|
|
size_t index = 0;
|
|
|
|
char delimiterChar = 0;
|
2016-11-25 14:46:14 +00:00
|
|
|
|
2023-06-26 20:21:07 +00:00
|
|
|
DEBUG_ASSERT(inputString);
|
|
|
|
DEBUG_ASSERT(delimiter && (strlen(delimiter) == 1));
|
2016-11-25 14:46:14 +00:00
|
|
|
delimiterChar = delimiter[0];
|
|
|
|
while (*p) {
|
|
|
|
const char ctmp = *p;
|
|
|
|
if (ctmp == delimiterChar) {
|
|
|
|
++numTokens;
|
|
|
|
lastDelimiter = p;
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
strLength = strlen(inputString);
|
|
|
|
if (lastDelimiter < (inputString + strLength - 1)) {
|
2023-08-05 17:43:26 +00:00
|
|
|
++numTokens; // there is a trailing token
|
2016-11-25 14:46:14 +00:00
|
|
|
}
|
2023-08-05 17:43:26 +00:00
|
|
|
++numTokens; // terminating NULL string to mark the end
|
2016-11-25 14:46:14 +00:00
|
|
|
|
|
|
|
result = (char**)malloc(numTokens * sizeof(char*));
|
|
|
|
Assert(result);
|
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Start tokenizing
|
2022-10-25 20:37:45 +00:00
|
|
|
aToken = strtok_r(inputString, delimiter, &lasts);
|
2016-11-25 14:46:14 +00:00
|
|
|
while (aToken) {
|
|
|
|
Assert(index < numTokens);
|
|
|
|
*(result + index++) = strdup(aToken);
|
2022-10-25 20:37:45 +00:00
|
|
|
aToken = strtok_r(NULL, delimiter, &lasts);
|
2016-11-25 14:46:14 +00:00
|
|
|
}
|
|
|
|
Assert(index == numTokens - 1);
|
2016-12-21 17:20:12 +00:00
|
|
|
*(result + index) = NULL;
|
2016-11-25 14:46:14 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2018-02-21 18:20:41 +00:00
|
|
|
|
2023-08-04 18:58:14 +00:00
|
|
|
// Return GRIB_SUCCESS if we can convert 'input' to an integer, GRIB_INVALID_ARGUMENT otherwise.
|
|
|
|
// If 'strict' is 1 then disallow characters at the end which are not valid digits.
|
|
|
|
// E.g., in strict mode, "4i" will be rejected. Otherwise it will convert it to 4
|
|
|
|
int string_to_long(const char* input, long* output, int strict)
|
2018-02-21 18:20:41 +00:00
|
|
|
{
|
|
|
|
const int base = 10;
|
2020-01-22 13:10:59 +00:00
|
|
|
char* endptr;
|
2018-02-21 18:20:41 +00:00
|
|
|
long val = 0;
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (!input)
|
|
|
|
return GRIB_INVALID_ARGUMENT;
|
2018-02-21 18:20:41 +00:00
|
|
|
|
2018-02-22 14:46:21 +00:00
|
|
|
errno = 0;
|
2020-01-22 13:10:59 +00:00
|
|
|
val = strtol(input, &endptr, base);
|
|
|
|
if ((errno == ERANGE && (val == LONG_MAX || val == LONG_MIN)) ||
|
|
|
|
(errno != 0 && val == 0)) {
|
2023-08-04 18:58:14 +00:00
|
|
|
// perror("strtol");
|
2018-02-22 14:46:21 +00:00
|
|
|
return GRIB_INVALID_ARGUMENT;
|
2018-02-21 18:20:41 +00:00
|
|
|
}
|
|
|
|
if (endptr == input) {
|
2023-08-04 18:58:14 +00:00
|
|
|
// fprintf(stderr, "No digits were found\n");
|
|
|
|
return GRIB_INVALID_ARGUMENT;
|
|
|
|
}
|
|
|
|
if (strict && *endptr != 0) {
|
|
|
|
// fprintf(stderr, "Left over characters at the end; not a pure number\n");
|
2018-02-22 14:46:21 +00:00
|
|
|
return GRIB_INVALID_ARGUMENT;
|
2018-02-21 18:20:41 +00:00
|
|
|
}
|
|
|
|
*output = val;
|
2018-02-22 14:46:21 +00:00
|
|
|
return GRIB_SUCCESS;
|
2018-02-21 18:20:41 +00:00
|
|
|
}
|
2018-03-06 16:38:58 +00:00
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Return 1 if 's' ends with 'suffix', 0 otherwise
|
2022-03-14 15:15:05 +00:00
|
|
|
int string_ends_with(const char* s, const char* suffix)
|
2018-03-06 16:38:58 +00:00
|
|
|
{
|
2022-03-14 15:15:05 +00:00
|
|
|
const size_t len1 = strlen(s);
|
|
|
|
const size_t len2 = strlen(suffix);
|
2018-03-06 16:38:58 +00:00
|
|
|
if (len2 > len1)
|
|
|
|
return 0;
|
|
|
|
|
2022-03-14 15:15:05 +00:00
|
|
|
if (strcmp(&s[len1 - len2], suffix) == 0)
|
2018-03-06 16:38:58 +00:00
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
2021-02-04 11:52:57 +00:00
|
|
|
|
2022-06-10 19:51:39 +00:00
|
|
|
int string_count_char(const char* str, char c)
|
2021-02-04 11:52:57 +00:00
|
|
|
{
|
|
|
|
int i = 0, count = 0;
|
2023-06-26 20:21:07 +00:00
|
|
|
DEBUG_ASSERT(str);
|
2023-04-12 21:09:51 +00:00
|
|
|
for (i=0; str[i]; i++) {
|
2021-02-04 11:52:57 +00:00
|
|
|
if (str[i] == c) count++;
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
2022-03-15 16:52:54 +00:00
|
|
|
|
|
|
|
const char* codes_get_product_name(ProductKind product)
|
|
|
|
{
|
|
|
|
switch (product) {
|
|
|
|
case PRODUCT_GRIB:
|
|
|
|
return "GRIB";
|
|
|
|
case PRODUCT_BUFR:
|
|
|
|
return "BUFR";
|
|
|
|
case PRODUCT_METAR:
|
|
|
|
return "METAR";
|
|
|
|
case PRODUCT_GTS:
|
|
|
|
return "GTS";
|
|
|
|
case PRODUCT_TAF:
|
|
|
|
return "TAF";
|
|
|
|
case PRODUCT_ANY:
|
|
|
|
return "ANY";
|
|
|
|
}
|
|
|
|
return "unknown";
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* grib_get_type_name(int type)
|
|
|
|
{
|
|
|
|
switch (type) {
|
|
|
|
case GRIB_TYPE_LONG:
|
|
|
|
return "long";
|
|
|
|
case GRIB_TYPE_STRING:
|
|
|
|
return "string";
|
|
|
|
case GRIB_TYPE_BYTES:
|
|
|
|
return "bytes";
|
|
|
|
case GRIB_TYPE_DOUBLE:
|
|
|
|
return "double";
|
|
|
|
case GRIB_TYPE_LABEL:
|
|
|
|
return "label";
|
|
|
|
case GRIB_TYPE_SECTION:
|
|
|
|
return "section";
|
|
|
|
}
|
|
|
|
return "unknown";
|
|
|
|
}
|
2022-06-09 16:47:20 +00:00
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Replace all occurrences of character in string.
|
|
|
|
// Returns pointer to the NUL byte at the end of 's'
|
2023-04-12 21:09:51 +00:00
|
|
|
char* string_replace_char(char *s, char oldc, char newc)
|
2022-06-09 16:47:20 +00:00
|
|
|
{
|
|
|
|
for (; *s; ++s)
|
2022-06-09 17:10:07 +00:00
|
|
|
if (*s == oldc)
|
|
|
|
*s = newc;
|
2022-06-09 16:47:20 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2023-08-05 17:43:26 +00:00
|
|
|
// Remove all instances of character 'c' from 'str'
|
2023-04-12 21:09:51 +00:00
|
|
|
void string_remove_char(char* str, char c)
|
2022-06-09 16:47:20 +00:00
|
|
|
{
|
|
|
|
size_t i, j;
|
2023-06-26 20:21:07 +00:00
|
|
|
DEBUG_ASSERT(str);
|
2022-06-09 16:47:20 +00:00
|
|
|
size_t len = strlen(str);
|
|
|
|
for(i=0; i<len; i++) {
|
|
|
|
if(str[i] == c) {
|
|
|
|
for(j=i; j<len; j++) {
|
|
|
|
str[j] = str[j+1];
|
|
|
|
}
|
|
|
|
len--;
|
|
|
|
i--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|