34 #include <boost/algorithm/string.hpp>
37 #define ERR_GENERAL LOG_STREAM(err, lg::general())
38 #define ERR_NG LOG_STREAM(err, log_engine)
44 return c ==
'\r' ||
c ==
'\n';
52 if (
static_cast<unsigned char>(
c) >= 128)
54 return isnewline(
c) || isspace(
static_cast<unsigned char>(
c));
66 s.remove_prefix(std::min(
s.find_first_not_of(
" \t\r\n"),
s.size()));
71 std::size_t first_to_trim =
s.find_last_not_of(
" \t\r\n") + 1;
72 s =
s.substr(0, first_to_trim);
84 std::vector<std::string>
split(std::string_view
s,
const char sep,
const int flags)
86 std::vector<std::string> res;
88 res.emplace_back(item);
93 std::set<std::string>
split_set(std::string_view
s,
char sep,
const int flags)
95 std::set<std::string> res;
102 std::vector<std::string_view>
split_view(std::string_view
s,
const char sep,
const int flags)
104 std::vector<std::string_view> res;
112 const char separator,
const std::string& left,
113 const std::string& right,
const int flags)
115 std::vector< std::string > res;
116 std::vector<char> part;
117 bool in_parenthesis =
false;
118 std::vector<std::string::const_iterator> square_left;
119 std::vector<std::string::const_iterator> square_right;
120 std::vector< std::string > square_expansion;
123 std::string rp=right;
125 std::string::const_iterator i1 = val.begin();
126 std::string::const_iterator i2;
127 std::string::const_iterator j1;
135 if (i1 == val.end())
return res;
138 ERR_GENERAL <<
"Separator must be specified for square bracket split function.";
142 if(left.size()!=right.size()){
143 ERR_GENERAL <<
"Left and Right Parenthesis lists not same length";
148 if(i2 == val.end() || (!in_parenthesis && *i2 == separator)) {
150 std::size_t size_square_exp = 0;
151 for (std::size_t
i=0;
i < square_left.size();
i++) {
152 std::string tmp_val(square_left[
i]+1,square_right[
i]);
153 std::vector< std::string > tmp =
split(tmp_val);
154 for(
const std::string& piece : tmp) {
155 std::size_t found_tilde = piece.find_first_of(
'~');
156 if (found_tilde == std::string::npos) {
157 std::size_t found_asterisk = piece.find_first_of(
'*');
158 if (found_asterisk == std::string::npos) {
159 std::string tmp2(piece);
161 square_expansion.push_back(tmp2);
164 std::string s_begin = piece.substr(0,found_asterisk);
166 std::string s_end = piece.substr(found_asterisk+1);
168 for (
int ast=
std::stoi(s_end); ast>0; --ast)
169 square_expansion.push_back(s_begin);
173 std::string s_begin = piece.substr(0,found_tilde);
176 std::size_t padding = 0, padding_end = 0;
177 while (padding<s_begin.size() && s_begin[padding]==
'0') {
180 std::string s_end = piece.substr(found_tilde+1);
183 while (padding_end<s_end.size() && s_end[padding_end]==
'0') {
186 if (padding*padding_end > 0 && s_begin.size() != s_end.size()) {
187 ERR_GENERAL <<
"Square bracket padding sizes not matching: "
188 << s_begin <<
" and " << s_end <<
".";
190 if (padding_end > padding) padding = padding_end;
192 int increment = (end >= begin ? 1 : -1);
194 for (
int k=begin; k!=end; k+=increment) {
195 std::string pb = std::to_string(k);
196 for (std::size_t
p=pb.size();
p<=padding;
p++)
197 pb = std::string(
"0") + pb;
198 square_expansion.push_back(pb);
202 if (
i*square_expansion.size() != (
i+1)*size_square_exp ) {
203 std::string tmp2(i1, i2);
204 ERR_GENERAL <<
"Square bracket lengths do not match up: " << tmp2;
207 size_square_exp = square_expansion.size();
212 std::size_t j_max = 0;
213 if (!square_left.empty())
214 j_max = square_expansion.size() / square_left.size();
218 for (std::size_t
i=0;
i < square_left.size();
i++) {
219 std::string tmp_val(j1, square_left[
i]);
220 new_val.append(tmp_val);
221 std::size_t k = j+
i*j_max;
222 if (k < square_expansion.size())
223 new_val.append(square_expansion[k]);
224 j1 = square_right[
i]+1;
226 std::string tmp_val(j1, i2);
227 new_val.append(tmp_val);
229 boost::trim_right(new_val);
231 res.push_back(new_val);
244 square_right.clear();
245 square_expansion.clear();
248 if(!part.empty() && *i2 == part.back()) {
250 if (*i2 ==
']') square_right.push_back(i2);
252 in_parenthesis =
false;
257 for(std::size_t
i=0;
i < lp.size();
i++) {
260 square_left.push_back(i2);
262 part.push_back(rp[
i]);
270 in_parenthesis =
true;
281 const std::string& val
285 ,
const std::string& default_value)
288 std::vector< std::string > v =
split(val, major, flags);
291 std::map< std::string, std::string > res;
294 std::size_t pos =
i->find_first_of(minor);
295 std::string key, value;
297 if(pos == std::string::npos) {
299 value = default_value;
301 key =
i->substr(0, pos);
302 value =
i->substr(pos + 1);
312 const char separator, std::string_view left,
313 std::string_view right,
const int flags)
315 std::vector< std::string > res;
316 std::vector<char> part;
317 bool in_parenthesis =
false;
319 std::string_view::const_iterator i1 = val.begin();
320 std::string_view::const_iterator i2;
327 if(left.size()!=right.size()){
328 ERR_GENERAL <<
"Left and Right Parenthesis lists not same length";
332 while (i2 != val.end()) {
333 if(!in_parenthesis && separator && *i2 == separator){
334 std::string new_val(i1, i2);
336 boost::trim_right(new_val);
338 res.push_back(new_val);
347 if(!part.empty() && *i2 == part.back()){
349 if(!separator && part.empty()){
350 std::string new_val(i1, i2);
353 res.push_back(new_val);
358 in_parenthesis =
false;
364 for(std::size_t
i=0;
i < left.size();
i++){
366 if (!separator && part.empty()){
367 std::string new_val(i1, i2);
370 res.push_back(new_val);
376 part.push_back(right[
i]);
384 in_parenthesis =
true;
387 std::string new_val(i1, i2);
391 res.push_back(std::move(new_val));
401 int apply_modifier(
const int number,
const std::string &amount,
const int minimum ) {
406 }
catch(
const std::invalid_argument&) {}
407 if(amount[amount.size()-1] ==
'%') {
411 if (( minimum > 0 ) && ( value < minimum ))
416 std::string
escape(std::string_view str,
const char *special_chars)
418 std::string::size_type pos = str.find_first_of(special_chars);
419 if (pos == std::string::npos) {
421 return std::string(str);
423 std::string res = std::string(str);
425 res.insert(pos, 1,
'\\');
426 pos = res.find_first_of(special_chars, pos + 2);
427 }
while (pos != std::string::npos);
433 std::string::size_type pos = str.find(
'\\');
434 if (pos == std::string::npos) {
436 return std::string(str);
438 std::string res = std::string(str);
441 pos = res.find(
'\\', pos + 1);
442 }
while (pos != std::string::npos);
448 static const std::string nonresv_str =
451 "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
453 "abcdefghijklmnopqrstuvwxyz"
455 static const std::set<char> nonresv(nonresv_str.begin(), nonresv_str.end());
457 std::ostringstream res;
462 if(nonresv.count(
c) != 0) {
469 res << static_cast<int>(
c);
476 if (str.empty())
return def;
479 if (str ==
"yes")
return true;
480 if (str ==
"no"|| str ==
"false" || str ==
"off" || str ==
"0" || str ==
"0.0")
489 std::ostringstream ss;
490 ss << std::boolalpha << value;
497 std::ostringstream oss;
504 std::ostringstream oss;
507 oss << std::abs(val);
512 std::streamsize oldprec = ss.precision();
524 input = std::round(input);
530 if (input >= 999.5) {
537 ss.precision(oldprec);
541 const double multiplier = base2 ? 1024 : 1000;
543 typedef std::array<std::string, 9> strings9;
550 strings9::const_iterator prefix;
552 strings9 tmp { {
"",
"",
"",
"",
"",
"",
"",
"",
"" } };
554 prefix = prefixes.begin();
555 }
else if (input < 1.0) {
568 prefix = prefixes.begin();
569 while (input < 1.0 && *prefix != prefixes.back()) {
590 prefix = prefixes.begin();
591 while (input > multiplier && *prefix != prefixes.back()) {
597 std::stringstream ss;
602 << (base2 && (!(*prefix).empty()) ?
_(
"infix_binary^i") :
"")
608 return ((
c ==
'_') || (
c ==
'-'));
612 return ((
c ==
'?') || (
c ==
'*'));
616 const std::size_t alnum = std::count_if(username.begin(), username.end(), isalnum);
617 const std::size_t valid_char =
619 if ((alnum + valid_char != username.size())
620 || valid_char == username.size() || username.empty() )
628 const std::size_t alnum = std::count_if(username.begin(), username.end(), isalnum);
629 const std::size_t valid_char =
631 const std::size_t wild_char =
633 if ((alnum + valid_char + wild_char != username.size())
634 || valid_char == username.size() || username.empty() )
643 std::vector<std::string> matches;
644 const std::size_t last_space = text.rfind(
" ");
646 if (last_space == text.size() -1) {
652 std::string semiword;
653 if (last_space == std::string::npos) {
658 semiword.assign(text, last_space + 1, text.size());
661 std::string best_match = semiword;
662 for (std::vector<std::string>::const_iterator word = wordlist.begin();
663 word != wordlist.end(); ++word)
665 if (word->size() < semiword.size()
666 || !std::equal(semiword.begin(), semiword.end(), word->begin(),
667 [](
char a,
char b) { return tolower(a) == tolower(b); }))
671 if (matches.empty()) {
675 while (toupper(best_match[j]) == toupper((*word)[j])) j++;
676 if (best_match.begin() + j < best_match.end()) {
677 best_match.erase(best_match.begin() + j, best_match.end());
680 matches.push_back(*word);
682 if(!matches.empty()) {
683 text.replace(last_space + 1, best_match.size(), best_match);
690 return (
c ==
' ' ||
c ==
',' ||
c ==
':' ||
c ==
'\'' ||
c ==
'"' ||
c ==
'-');
693 bool word_match(
const std::string& message,
const std::string& word) {
694 std::size_t first = message.find(word);
695 if (first == std::string::npos)
return false;
697 std::size_t next = first + word.size();
706 const bool wild_matching = (!match.empty() && (match[0] ==
'*' || match[0] ==
'+'));
707 const std::string::size_type solid_begin = match.find_first_not_of(
"*+");
708 const bool have_solids = (solid_begin != std::string::npos);
711 const std::string::size_type plus_count = std::count(match.begin(), match.end(),
'+');
712 return match.empty() ? str.empty() : str.length() >= plus_count;
713 }
else if(str.empty()) {
717 const std::string::size_type solid_end = match.find_first_of(
"*+", solid_begin);
718 const std::string::size_type solid_len = (solid_end == std::string::npos)
719 ? match.length() - solid_begin : solid_end - solid_begin;
722 std::string::size_type current = match[0] ==
'+' ? 1 : 0;
727 const std::string::size_type test_len = str.length() - current;
728 for(std::string::size_type
i=0;
i < solid_len && matches; ++
i) {
729 char solid_c = match[solid_begin +
i];
730 if(
i > test_len || !(solid_c ==
'?' || solid_c == str[current+
i])) {
736 const std::string consumed_match = (solid_begin+solid_len < match.length())
737 ? match.substr(solid_end) :
"";
738 const std::string consumed_str = (solid_len < test_len)
739 ? str.substr(current+solid_len) :
"";
742 }
while(wild_matching && !matches && ++current < str.length());
748 std::replace(str.begin(), str.end(),
'*',
'%');
752 while((
n = str.find(
"_",
n)) != std::string::npos)
754 str.replace(
n, 1,
"\\_");
760 std::string
indent(
const std::string&
string, std::size_t indent_size)
762 if(indent_size == 0) {
766 const std::string
indent(indent_size,
' ');
772 const std::vector<std::string>& lines =
split(
string,
'\x0A', 0);
775 for(std::size_t lno = 0; lno < lines.size();) {
776 const std::string&
line = lines[lno];
779 if(!
line.empty() &&
line !=
"\x0D") {
785 if(++lno < lines.size()) {
795 std::vector<std::string> res;
797 std::string::const_iterator i1 = val.begin();
798 std::string::const_iterator i2 = val.begin();
800 while (i2 != val.end()) {
804 if (i2 != val.end()) ++i2;
805 }
else if (*i2 ==
c) {
806 std::string new_val(i1, i2);
810 res.push_back(std::move(new_val));
813 while(i2 != val.end() && *i2 ==
' ')
823 std::string new_val(i1, i2);
827 res.push_back(new_val);
840 std::pair<std::string_view, utils::optional<std::string_view>> parse_range_internal_separator(std::string_view str)
844 static const auto separator = std::string{
"-"};
849 auto pos = str.find(separator, 1);
850 auto length = separator.size();
852 if(pos != std::string::npos && pos + length < str.size()) {
853 return {str.substr(0, pos), str.substr(pos + length)};
856 return {str, utils::nullopt};
862 auto [a,
b] = parse_range_internal_separator(str);
863 std::pair<int, int> res{0, 0};
865 if(a ==
"-infinity" &&
b) {
868 res.first = std::numeric_limits<int>::min();
874 res.second = res.first;
875 }
else if(*
b ==
"infinity") {
876 res.second = std::numeric_limits<int>::max();
879 if(res.second < res.first) {
880 res.second = res.first;
883 }
catch(
const std::invalid_argument&) {
892 auto [a,
b] = parse_range_internal_separator(str);
893 std::pair<double, double> res{0, 0};
895 if(a ==
"-infinity" &&
b) {
898 static_assert(std::numeric_limits<double>::is_iec559,
899 "Don't know how negative infinity is treated on this architecture");
900 res.first = -std::numeric_limits<double>::infinity();
906 res.second = res.first;
907 }
else if(*
b ==
"infinity") {
908 res.second = std::numeric_limits<double>::infinity();
911 if(res.second < res.first) {
912 res.second = res.first;
915 }
catch(
const std::invalid_argument&) {
925 if(std::any_of(to_return.begin(), to_return.end(), [](
const std::pair<int, int>& r) { return r.first < 0; })) {
926 ERR_GENERAL <<
"Invalid range (expected values to be zero or positive): " << str;
935 std::vector<std::pair<double, double>> to_return;
945 std::vector<std::pair<int, int>> to_return;
955 const std::size_t prev_size = str.length();
959 if(str.length() != prev_size) {
This class represents a single unit of a specific type.
static std::string _(const char *str)
Standard logging facilities (interface).
constexpr int div100rounded(int num)
Guarantees portable results for division by 100; round half up, to the nearest integer.
void line(int from_x, int from_y, int to_x, int to_y)
Draw a line.
const std::string ellipsis
const std::string unicode_minus
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
std::string & truncate(std::string &str, const std::size_t size)
Truncates a UTF-8 string to the specified number of characters.
@ STRIP_SPACES
REMOVE_EMPTY: remove empty elements.
std::vector< std::string_view > split_view(std::string_view s, const char sep, const int flags)
std::string si_string(double input, bool base2, const std::string &unit)
Convert into a string with an SI-postfix.
void trim(std::string_view &s)
std::string indent(const std::string &string, std::size_t indent_size)
Indent a block of text.
std::map< std::string, std::string > map_split(const std::string &val, char major, char minor, int flags, const std::string &default_value)
Splits a string based on two separators into a map.
bool isvalid_wildcard(const std::string &username)
Check if the username pattern contains only valid characters.
std::string quote(std::string_view str)
Surround the string 'str' with double quotes.
std::set< std::string > split_set(std::string_view s, char sep, const int flags)
std::vector< std::string > quoted_split(const std::string &val, char c, int flags, char quote)
This function is identical to split(), except it does not split when it otherwise would if the previo...
int stoi(std::string_view str)
Same interface as std::stoi and meant as a drop in replacement, except:
static void si_string_impl_stream_write(std::stringstream &ss, double input)
std::vector< std::pair< int, int > > parse_ranges_int(const std::string &str)
Handles a comma-separated list of inputs to parse_range.
std::vector< std::string > parenthetical_split(std::string_view val, const char separator, std::string_view left, std::string_view right, const int flags)
Splits a string based either on a separator, except then the text appears within specified parenthesi...
std::string half_signed_value(int val)
Sign with Unicode "−" if negative.
std::string bool_string(const bool value)
Converts a bool value to 'true' or 'false'.
static bool is_word_boundary(char c)
void ellipsis_truncate(std::string &str, const std::size_t size)
Truncates a string to a given utf-8 character count and then appends an ellipsis.
std::vector< std::pair< int, int > > parse_ranges_unsigned(const std::string &str)
Handles a comma-separated list of inputs to parse_range, in a context that does not expect negative v...
std::string urlencode(std::string_view str)
Percent-escape characters in a UTF-8 string intended to be part of a URL.
void to_sql_wildcards(std::string &str, bool underscores)
Converts '*' to '' and optionally escapes '_'.
void split_foreach(std::string_view s, char sep, const int flags, const F &f)
bool wildcard_string_match(const std::string &str, const std::string &match)
Match using '*' as any number of characters (including none), '+' as one or more characters,...
bool string_bool(const std::string &str, bool def)
Convert no, false, off, 0, 0.0 to false, empty to def, and others to true.
bool isvalid_username(const std::string &username)
Check if the username contains only valid characters.
double stod(std::string_view str)
Same interface as std::stod and meant as a drop in replacement, except:
bool portable_isspace(const char c)
int apply_modifier(const int number, const std::string &amount, const int minimum)
std::vector< std::string > square_parenthetical_split(const std::string &val, const char separator, const std::string &left, const std::string &right, const int flags)
Similar to parenthetical_split, but also expands embedded square brackets.
std::pair< int, int > parse_range(std::string_view str)
Recognises the following patterns, and returns a {min, max} pair.
bool isnewline(const char c)
bool notspace(const char c)
std::string unescape(std::string_view str)
Remove all escape characters (backslash)
std::vector< std::pair< double, double > > parse_ranges_real(const std::string &str)
bool word_match(const std::string &message, const std::string &word)
Check if a message contains a word.
std::string escape(std::string_view str, const char *special_chars)
Prepends a configurable set of characters with a backslash.
std::string signed_value(int val)
Convert into a signed value (using the Unicode "−" and +0 convention.
std::pair< double, double > parse_range_real(std::string_view str)
Recognises similar patterns to parse_range, and returns a {min, max} pair.
std::vector< std::string > split(const config_attribute_value &val)
bool word_completion(std::string &text, std::vector< std::string > &wordlist)
Try to complete the last word of 'text' with the 'wordlist'.
static bool is_wildcard_char(char c)
static bool is_username_char(char c)
std::string::const_iterator iterator
static lg::log_domain log_engine("engine")
static map_location::direction n
static map_location::direction s