/* -------------------------------------------------------------------------- */ /* Copyright 2002-2020, OpenNebula Project, OpenNebula Systems */ /* */ /* Licensed under the Apache License, Version 2.0 (the "License"); you may */ /* not use this file except in compliance with the License. You may obtain */ /* a copy of the License at */ /* */ /* http://www.apache.org/licenses/LICENSE-2.0 */ /* */ /* Unless required by applicable law or agreed to in writing, software */ /* distributed under the License is distributed on an "AS IS" BASIS, */ /* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */ /* See the License for the specific language governing permissions and */ /* limitations under the License. */ /* -------------------------------------------------------------------------- */ #ifndef _NEBULA_UTIL_H_ #define _NEBULA_UTIL_H_ #include #include #include #include #include #include #include namespace one_util { std::string& toupper(std::string& st); std::string& tolower(std::string& st); std::string log_time(time_t the_time); std::string log_time(); /** * Escape XML entity and character references * @param in the string to be escaped * @return a string copy */ std::string xml_escape(const std::string& in); /** * sha1 digest * @param in the string to be hashed * @return sha1 hash of str */ std::string sha1_digest(const std::string& in); /** * sha256 digest * @param in the string to be hashed * @return sha256 hash of str */ std::string sha256_digest(const std::string& in); /** * Base 64 encoding * @param in the string to encoded * @return a pointer to the encoded string (must be freed) or nullptr in case of * error */ std::string * base64_encode(const std::string& in); /** * Base 64 decoding * @param in the string to decode * @return a pointer to the decoded string (must be freed) or nullptr in case of * error */ std::string * base64_decode(const std::string& in); /** * AES256 encryption * @param in the string to encrypt * @param password to encrypt data * @return a pointer to the encrypted string (must be freed) or nullptr in case of * error */ std::string * aes256cbc_encrypt(const std::string& in, const std::string& password); /** * AES256 decryption * @param in the base64 string to decrypt * @param password to decrypt data * @return a pointer to the decrypted string (must be freed) or nullptr in case of * error */ std::string * aes256cbc_decrypt(const std::string& in, const std::string& password); /** * Creates a random number, using time(0) as seed, and performs an sha1 hash * @return a new random password */ std::string random_password(); /** * Returns random number, default range is <0, Type Max Value>, specialization for integer types * @param min - minimal potentially generated number, defaults to 0 * @param max - maximal potentially generated number, defaults to type max value * @return number between min, max */ template::value>::type* = nullptr> Integer random(Integer min = 0, Integer max = std::numeric_limits::max()) { static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; static std::random_device rd; static std::mt19937_64 rng(rd()); std::uniform_int_distribution distribution(min, max); pthread_mutex_lock(&mutex); Integer i = distribution(rng); pthread_mutex_unlock(&mutex); return i; } /** * Returns random number, default range is <0, Type Max Value>, specialization for floating types * @param min - minimal potentially generated number, defaults to 0 * @param max - maximal potentially generated number, defaults to type max value * @return number between min, max */ template::value>::type* = nullptr> Floating random(Floating min = 0, Floating max = std::numeric_limits::max()) { static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; static std::random_device rd; static std::mt19937_64 rng(rd()); std::uniform_real_distribution distribution(min, max); pthread_mutex_lock(&mutex); Floating f = distribution(rng); pthread_mutex_unlock(&mutex); return f; } /** * Splits a string, using the given delimiter * * @param st string to split * @param delim delimiter character * @param clean_empty true to clean empty split parts. * Example for st "a::b:c" * clean_empty true will return ["a", "b", "c"] * clean_empty fase will return ["a", "", "b", "c"] * * @return a vector containing the resulting substrings */ template void split(const std::string &st, char delim, std::vector &parts) { std::string part; std::stringstream ss(st); while (getline(ss, part, delim)) { if (part.empty()) { continue; } std::istringstream iss(part); T part_t; iss >> part_t; if ( iss.fail() ) { continue; } parts.push_back(part_t); } } std::vector split(const std::string& st, char delim, bool clean_empty = true); /** * Splits a string, using the given delimiter * * @param st string to split * @param delim delimiter character * @param result where the result will be saved */ template void split_unique(const std::string& st, char delim, std::set& result) { T elem; std::vector::const_iterator it; std::vector strings = split(st, delim, true); for (it = strings.begin(); it != strings.end(); it++) { std::istringstream iss(*it); iss >> elem; if ( iss.fail() ) { continue; } result.insert(elem); } } /** * Explicit specialization for strings */ template <> void split_unique(const std::string& st, char delim, std::set& result); /** * Joins the given element with the delimiter * * @param first iterator * @param last iterator * @param delim delimiter character * @return the joined strings */ template std::string join(Iterator first, Iterator last, char delim) { std::ostringstream oss; for (Iterator it = first; it != last; it++) { if (it != first) { oss << delim; } oss << *it; } return oss.str(); } /** * Joins the given element with the delimiter * * @param values set of values * @param delim delimiter character * @return the joined strings */ template std::string join(const std::set& values, char delim) { return join(values.begin(), values.end(), delim); } /** * Creates a string from the given float, using fixed notation. If the * number has any decimals, they will be truncated to 2. * * @param num * @return */ std::string float_to_str(const float &num); /** * Returns a scaped version of a value in the from "" * @param v the value to be escaped * @param op the opening escape string * @param cl the closing escape string */ template inline std::string escape(const ValueType& v, const char * op, const char * cl) { std::ostringstream oss; oss << op << v << cl; return oss.str(); } template inline std::string escape_xml(const ValueType &v) { return escape(v, ""); } template inline std::string escape_xml_attr(const ValueType &v) { return escape(v, "'", "'"); } void escape_json(const std::string& str, std::ostringstream& s); void escape_token(const std::string& str, std::ostringstream& s); /** * Checks if a strings matches a regular expression * * @param pattern PCRE extended pattern * @param subject the string to test * @return 0 on match, another value otherwise */ int regex_match(const char *pattern, const char *subject); /** * Trim an string using the isspace function * @param the string * @return trimed string */ std::string trim(const std::string& str); /** * Returns a copy of st with the all occurrences of "find" substituted * for "replacement" * @param st string input * @param sfind string to search for * @param replacement string to replace occurrences with * @return a string copy */ std::string gsub(const std::string& st, const std::string& sfind, const std::string& replacement); template std::set set_intersection(const std::set &first, const std::set &second) { std::set output; std::set_intersection( first.begin(), first.end(), second.begin(), second.end(), std::inserter(output, output.begin())); return output; } /** * Compress the input string unsing zlib * @param in input string * @param bool64 true to base64 encode output * @return pointer to the compressed sting (must be freed) or 0 in case * of error */ std::string * zlib_compress(const std::string& in, bool base64); /** * Decompress the input string unsing zlib * @param in input string * @param base64 true if the input is base64 encoded * @return pointer to the decompressed sting (must be freed) or 0 in case * of error */ std::string * zlib_decompress(const std::string& in, bool base64); extern "C" void sslmutex_lock_callback(int mode, int type, char *file, int line); extern "C" unsigned long sslmutex_id_callback(); class SSLMutex { public: static void initialize(); static void finalize(); private: friend void sslmutex_lock_callback(int mode, int type, char *file, int line); SSLMutex(); ~SSLMutex(); static SSLMutex * ssl_mutex; static std::vector vmutex; }; } // namespace one_util #endif /* _NEBULA_UTIL_H_ */