2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2012-09-17 13:35:55 +02:00
|
|
|
|
2014-10-19 14:48:19 +02:00
|
|
|
#include "base/string.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/value.hpp"
|
2014-10-31 08:49:14 +01:00
|
|
|
#include "base/primitivetype.hpp"
|
2014-12-12 15:19:23 +01:00
|
|
|
#include "base/dictionary.hpp"
|
2018-01-04 18:24:45 +01:00
|
|
|
#include <boost/algorithm/string/case_conv.hpp>
|
|
|
|
#include <boost/algorithm/string/trim.hpp>
|
|
|
|
#include <boost/algorithm/string/split.hpp>
|
2014-05-11 17:14:35 +02:00
|
|
|
#include <ostream>
|
2012-08-02 09:38:08 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2018-01-03 06:01:02 +01:00
|
|
|
template class std::vector<String>;
|
|
|
|
|
2014-12-12 15:19:23 +01:00
|
|
|
REGISTER_BUILTIN_TYPE(String, String::GetPrototype());
|
2014-10-31 08:49:14 +01:00
|
|
|
|
2014-05-11 06:30:50 +02:00
|
|
|
const String::SizeType String::NPos = std::string::npos;
|
2012-08-02 09:38:08 +02:00
|
|
|
|
2018-01-03 06:01:02 +01:00
|
|
|
String::String(const char *data)
|
|
|
|
: m_Data(data)
|
|
|
|
{ }
|
|
|
|
|
2018-01-04 08:54:18 +01:00
|
|
|
String::String(std::string data)
|
2018-01-03 06:01:02 +01:00
|
|
|
: m_Data(std::move(data))
|
|
|
|
{ }
|
|
|
|
|
|
|
|
String::String(String::SizeType n, char c)
|
|
|
|
: m_Data(n, c)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
String::String(const String& other)
|
|
|
|
: m_Data(other)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
String::String(String&& other)
|
|
|
|
: m_Data(std::move(other.m_Data))
|
|
|
|
{ }
|
|
|
|
|
2016-08-31 13:38:55 +02:00
|
|
|
#ifndef _MSC_VER
|
2016-08-31 12:48:49 +02:00
|
|
|
String::String(Value&& other)
|
|
|
|
{
|
|
|
|
*this = std::move(other);
|
|
|
|
}
|
2016-08-31 13:38:55 +02:00
|
|
|
#endif /* _MSC_VER */
|
2016-08-31 12:48:49 +02:00
|
|
|
|
|
|
|
String& String::operator=(Value&& other)
|
|
|
|
{
|
2016-09-01 15:25:14 +02:00
|
|
|
if (other.IsString())
|
|
|
|
m_Data = std::move(other.Get<String>());
|
2016-08-31 12:48:49 +02:00
|
|
|
else
|
2016-09-02 08:14:15 +02:00
|
|
|
*this = static_cast<String>(other);
|
2016-08-31 12:48:49 +02:00
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2013-11-07 13:41:24 +01:00
|
|
|
String& String::operator+=(const Value& rhs)
|
|
|
|
{
|
|
|
|
m_Data += static_cast<String>(rhs);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2018-01-03 06:01:02 +01:00
|
|
|
String& String::operator=(const String& rhs)
|
|
|
|
{
|
|
|
|
m_Data = rhs.m_Data;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
String& String::operator=(String&& rhs)
|
|
|
|
{
|
|
|
|
m_Data = std::move(rhs.m_Data);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
String& String::operator=(const std::string& rhs)
|
|
|
|
{
|
|
|
|
m_Data = rhs;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
String& String::operator=(const char *rhs)
|
|
|
|
{
|
|
|
|
m_Data = rhs;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char& String::operator[](String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data[pos];
|
|
|
|
}
|
|
|
|
|
|
|
|
char& String::operator[](String::SizeType pos)
|
|
|
|
{
|
|
|
|
return m_Data[pos];
|
|
|
|
}
|
|
|
|
|
|
|
|
String& String::operator+=(const String& rhs)
|
|
|
|
{
|
|
|
|
m_Data += rhs.m_Data;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
String& String::operator+=(const char *rhs)
|
|
|
|
{
|
|
|
|
m_Data += rhs;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
String& String::operator+=(char rhs)
|
|
|
|
{
|
|
|
|
m_Data += rhs;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
bool String::IsEmpty() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool String::operator<(const String& rhs) const
|
|
|
|
{
|
|
|
|
return m_Data < rhs.m_Data;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::operator const std::string&() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data;
|
|
|
|
}
|
|
|
|
|
2020-12-22 14:36:48 +01:00
|
|
|
/**
|
|
|
|
* Conversion function to boost::string_view.
|
|
|
|
*
|
|
|
|
* This allows using String as the value for HTTP headers in boost::beast::http::basic_fields::set.
|
|
|
|
*
|
|
|
|
* @return A boost::string_view representing this string.
|
|
|
|
*/
|
|
|
|
String::operator boost::string_view() const
|
|
|
|
{
|
|
|
|
return boost::string_view(m_Data);
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
const char *String::CStr() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.c_str();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void String::Clear()
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
m_Data.clear();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::SizeType String::GetLength() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.size();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
std::string& String::GetData()
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
const std::string& String::GetData() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::Find(const String& str, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.find(str, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::RFind(const String& str, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.rfind(str, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::FindFirstOf(const char *s, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.find_first_of(s, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::FindFirstOf(char ch, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.find_first_of(ch, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::FindFirstNotOf(const char *s, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.find_first_not_of(s, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::FindFirstNotOf(char ch, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.find_first_not_of(ch, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::FindLastOf(const char *s, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.find_last_of(s, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::SizeType String::FindLastOf(char ch, String::SizeType pos) const
|
|
|
|
{
|
|
|
|
return m_Data.find_last_of(ch, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
String String::SubStr(String::SizeType first, String::SizeType len) const
|
|
|
|
{
|
|
|
|
return m_Data.substr(first, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<String> String::Split(const char *separators) const
|
|
|
|
{
|
|
|
|
std::vector<String> result;
|
|
|
|
boost::algorithm::split(result, m_Data, boost::is_any_of(separators));
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void String::Replace(String::SizeType first, String::SizeType second, const String& str)
|
|
|
|
{
|
|
|
|
m_Data.replace(first, second, str);
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String String::Trim() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
String t = m_Data;
|
|
|
|
boost::algorithm::trim(t);
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String String::ToLower() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
String t = m_Data;
|
|
|
|
boost::algorithm::to_lower(t);
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String String::ToUpper() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
String t = m_Data;
|
|
|
|
boost::algorithm::to_upper(t);
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String String::Reverse() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
String t = m_Data;
|
|
|
|
std::reverse(t.m_Data.begin(), t.m_Data.end());
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
void String::Append(int count, char ch)
|
|
|
|
{
|
|
|
|
m_Data.append(count, ch);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool String::Contains(const String& str) const
|
|
|
|
{
|
|
|
|
return (m_Data.find(str) != std::string::npos);
|
|
|
|
}
|
|
|
|
|
|
|
|
void String::swap(String& str)
|
|
|
|
{
|
|
|
|
m_Data.swap(str.m_Data);
|
|
|
|
}
|
|
|
|
|
|
|
|
String::Iterator String::erase(String::Iterator first, String::Iterator last)
|
|
|
|
{
|
|
|
|
return m_Data.erase(first, last);
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::Iterator String::Begin()
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.begin();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::ConstIterator String::Begin() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.begin();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::Iterator String::End()
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.end();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::ConstIterator String::End() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.end();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::ReverseIterator String::RBegin()
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.rbegin();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::ConstReverseIterator String::RBegin() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.rbegin();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::ReverseIterator String::REnd()
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.rend();
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String::ConstReverseIterator String::REnd() const
|
2018-01-03 06:01:02 +01:00
|
|
|
{
|
|
|
|
return m_Data.rend();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream& icinga::operator<<(std::ostream& stream, const String& str)
|
|
|
|
{
|
|
|
|
stream << str.GetData();
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::istream& icinga::operator>>(std::istream& stream, String& str)
|
|
|
|
{
|
|
|
|
std::string tstr;
|
|
|
|
stream >> tstr;
|
|
|
|
str = tstr;
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
String icinga::operator+(const String& lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() + rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
String icinga::operator+(const String& lhs, const char *rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() + rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
String icinga::operator+(const char *lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs + rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator==(const String& lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() == rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator==(const String& lhs, const char *rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() == rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator==(const char *lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs == rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator<(const String& lhs, const char *rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() < rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator<(const char *lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs < rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator>(const String& lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() > rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator>(const String& lhs, const char *rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() > rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator>(const char *lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs > rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator<=(const String& lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() <= rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator<=(const String& lhs, const char *rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() <= rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator<=(const char *lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs <= rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator>=(const String& lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() >= rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator>=(const String& lhs, const char *rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() >= rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator>=(const char *lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs >= rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator!=(const String& lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() != rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator!=(const String& lhs, const char *rhs)
|
|
|
|
{
|
|
|
|
return lhs.GetData() != rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool icinga::operator!=(const char *lhs, const String& rhs)
|
|
|
|
{
|
|
|
|
return lhs != rhs.GetData();
|
|
|
|
}
|
|
|
|
|
|
|
|
String::Iterator icinga::begin(String& x)
|
|
|
|
{
|
|
|
|
return x.Begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
String::ConstIterator icinga::begin(const String& x)
|
|
|
|
{
|
|
|
|
return x.Begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
String::Iterator icinga::end(String& x)
|
|
|
|
{
|
|
|
|
return x.End();
|
|
|
|
}
|
|
|
|
|
|
|
|
String::ConstIterator icinga::end(const String& x)
|
|
|
|
{
|
|
|
|
return x.End();
|
|
|
|
}
|
|
|
|
String::Iterator icinga::range_begin(String& x)
|
|
|
|
{
|
|
|
|
return x.Begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
String::ConstIterator icinga::range_begin(const String& x)
|
|
|
|
{
|
|
|
|
return x.Begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
String::Iterator icinga::range_end(String& x)
|
|
|
|
{
|
|
|
|
return x.End();
|
|
|
|
}
|
|
|
|
|
|
|
|
String::ConstIterator icinga::range_end(const String& x)
|
|
|
|
{
|
|
|
|
return x.End();
|
|
|
|
}
|
2022-10-18 10:26:13 +02:00
|
|
|
|
|
|
|
std::size_t std::hash<String>::operator()(const String& s) const noexcept
|
|
|
|
{
|
|
|
|
return std::hash<std::string>{}(s.GetData());
|
|
|
|
}
|