All checks were successful
		
		
	
	BlockStorage/repertory/pipeline/head This commit looks good
				
			Reviewed-on: #50
		
			
				
	
	
		
			179 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|   Copyright <2018-2025> <scott.e.graves@protonmail.com>
 | |
| 
 | |
|   Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
|   of this software and associated documentation files (the "Software"), to deal
 | |
|   in the Software without restriction, including without limitation the rights
 | |
|   to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | |
|   copies of the Software, and to permit persons to whom the Software is
 | |
|   furnished to do so, subject to the following conditions:
 | |
| 
 | |
|   The above copyright notice and this permission notice shall be included in all
 | |
|   copies or substantial portions of the Software.
 | |
| 
 | |
|   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|   IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
|   AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
|   LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | |
|   OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | |
|   SOFTWARE.
 | |
| */
 | |
| #ifndef REPERTORY_INCLUDE_UTILS_COLLECTION_HPP_
 | |
| #define REPERTORY_INCLUDE_UTILS_COLLECTION_HPP_
 | |
| 
 | |
| #include "utils/config.hpp"
 | |
| 
 | |
| #include "utils/error.hpp"
 | |
| #include "utils/string.hpp"
 | |
| 
 | |
| namespace repertory::utils::collection {
 | |
| template <typename col_t>
 | |
| [[nodiscard]] inline auto excludes(const col_t &collection,
 | |
|                                    const typename col_t::value_type &val)
 | |
|     -> bool;
 | |
| 
 | |
| template <typename col_t>
 | |
| [[nodiscard]] inline auto includes(const col_t &collection,
 | |
|                                    const typename col_t::value_type &val)
 | |
|     -> bool;
 | |
| 
 | |
| template <typename val_t>
 | |
| [[nodiscard]] inline auto from_hex_string(std::string_view str, val_t &val)
 | |
|     -> bool;
 | |
| 
 | |
| template <typename val_t>
 | |
| [[nodiscard]] inline auto from_hex_string(std::wstring_view str, val_t &val)
 | |
|     -> bool;
 | |
| template <typename col_t>
 | |
| inline auto remove_element(col_t &collection,
 | |
|                            const typename col_t::value_type &value) -> col_t &;
 | |
| 
 | |
| template <typename col_t>
 | |
| [[nodiscard]] inline auto to_hex_string(const col_t &collection) -> std::string;
 | |
| 
 | |
| template <typename col_t>
 | |
| [[nodiscard]] inline auto to_hex_wstring(const col_t &collection)
 | |
|     -> std::wstring;
 | |
| 
 | |
| template <typename col_t>
 | |
| inline auto excludes(const col_t &collection,
 | |
|                      const typename col_t::value_type &val) -> bool {
 | |
|   return std::find(collection.begin(), collection.end(), val) ==
 | |
|          collection.end();
 | |
| }
 | |
| 
 | |
| template <typename col_t>
 | |
| inline auto includes(const col_t &collection,
 | |
|                      const typename col_t::value_type &val) -> bool {
 | |
|   return std::find(collection.begin(), collection.end(), val) !=
 | |
|          collection.end();
 | |
| }
 | |
| 
 | |
| template <typename val_t>
 | |
| [[nodiscard]] inline auto from_hex_string_t(std::string_view str, val_t &val)
 | |
|     -> bool {
 | |
|   REPERTORY_USES_FUNCTION_NAME();
 | |
| 
 | |
|   static constexpr auto base16{16};
 | |
| 
 | |
|   try {
 | |
|     val.clear();
 | |
| 
 | |
|     std::string fmt_val{str};
 | |
|     utils::string::trim(fmt_val);
 | |
|     if (fmt_val.empty()) {
 | |
|       return true;
 | |
|     }
 | |
| 
 | |
|     fmt_val = utils::string::to_lower(fmt_val);
 | |
|     if (utils::string::begins_with(fmt_val, "0x")) {
 | |
|       fmt_val = fmt_val.substr(2U);
 | |
|     }
 | |
| 
 | |
|     if (fmt_val.empty()) {
 | |
|       throw utils::error::create_exception(function_name,
 | |
|                                            {
 | |
|                                                "hex string is invalid",
 | |
|                                                str,
 | |
|                                            });
 | |
|     }
 | |
| 
 | |
|     if (fmt_val.length() % 2U) {
 | |
|       fmt_val = '0' + fmt_val;
 | |
|     }
 | |
| 
 | |
|     auto iter = std::find_if_not(
 | |
|         fmt_val.begin(), fmt_val.end(), [](auto cur_char) -> bool {
 | |
|           auto check = static_cast<std::uint32_t>(cur_char);
 | |
|           return ((check >= 48U && check <= 57U) ||
 | |
|                   (check >= 97U && check <= 102U));
 | |
|         });
 | |
|     if (iter != fmt_val.end()) {
 | |
|       auto invalid_idx{std::distance(fmt_val.begin(), iter)};
 | |
|       throw std::range_error(utils::error::create_error_message({
 | |
|           function_name,
 | |
|           "invalid character in hex string",
 | |
|           std::to_string(invalid_idx),
 | |
|           std::string(1U, str.at(invalid_idx)),
 | |
|           str,
 | |
|       }));
 | |
|     }
 | |
| 
 | |
|     val.resize(fmt_val.length() / 2U);
 | |
|     for (std::size_t idx = 0U; idx < fmt_val.length(); idx += 2U) {
 | |
|       val.at(idx / 2U) = static_cast<typename val_t::value_type>(
 | |
|           std::strtoul(fmt_val.substr(idx, 2U).c_str(), nullptr, base16));
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
|   } catch (const std::exception &e) {
 | |
|     utils::error::handle_exception(function_name, e);
 | |
|   } catch (...) {
 | |
|     utils::error::handle_exception(function_name);
 | |
|   }
 | |
| 
 | |
|   val.clear();
 | |
|   return false;
 | |
| }
 | |
| template <typename val_t>
 | |
| inline auto from_hex_string(std::string_view str, val_t &val) -> bool {
 | |
|   return from_hex_string_t<val_t>(str, val);
 | |
| }
 | |
| 
 | |
| template <typename val_t>
 | |
| inline auto from_hex_string(std::wstring_view str, val_t &val) -> bool {
 | |
|   return from_hex_string_t<val_t>(utils::string::to_utf8(str), val);
 | |
| }
 | |
| 
 | |
| template <typename col_t>
 | |
| inline auto remove_element(col_t &collection,
 | |
|                            const typename col_t::value_type &value) -> col_t & {
 | |
|   collection.erase(std::remove(collection.begin(), collection.end(), value),
 | |
|                    collection.end());
 | |
|   return collection;
 | |
| }
 | |
| 
 | |
| template <typename col_t>
 | |
| inline auto to_hex_string(const col_t &collection) -> std::string {
 | |
|   static_assert(sizeof(typename col_t::value_type) == 1U,
 | |
|                 "value_type must be 1 byte in size");
 | |
|   static constexpr auto mask{0xFF};
 | |
| 
 | |
|   std::stringstream stream{};
 | |
|   for (auto &&val : collection) {
 | |
|     stream << std::setfill('0') << std::setw(2) << std::hex
 | |
|            << (static_cast<std::uint32_t>(val) & mask);
 | |
|   }
 | |
| 
 | |
|   return stream.str();
 | |
| }
 | |
| 
 | |
| template <typename col_t>
 | |
| inline auto to_hex_wstring(const col_t &collection) -> std::wstring {
 | |
|   return utils::string::from_utf8(to_hex_string<col_t>(collection));
 | |
| }
 | |
| } // namespace repertory::utils::collection
 | |
| 
 | |
| #endif // REPERTORY_INCLUDE_UTILS_COLLECTION_HPP_
 |