42#define KM_PER_MILE 1.609344
56 const std::string::size_type endpos = str.find_last_not_of(
" \t\n\r");
57 if (std::string::npos != endpos) {
58 const int startpos = (int)str.find_first_not_of(
" \t\n\r");
59 return str.substr(startpos, endpos - startpos + 1);
67 const std::string::size_type endpos = str.find_last_not_of(
"0");
68 if (endpos != std::string::npos && str.back() ==
'0') {
69 std::string res = str.substr(0,
MAX2((
int)str.size() - max, (
int)endpos + 1));
78 std::transform(s.begin(), s.end(), s.begin(), [](
char c) {
79 return (char)::tolower(c);
88 std::transform(s.begin(), s.end(), s.begin(), [](
char c) {
89 return (char)::toupper(c);
99 for (
const auto& c : str) {
100 const unsigned char uc = (
unsigned char)c;
104 result += (char)(0xc2 + (uc > 0xbf));
105 result += (char)((uc & 0x3f) + 0x80);
114 str =
replace(str,
"\xE4",
"ae");
115 str =
replace(str,
"\xC4",
"Ae");
116 str =
replace(str,
"\xF6",
"oe");
117 str =
replace(str,
"\xD6",
"Oe");
118 str =
replace(str,
"\xFC",
"ue");
119 str =
replace(str,
"\xDC",
"Ue");
120 str =
replace(str,
"\xDF",
"ss");
121 str =
replace(str,
"\xC9",
"E");
122 str =
replace(str,
"\xE9",
"e");
123 str =
replace(str,
"\xC8",
"E");
124 str =
replace(str,
"\xE8",
"e");
131 std::string::size_type idx = str.find(what);
132 const int what_len = (int)what.length();
134 const int by_len = (int)by.length();
135 while (idx != std::string::npos) {
136 str = str.replace(idx, what_len, by);
137 idx = str.find(what, idx + by_len);
147 if (timeRef !=
nullptr) {
148 const std::string::size_type localTimeIndex = str.find(
"${LOCALTIME}");
149 const std::string::size_type utcIndex = str.find(
"${UTC}");
150 const bool isUTC = utcIndex != std::string::npos;
151 if (localTimeIndex != std::string::npos || isUTC) {
152 const time_t rawtime = std::chrono::system_clock::to_time_t(*timeRef);
154 struct tm* timeinfo = isUTC ? gmtime(&rawtime) : localtime(&rawtime);
155 strftime(buffer, 80,
"%Y-%m-%d-%H-%M-%S.", timeinfo);
156 auto seconds = std::chrono::time_point_cast<std::chrono::seconds>(*timeRef);
157 auto microseconds = std::chrono::duration_cast<std::chrono::microseconds>(*timeRef - seconds);
158 const std::string micro = buffer +
toString(microseconds.count());
160 s.replace(utcIndex, 6, micro);
162 s.replace(localTimeIndex, 12, micro);
166 const std::string::size_type pidIndex = str.find(
"${PID}");
167 if (pidIndex != std::string::npos) {
169 s.replace(pidIndex, 6,
toString(::GetCurrentProcessId()));
171 s.replace(pidIndex, 6,
toString(::getpid()));
174 if (std::getenv(
"SUMO_LOGO") ==
nullptr) {
175 s =
replace(s,
"${SUMO_LOGO}",
"${SUMO_HOME}/data/logo/sumo-128x138.png");
177 const std::string::size_type tildeIndex = str.find(
"~");
178 if (tildeIndex == 0) {
179 s.replace(0, 1,
"${HOME}");
181 s =
replace(s,
",~",
",${HOME}");
183 if (std::getenv(
"HOME") ==
nullptr) {
184 s =
replace(s,
"${HOME}",
"${USERPROFILE}");
192 std::regex envVarExpr(R
"(\$\{(.+?)\})");
196 std::string strIter = s;
199 while (std::regex_search(strIter, match, envVarExpr)) {
200 std::string varName = match[1];
203 std::string varValue;
204 if (std::getenv(varName.c_str()) !=
nullptr) {
205 varValue = std::getenv(varName.c_str());
209 s = std::regex_replace(s, std::regex(
"\\$\\{" + varName +
"\\}"), varValue);
212 strIter = match.suffix();
220 const std::chrono::system_clock::time_point now = timeRef ==
nullptr ? std::chrono::system_clock::now() : *timeRef;
221 const auto now_seconds = std::chrono::time_point_cast<std::chrono::seconds>(now);
222 const std::time_t now_c = std::chrono::system_clock::to_time_t(now);
223 const auto microseconds = std::chrono::duration_cast<std::chrono::microseconds>(now - now_seconds).count();
224 std::tm local_tm = *std::localtime(&now_c);
227 std::time_t utc_time = std::time(
nullptr);
228 std::tm utc_tm = *std::gmtime(&utc_time);
229 const double offset = std::difftime(std::mktime(&local_tm), std::mktime(&utc_tm)) / 3600.0;
230 const int hours_offset =
static_cast<int>(offset);
231 const int minutes_offset =
static_cast<int>((offset - hours_offset) * 60);
234 std::ostringstream oss;
236 std::strftime(buf,
sizeof(buf),
"%Y-%m-%dT%H:%M:%S", &local_tm);
238 << std::setw(6) << std::setfill(
'0') << std::abs(microseconds)
239 << (hours_offset >= 0 ?
"+" :
"-")
240 << std::setw(2) << std::setfill(
'0') << std::abs(hours_offset) <<
":"
241 << std::setw(2) << std::setfill(
'0') << std::abs(minutes_offset);
248 return str.compare(0, prefix.length(), prefix) == 0;
254 if (str.length() >= suffix.length()) {
255 return str.compare(str.length() - suffix.length(), suffix.length(), suffix) == 0;
264 return std::string(
MAX2(0, length - (
int)str.size()), padding) + str;
270 std::string result =
replace(orig,
"&",
"&");
271 result =
replace(result,
">",
">");
272 result =
replace(result,
"<",
"<");
273 result =
replace(result,
"\"",
""");
274 if (maskDoubleHyphen) {
275 result =
replace(result,
"--",
"--");
277 for (
char invalid =
'\1'; invalid <
' '; invalid++) {
278 result =
replace(result, std::string(1, invalid).c_str(),
"");
280 return replace(result,
"'",
"'");
286 std::string result =
replace(orig,
"\"",
"\\\"");
293 std::ostringstream out;
295 for (
int i = 0; i < (int)toEncode.length(); ++i) {
296 const char t = toEncode.at(i);
298 if ((encodeWhich !=
"" && encodeWhich.find(t) == std::string::npos) ||
299 (encodeWhich ==
"" &&
300 ((t >= 45 && t <= 57) ||
301 (t >= 65 && t <= 90) ||
303 (t >= 97 && t <= 122) ||
306 out << toEncode.at(i);
318 std::ostringstream out;
320 for (
int i = 0; i < (int)toDecode.length(); ++i) {
321 if (toDecode.at(i) ==
'%') {
322 std::string str(toDecode.substr(i + 1, 2));
326 out << toDecode.at(i);
339 s <<
"%" << std::setw(2) << std::setfill(
'0') << std::hex << i;
349 std::istringstream in(str);
355 return static_cast<unsigned char>(c);
361 long long int result =
toLong(sData);
362 if (result > std::numeric_limits<int>::max() || result < std::numeric_limits<int>::min()) {
373 const long long int result =
toLong(sData);
375 return ((result <= std::numeric_limits<int>::max()) && (result >= std::numeric_limits<int>::min()));
383 if (sData.length() == 0) {
392 const char*
const data = sData.c_str();
393 if (data == 0 || data[0] == 0) {
399 long long int ret = _strtoi64(data, &end, 10);
401 long long int ret = strtoll(data, &end, 10);
403 if (errno == ERANGE) {
407 if ((
int)(end - data) != (
int)strlen(data)) {
416 const char*
const data = sData.c_str();
417 if (data == 0 || data[0] == 0) {
425 _strtoi64(data, &end, 10);
427 strtoll(data, &end, 10);
430 if (errno == ERANGE) {
434 if ((
int)(end - data) != (
int)strlen(data)) {
443 if (sData.length() == 0) {
449 if (sData[0] ==
'#') {
450 result = std::stoi(sData.substr(1), &idx, 16);
453 result = std::stoi(sData, &idx, 16);
458 if (idx != sData.length()) {
467 if (sData.length() == 0) {
471 if (sData[0] ==
'#') {
472 sData = sData.substr(1);
474 const char* sDataPtr = sData.c_str();
479 strtol(sDataPtr, &returnPtr, 16);
481 if (errno == ERANGE) {
485 if (sDataPtr == returnPtr) {
489 if (
static_cast<size_t>(returnPtr - sDataPtr) != sData.size()) {
498 if (sData.size() == 0) {
503 const double result = std::stod(sData, &idx);
504 if (idx != sData.size()) {
518 if (sData.size() == 0) {
521 const char* sDataPtr = sData.c_str();
526 strtod(sDataPtr, &returnPtr);
528 if (errno == ERANGE) {
532 if (sDataPtr == returnPtr) {
536 if (
static_cast<size_t>(returnPtr - sDataPtr) != sData.size()) {
545 if (sData.length() == 0) {
554 if (sData.length() == 0) {
558 if (s ==
"1" || s ==
"yes" || s ==
"true" || s ==
"on" || s ==
"x" || s ==
"t") {
561 if (s ==
"0" || s ==
"no" || s ==
"false" || s ==
"off" || s ==
"-" || s ==
"f") {
570 if (sData.length() == 0) {
575 if (s ==
"1" || s ==
"yes" || s ==
"true" || s ==
"on" || s ==
"x" || s ==
"t") {
579 if (s ==
"0" || s ==
"no" || s ==
"false" || s ==
"off" || s ==
"-" || s ==
"f") {
596 if (sData.size() == 0) {
601 const double result = std::stod(sData, &idx);
602 if (idx != sData.size()) {
603 const std::string unit =
prune(sData.substr(idx));
604 if (unit ==
"m" || unit ==
"metre" || unit ==
"meter" || unit ==
"metres" || unit ==
"meters") {
607 if (unit ==
"km" || unit ==
"kilometre" || unit ==
"kilometer" || unit ==
"kilometres" || unit ==
"kilometers") {
608 return result * 1000.;
610 if (unit ==
"mi" || unit ==
"mile" || unit ==
"miles") {
614 return result * 1852.;
616 if (unit ==
"ft" || unit ==
"foot" || unit ==
"feet") {
617 return result * 12. * 0.0254;
619 if (unit ==
"\"" || unit ==
"in" || unit ==
"inch" || unit ==
"inches") {
620 return result * 0.0254;
622 if (unit[0] ==
'\'') {
623 double inches = 12 * result;
624 if (unit.length() > 1) {
625 inches += std::stod(unit.substr(1), &idx);
626 if (unit.substr(idx) ==
"\"") {
627 return inches * 0.0254;
644 if (sData.size() == 0) {
649 const double result = std::stod(sData, &idx);
650 if (idx != sData.size()) {
651 const std::string unit =
prune(sData.substr(idx));
652 if (unit ==
"km/h" || unit ==
"kph" || unit ==
"kmh" || unit ==
"kmph") {
661 if (unit ==
"knots") {
662 return result * 1.852 / 3.6;
666 return defaultKmph ? result / 3.6 : result;
678 std::string result = s;
679 result.erase(0, s.find_first_not_of(t));
685 std::string result = s;
686 result.erase(s.find_last_not_of(t) + 1);
701 bool firstLine =
true;
702 bool firstWord =
true;
703 for (std::string p : parts) {
704 if ((
int)(line.size() + p.size()) < width || firstWord) {
722 if (line.size() > 0) {
737 auto valueStr =
toString(value, precision);
739 while (valueStr.size() > 1) {
740 if (valueStr.back() ==
'0') {
742 }
else if (valueStr.back() ==
'.') {
std::pair< int, double > MMVersion
(M)ajor/(M)inor version for written networks and default version for loading
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
std::vector< std::string > getVector()
return vector of strings
static std::string pruneZeros(const std::string &str, int max)
Removes trailing zeros (at most 'max')
static std::string urlEncode(const std::string &url, const std::string encodeWhich="")
encode url (stem from http://bogomip.net/blog/cpp-url-encoding-and-decoding/)
static bool isDouble(const std::string &sData)
check if the given sData can be conveted to double
static MMVersion toVersion(const std::string &sData)
parse a (network) version string
static bool isBool(const std::string &sData)
check if the given value can be converted to bool
static std::string to_upper_case(const std::string &str)
Transfers the content to upper case.
static std::string charToHex(unsigned char c)
char to hexadecimal
static std::string urlDecode(const std::string &encoded)
decode url (stem from http://bogomip.net/blog/cpp-url-encoding-and-decoding/)
static long long int toLong(const std::string &sData)
converts a string into the long value described by it by calling the char-type converter,...
static double toDoubleSecure(const std::string &sData, const double def)
converts a string into the integer value described by it
static std::string trim(const std::string s, const std::string &t=" \t\n")
remove leading and trailing whitespace
static std::string to_lower_case(const std::string &str)
Transfers the content to lower case.
static std::string trim_right(const std::string s, const std::string &t=" \t\n")
remove trailing whitespace from string
static std::string trim_left(const std::string s, const std::string &t=" \t\n")
remove leading whitespace from string
static std::string escapeShell(const std::string &orig)
Escape special characters with backslash.
static std::string replace(std::string str, const std::string &what, const std::string &by)
Replaces all occurrences of the second string by the third string within the first string.
static int hexToInt(const std::string &sData)
converts a string with a hex value into the integer value described by it by calling the char-type co...
static double toDouble(const std::string &sData)
converts a string into the double value described by it by calling the char-type converter
static std::string escapeXML(const std::string &orig, const bool maskDoubleHyphen=false)
Replaces the standard escapes by their XML entities.
static bool isHex(std::string sData)
check if the given string can be converted to hex
static std::string latin1_to_utf8(std::string str)
Transfers from Latin 1 (ISO-8859-1) to UTF-8.
static std::string prune(const std::string &str)
Removes trailing and leading whitechars.
static std::string padFront(const std::string &str, int length, char padding)
static std::string convertUmlaute(std::string str)
Converts german "Umlaute" to their latin-version.
static double parseDist(const std::string &sData)
parse a distance, length or width value with a unit
static std::string adjustDecimalValue(double value, int precision)
write with maximum precision if needed but remove trailing zeros
static unsigned char hexToChar(const std::string &str)
hexadecimal to char
static bool startsWith(const std::string &str, const std::string prefix)
Checks whether a given string starts with the prefix.
static std::string wrapText(const std::string s, int width)
remove leading and trailing whitespace
static double parseSpeed(const std::string &sData, const bool defaultKmph=true)
parse a speed value with a unit
static std::string emptyString
An empty string.
static bool endsWith(const std::string &str, const std::string suffix)
Checks whether a given string ends with the suffix.
static std::string substituteEnvironment(const std::string &str, const std::chrono::time_point< std::chrono::system_clock > *const timeRef=nullptr)
Replaces an environment variable with its value (similar to bash); syntax for a variable is ${NAME}.
static bool isLong(const std::string &sData)
Check if the given sData can be converted to long.
static int toIntSecure(const std::string &sData, int def)
converts a string into the integer value described by it
static std::string isoTimeString(const std::chrono::time_point< std::chrono::system_clock > *const timeRef=nullptr)
Returns an ISO8601 formatted time string with microsecond precision.
static int toInt(const std::string &sData)
converts a string into the integer value described by it by calling the char-type converter,...
static bool isInt(const std::string &sData)
check if the given sData can be converted to int
static bool toBool(const std::string &sData)
converts a string into the bool value described by it by calling the char-type converter