2016-11-25 14:46:14 +00:00
|
|
|
/*
|
2019-04-15 13:44:45 +00:00
|
|
|
* Copyright 2005-2019 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"
|
|
|
|
|
|
|
|
/* 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
|
|
|
}
|
|
|
|
|
|
|
|
/* Strip whitespace from the end of a string */
|
|
|
|
void rtrim(char* s)
|
|
|
|
{
|
|
|
|
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';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return the component after final slash */
|
|
|
|
/* "/tmp/x" -> "x" */
|
|
|
|
/* "/tmp/" -> "" */
|
|
|
|
const char* extract_filename(const char* filepath)
|
|
|
|
{
|
2019-12-11 14:16:19 +00:00
|
|
|
/* Note: Windows users could pass in fwd slashes!
|
|
|
|
* so have to check both separators
|
|
|
|
*/
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2019-09-05 12:28:47 +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;
|
|
|
|
size_t numTokens = 0;
|
|
|
|
size_t strLength = 0;
|
|
|
|
size_t index = 0;
|
|
|
|
char delimiterChar = 0;
|
2016-11-25 14:46:14 +00:00
|
|
|
|
|
|
|
DebugAssert(inputString);
|
2020-01-22 13:10:59 +00:00
|
|
|
DebugAssert(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)) {
|
|
|
|
++numTokens; /* there is a trailing token */
|
|
|
|
}
|
|
|
|
++numTokens; /* terminating NULL string to mark the end */
|
|
|
|
|
|
|
|
result = (char**)malloc(numTokens * sizeof(char*));
|
|
|
|
Assert(result);
|
|
|
|
|
|
|
|
/* Start tokenizing */
|
|
|
|
aToken = strtok(inputString, delimiter);
|
|
|
|
while (aToken) {
|
|
|
|
Assert(index < numTokens);
|
|
|
|
*(result + index++) = strdup(aToken);
|
2020-01-22 13:10:59 +00:00
|
|
|
aToken = strtok(NULL, delimiter);
|
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
|
|
|
|
2018-02-22 14:46:21 +00:00
|
|
|
/* Return GRIB_SUCCESS if can convert input to an integer, GRIB_INVALID_ARGUMENT otherwise */
|
2018-02-21 18:20:41 +00:00
|
|
|
int string_to_long(const char* input, long* output)
|
|
|
|
{
|
|
|
|
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)) {
|
2018-02-22 14:46:21 +00:00
|
|
|
/*perror("strtol");*/
|
|
|
|
return GRIB_INVALID_ARGUMENT;
|
2018-02-21 18:20:41 +00:00
|
|
|
}
|
|
|
|
if (endptr == input) {
|
2018-02-22 14:46:21 +00:00
|
|
|
/*fprintf(stderr, "No digits were found. EXIT_FAILURE\n");*/
|
|
|
|
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
|
|
|
|
|
|
|
/* Return 1 if str1 ends with str2, 0 otherwise */
|
|
|
|
int string_ends_with(const char* str1, const char* str2)
|
|
|
|
{
|
|
|
|
const size_t len1 = strlen(str1);
|
|
|
|
const size_t len2 = strlen(str2);
|
|
|
|
if (len2 > len1)
|
|
|
|
return 0;
|
|
|
|
|
2020-01-22 13:10:59 +00:00
|
|
|
if (strcmp(&str1[len1 - len2], str2) == 0)
|
2018-03-06 16:38:58 +00:00
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|