yoda is hosted by Hepforge, IPPP Durham
YODA - Yet more Objects for Data Analysis 2.1.0
StringUtils.h
Go to the documentation of this file.
1// -*- C++ -*-
2//
3// This file is part of YODA -- Yet more Objects for Data Analysis
4// Copyright (C) 2008-2025 The YODA collaboration (see AUTHORS for details)
5//
6#ifndef YODA_STRINGUTILS_H
7#define YODA_STRINGUTILS_H
8
9#include <algorithm>
10#include <cstdlib>
11#include <limits>
12#include <sstream>
13#include <stdexcept>
14#include <vector>
15
16namespace YODA {
17 namespace Utils {
18
19
21 struct bad_lexical_cast : public std::runtime_error {
22 bad_lexical_cast(const std::string& what) : std::runtime_error(what) {}
23 };
24
26 template<typename T, typename U>
27 T lexical_cast(const U& in) {
28 try {
29 std::stringstream ss;
30 ss << in;
31 T out;
32 ss >> out;
33 return out;
34 } catch (const std::exception& e) {
35 throw bad_lexical_cast(e.what());
36 }
37 }
38
39
41 template <typename T>
42 inline std::string toStr(const T& x) {
43 // std::ostringstream ss; ss << x;
44 // return ss.str();
45 return lexical_cast<std::string>(x);
46 }
47
48
50 inline double toDbl(const std::string& s) {
51 try {
52 return std::stod(s);
53 }
54 catch (...) {
55 return std::numeric_limits<double>::quiet_NaN();
56 }
57 }
58
59
61 inline std::string toLower(const std::string& s) {
62 std::string out = s;
63 std::transform(out.begin(), out.end(), out.begin(), (int(*)(int)) tolower);
64 return out;
65 }
66
67
69 inline std::string toUpper(const std::string& s) {
70 std::string out = s;
71 std::transform(out.begin(), out.end(), out.begin(), (int(*)(int)) toupper);
72 return out;
73 }
74
75
77 inline std::string encodeForXML(const std::string& in) {
78 std::string out = in;
79 typedef std::pair<std::string, std::string> CharsToEntities;
80 std::vector<CharsToEntities> cs2es;
81 cs2es.push_back(std::make_pair("&", "&amp;"));
82 cs2es.push_back(std::make_pair("<", "&lt;"));
83 cs2es.push_back(std::make_pair(">", "&gt;"));
84 for (std::vector<CharsToEntities>::const_iterator c2e = cs2es.begin(); c2e != cs2es.end(); ++c2e) {
85 std::string::size_type pos = -1;
86 while ( ( pos = out.find(c2e->first, pos + 1) ) != std::string::npos ) {
87 out.replace(pos, 1, c2e->second);
88 }
89 }
90 return out;
91 }
92
93
95 inline bool contains(const std::string& s, const std::string& sub) {
96 return s.find(sub) != std::string::npos;
97 }
98
100 inline bool startswith(const std::string& s, const std::string& sub) {
101 return s.find(sub) == 0;
102 }
103
105 inline bool endswith(const std::string& s, const std::string& sub) {
106 const size_t pos = s.find(sub);
107 return pos != std::string::npos && pos == (s.size()-sub.size());
108 }
109
110
112 inline std::string& iltrim(std::string& s) {
113 s.erase(s.begin(), std::find_if(s.begin(), s.end(), [](unsigned char c){ return !std::isspace(c); }));
114
115 return s;
116 }
117
119 inline std::string& irtrim(std::string& s) {
120 s.erase(std::find_if(s.rbegin(), s.rend(), [](unsigned char c){ return !std::isspace(c); }).base(), s.end());
121 return s;
122 }
123
125 inline std::string& itrim(std::string& s) {
126 return iltrim(irtrim(s));
127 }
128
129
131 inline std::string ltrim(const std::string& s) { //< @note Could just pass by value, but const& sends a symbolic message
132 std::string s2 = s;
133 return iltrim(s2);
134 }
135
137 inline std::string rtrim(const std::string& s) { //< @note Could just pass by value, but const& sends a symbolic message
138 std::string s2 = s;
139 return irtrim(s2);
140 }
141
143 inline std::string trim(const std::string& s) { //< @note Could just pass by value, but const& sends a symbolic message
144 std::string s2 = s;
145 return itrim(s2);
146 }
147
149 inline std::vector<std::string> split(const std::string& s, const std::string& sep) {
150 std::vector<std::string> dirs;
151 std::string tmp = s;
152 while (true) {
153 const size_t delim_pos = tmp.find(sep);
154 if (delim_pos == std::string::npos) break;
155 const std::string dir = tmp.substr(0, delim_pos);
156 if (dir.length()) dirs.push_back(dir); // Don't insert "empties"
157 tmp.replace(0, delim_pos+1, "");
158 }
159 if (tmp.length()) dirs.push_back(tmp); // Don't forget the trailing component!
160 return dirs;
161 }
162
166 inline std::vector<std::string> pathsplit(const std::string& path) {
167 return split(path, ":");
168 }
169
170 // Annoyingly, this doesn't work without significant SFINAE etc., due to clashes with existing string & stream operators
171 // /// string + any appending operator for convenience in the YODA namespace
172 // template <typename T>
173 // std::string& operator + (std::string& s, const T& x) {
174 // s += Utils::toStr(x);
175 // return s;
176 // }
177 // template <typename T>
178 // std::string operator + (const T& x, std::string& s) {
179 // return Utils::toStr(x) + s;
180 // }
181 // template <typename T>
182 // std::string operator + (const char* s, const T& x) {
183 // return s + Utils::toStr(x);
184 // }
185 // template <typename T>
186 // std::string operator + (const T& x, const char* s) {
187 // return Utils::toStr(x) + x;
188 // }
189
190
194 template <typename T>
195 T getEnvParam(const std::string& name, const T& fallback) {
196 char* env = getenv(name.c_str());
197 return env ? lexical_cast<T>(env) : fallback;
198 }
199
200 }
201
202}
203
204#endif
Anonymous namespace to limit visibility.