osrm-backend/include/util/string_util.hpp

150 lines
3.3 KiB
C++
Raw Normal View History

2015-01-27 11:44:46 -05:00
#ifndef STRING_UTIL_HPP
#define STRING_UTIL_HPP
#include <cctype>
#include <random>
#include <string>
#include <vector>
2016-01-05 10:51:13 -05:00
namespace osrm
{
namespace util
{
2015-01-27 11:44:46 -05:00
// precision: position after decimal point
// length: maximum number of digits including comma and decimals
// work with negative values to prevent overflowing when taking -value
template <int length, int precision> char *printInt(char *buffer, int value)
2015-01-27 11:44:46 -05:00
{
static_assert(length > 0, "length must be positive");
static_assert(precision > 0, "precision must be positive");
2016-05-27 15:05:04 -04:00
const bool minus = [&value] {
if (value >= 0)
{
value = -value;
return false;
}
return true;
}();
2015-01-27 11:44:46 -05:00
buffer += length - 1;
for (int i = 0; i < precision; ++i)
2015-01-27 11:44:46 -05:00
{
*buffer = '0' - (value % 10);
value /= 10;
--buffer;
2015-01-27 11:44:46 -05:00
}
*buffer = '.';
--buffer;
for (int i = precision + 1; i < length; ++i)
2015-01-27 11:44:46 -05:00
{
*buffer = '0' - (value % 10);
value /= 10;
if (value == 0)
{
2015-01-27 11:44:46 -05:00
break;
}
--buffer;
2015-01-27 11:44:46 -05:00
}
2015-01-27 11:44:46 -05:00
if (minus)
{
--buffer;
2015-01-27 11:44:46 -05:00
*buffer = '-';
}
return buffer;
}
inline bool RequiresJSONStringEscaping(const std::string &string)
{
for (const char letter : string)
{
switch (letter)
{
case '\\':
case '"':
case '/':
case '\b':
case '\f':
case '\n':
case '\r':
case '\t':
return true;
default:
continue;
}
}
return false;
}
inline void EscapeJSONString(const std::string &input, std::string &output)
2015-01-27 11:44:46 -05:00
{
for (const char letter : input)
2015-01-27 11:44:46 -05:00
{
switch (letter)
2015-01-27 11:44:46 -05:00
{
case '\\':
output += "\\\\";
break;
case '"':
output += "\\\"";
break;
case '/':
output += "\\/";
break;
case '\b':
output += "\\b";
break;
case '\f':
output += "\\f";
break;
case '\n':
output += "\\n";
break;
case '\r':
output += "\\r";
break;
case '\t':
output += "\\t";
break;
default:
output.append(1, letter);
2015-01-27 11:44:46 -05:00
break;
}
}
}
inline std::size_t URIDecode(const std::string &input, std::string &output)
{
auto src_iter = std::begin(input);
const auto src_end = std::end(input);
2015-01-27 11:44:46 -05:00
output.resize(input.size() + 1);
std::size_t decoded_length = 0;
for (decoded_length = 0; src_iter != src_end; ++decoded_length)
2015-01-27 11:44:46 -05:00
{
if (src_iter[0] == '%' && src_iter[1] && src_iter[2] && isxdigit(src_iter[1]) &&
isxdigit(src_iter[2]))
{
std::string::value_type a = src_iter[1];
std::string::value_type b = src_iter[2];
a -= src_iter[1] < 58 ? 48 : src_iter[1] < 71 ? 55 : 87;
b -= src_iter[2] < 58 ? 48 : src_iter[2] < 71 ? 55 : 87;
output[decoded_length] = 16 * a + b;
src_iter += 3;
continue;
}
output[decoded_length] = *src_iter++;
}
output.resize(decoded_length);
return decoded_length;
}
inline std::size_t URIDecodeInPlace(std::string &URI) { return URIDecode(URI, URI); }
} // namespace util
} // namespace osrm
2016-01-05 10:51:13 -05:00
2015-01-27 11:44:46 -05:00
#endif // STRING_UTIL_HPP