| @@ -1,95 +1,110 @@ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_CLIENT_POOL_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_CLIENT_POOL_HPP_ | ||||
|  | ||||
| #include "comm/packet/packet.hpp" | ||||
| #include "types/repertory.hpp" | ||||
|  | ||||
| namespace repertory { | ||||
| class client_pool final { | ||||
| public: | ||||
|   using worker_callback = std::function<packet::error_type()>; | ||||
|   using worker_complete_callback = | ||||
|       std::function<void(const packet::error_type &)>; | ||||
|  | ||||
| private: | ||||
|   class pool final { | ||||
|   private: | ||||
|     struct work_item final { | ||||
|       work_item(worker_callback worker, | ||||
|                 worker_complete_callback worker_complete) | ||||
|           : work(std::move(worker)), | ||||
|             work_complete(std::move(worker_complete)) {} | ||||
|  | ||||
|       worker_callback work; | ||||
|       worker_complete_callback work_complete; | ||||
|     }; | ||||
|  | ||||
|     struct work_queue final { | ||||
|       std::mutex mutex; | ||||
|       std::condition_variable notify; | ||||
|       std::deque<std::shared_ptr<work_item>> queue; | ||||
|     }; | ||||
|  | ||||
|   public: | ||||
|     explicit pool(std::uint8_t pool_size); | ||||
|  | ||||
|     ~pool() { shutdown(); } | ||||
|  | ||||
|   private: | ||||
|     std::vector<std::unique_ptr<work_queue>> pool_queues_; | ||||
|     std::vector<std::thread> pool_threads_; | ||||
|     bool shutdown_ = false; | ||||
|     std::atomic<std::uint8_t> thread_index_; | ||||
|  | ||||
|   public: | ||||
|     void execute(std::uint64_t thread_id, const worker_callback &worker, | ||||
|                  const worker_complete_callback &worker_complete); | ||||
|  | ||||
|     void shutdown(); | ||||
|   }; | ||||
|  | ||||
| public: | ||||
|   explicit client_pool(std::uint8_t pool_size = 10u) | ||||
|       : pool_size_(pool_size ? pool_size : 10u) {} | ||||
|  | ||||
|   ~client_pool() { shutdown(); } | ||||
|  | ||||
| private: | ||||
|   const std::uint8_t pool_size_; | ||||
|   std::unordered_map<std::string, std::shared_ptr<pool>> pool_lookup_; | ||||
|   std::mutex pool_mutex_; | ||||
|   bool shutdown_ = false; | ||||
|  | ||||
| public: | ||||
|   void execute(const std::string &client_id, std::uint64_t thread_id, | ||||
|                const worker_callback &worker, | ||||
|                const worker_complete_callback &worker_complete); | ||||
|  | ||||
|   void remove_client(const std::string &client_id); | ||||
|  | ||||
|   void shutdown(); | ||||
| }; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_CLIENT_POOL_HPP_ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_CLIENT_POOL_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_CLIENT_POOL_HPP_ | ||||
|  | ||||
| #include "comm/packet/packet.hpp" | ||||
| #include "types/repertory.hpp" | ||||
|  | ||||
| namespace repertory { | ||||
| class client_pool final { | ||||
| public: | ||||
|   using worker_callback = std::function<packet::error_type()>; | ||||
|   using worker_complete_callback = | ||||
|       std::function<void(const packet::error_type &)>; | ||||
|  | ||||
| private: | ||||
|   class pool final { | ||||
|   private: | ||||
|     struct work_item final { | ||||
|       work_item(worker_callback worker, | ||||
|                 worker_complete_callback worker_complete) | ||||
|           : work(std::move(worker)), | ||||
|             work_complete(std::move(worker_complete)) {} | ||||
|  | ||||
|       worker_callback work; | ||||
|       worker_complete_callback work_complete; | ||||
|     }; | ||||
|  | ||||
|     struct work_queue final { | ||||
|       std::mutex mutex; | ||||
|       std::condition_variable notify; | ||||
|       std::deque<std::shared_ptr<work_item>> queue; | ||||
|     }; | ||||
|  | ||||
|   public: | ||||
|     explicit pool(std::uint8_t pool_size); | ||||
|  | ||||
|     ~pool() { shutdown(); } | ||||
|  | ||||
|   public: | ||||
|     pool(const pool &) = delete; | ||||
|     pool(pool &&) = delete; | ||||
|     auto operator=(const pool &) -> pool & = delete; | ||||
|     auto operator=(pool &&) -> pool & = delete; | ||||
|  | ||||
|   private: | ||||
|     std::vector<std::unique_ptr<work_queue>> pool_queues_; | ||||
|     std::vector<std::thread> pool_threads_; | ||||
|     bool shutdown_{false}; | ||||
|     std::atomic<std::uint8_t> thread_index_{}; | ||||
|  | ||||
|   public: | ||||
|     void execute(std::uint64_t thread_id, const worker_callback &worker, | ||||
|                  const worker_complete_callback &worker_complete); | ||||
|  | ||||
|     void shutdown(); | ||||
|   }; | ||||
|  | ||||
| public: | ||||
|   explicit client_pool(std::uint8_t pool_size = min_pool_size) | ||||
|       : pool_size_(pool_size == 0U ? min_pool_size : pool_size) {} | ||||
|  | ||||
|   ~client_pool() { shutdown(); } | ||||
|  | ||||
| public: | ||||
|   client_pool(const client_pool &) = delete; | ||||
|   client_pool(client_pool &&) = delete; | ||||
|   auto operator=(const client_pool &) -> client_pool & = delete; | ||||
|   auto operator=(client_pool &&) -> client_pool & = delete; | ||||
|  | ||||
| private: | ||||
|   std::uint8_t pool_size_; | ||||
|   std::unordered_map<std::string, std::shared_ptr<pool>> pool_lookup_; | ||||
|   std::mutex pool_mutex_; | ||||
|   bool shutdown_ = false; | ||||
|  | ||||
| private: | ||||
|   static constexpr const auto min_pool_size = 10U; | ||||
|  | ||||
| public: | ||||
|   void execute(const std::string &client_id, std::uint64_t thread_id, | ||||
|                const worker_callback &worker, | ||||
|                const worker_complete_callback &worker_complete); | ||||
|  | ||||
|   void remove_client(const std::string &client_id); | ||||
|  | ||||
|   void shutdown(); | ||||
| }; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_CLIENT_POOL_HPP_ | ||||
|   | ||||
| @@ -1,230 +1,232 @@ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_PACKET_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_PACKET_HPP_ | ||||
|  | ||||
| #include "types/remote.hpp" | ||||
| #include "types/repertory.hpp" | ||||
|  | ||||
| namespace repertory { | ||||
| #define DECODE_OR_RETURN(p, value)                                             \ | ||||
|   if ((ret = (p)->decode(value)) != 0)                                         \ | ||||
|   return ret | ||||
| #define DECODE_OR_IGNORE(p, value)                                             \ | ||||
|   if (ret == 0)                                                                \ | ||||
|   ret = (p)->decode(value) | ||||
|  | ||||
| class packet final { | ||||
| public: | ||||
|   using error_type = std::int32_t; | ||||
|  | ||||
| public: | ||||
|   packet() = default; | ||||
|  | ||||
|   explicit packet(data_buffer buffer) : buffer_(std::move(buffer)) {} | ||||
|  | ||||
|   explicit packet(data_buffer &&buffer) : buffer_(std::move(buffer)) {} | ||||
|  | ||||
|   packet(const packet &p) noexcept = default; | ||||
|  | ||||
|   packet(packet &&p) noexcept | ||||
|       : buffer_(std::move(p.buffer_)), decode_offset_(p.decode_offset_) {} | ||||
|  | ||||
| private: | ||||
|   data_buffer buffer_; | ||||
|   std::size_t decode_offset_ = 0u; | ||||
|  | ||||
| public: | ||||
|   [[nodiscard]] static auto decode_json(packet &response, json &json_data) | ||||
|       -> int; | ||||
|  | ||||
| public: | ||||
|   void clear(); | ||||
|  | ||||
|   [[nodiscard]] auto current_pointer() -> char * { | ||||
|     return (decode_offset_ < buffer_.size()) ? &buffer_[decode_offset_] | ||||
|                                              : nullptr; | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto current_pointer() const -> const char * { | ||||
|     return (decode_offset_ < buffer_.size()) ? &buffer_[decode_offset_] | ||||
|                                              : nullptr; | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::string &data) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::wstring &data) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(void *buffer, std::size_t size) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(void *&ptr) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int8_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint8_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int16_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint16_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int32_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint32_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int64_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint64_t &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::open_flags &i) -> error_type { | ||||
|     return decode(reinterpret_cast<std::uint32_t &>(i)); | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::setattr_x &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::stat &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::statfs &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::statfs_x &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::file_info &i) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decrypt(const std::string &token) -> error_type; | ||||
|  | ||||
|   void encode(const void *buffer, std::size_t size, bool should_reserve = true); | ||||
|  | ||||
|   void encode(char *str) { encode(std::string(str ? str : "")); } | ||||
|  | ||||
|   void encode(const char *str) { encode(std::string(str ? str : "")); } | ||||
|  | ||||
|   void encode(const std::string &str); | ||||
|  | ||||
|   void encode(wchar_t *str); | ||||
|  | ||||
|   void encode(const wchar_t *str); | ||||
|  | ||||
|   void encode(const std::wstring &str); | ||||
|  | ||||
|   void encode(void *ptr) { | ||||
|     encode(static_cast<std::uint64_t>(reinterpret_cast<std::uintptr_t>(ptr))); | ||||
|   } | ||||
|  | ||||
|   void encode(std::int8_t i); | ||||
|  | ||||
|   void encode(std::uint8_t i); | ||||
|  | ||||
|   void encode(std::int16_t i); | ||||
|  | ||||
|   void encode(std::uint16_t i); | ||||
|  | ||||
|   void encode(std::int32_t i); | ||||
|  | ||||
|   void encode(std::uint32_t i); | ||||
|  | ||||
|   void encode(std::int64_t i); | ||||
|  | ||||
|   void encode(std::uint64_t i); | ||||
|  | ||||
|   void encode(remote::open_flags i) { encode(static_cast<std::uint32_t>(i)); } | ||||
|  | ||||
|   void encode(remote::setattr_x i); | ||||
|  | ||||
|   void encode(remote::stat i); | ||||
|  | ||||
|   void encode(remote::statfs i, bool should_reserve = true); | ||||
|  | ||||
|   void encode(remote::statfs_x i); | ||||
|  | ||||
|   void encode(remote::file_info i); | ||||
|  | ||||
|   void encode_top(const void *buffer, std::size_t size, | ||||
|                   bool should_reserve = true); | ||||
|  | ||||
|   void encode_top(const std::string &str); | ||||
|  | ||||
|   void encode_top(const std::wstring &str); | ||||
|  | ||||
|   void encode_top(void *ptr) { | ||||
|     encode_top( | ||||
|         static_cast<std::uint64_t>(reinterpret_cast<std::uintptr_t>(ptr))); | ||||
|   } | ||||
|  | ||||
|   void encode_top(std::int8_t i); | ||||
|  | ||||
|   void encode_top(std::uint8_t i); | ||||
|  | ||||
|   void encode_top(std::int16_t i); | ||||
|  | ||||
|   void encode_top(std::uint16_t i); | ||||
|  | ||||
|   void encode_top(std::int32_t i); | ||||
|  | ||||
|   void encode_top(std::uint32_t i); | ||||
|  | ||||
|   void encode_top(std::int64_t i); | ||||
|  | ||||
|   void encode_top(std::uint64_t i); | ||||
|  | ||||
|   void encode_top(remote::open_flags i) { | ||||
|     encode_top(static_cast<std::uint32_t>(i)); | ||||
|   } | ||||
|  | ||||
|   void encode_top(remote::setattr_x i); | ||||
|  | ||||
|   void encode_top(remote::stat i); | ||||
|  | ||||
|   void encode_top(remote::statfs i, bool should_reserve = true); | ||||
|  | ||||
|   void encode_top(remote::statfs_x i); | ||||
|  | ||||
|   void encode_top(remote::file_info i); | ||||
|  | ||||
|   void encrypt(const std::string &token); | ||||
|  | ||||
|   [[nodiscard]] auto get_size() const -> std::uint32_t { | ||||
|     return static_cast<std::uint32_t>(buffer_.size()); | ||||
|   } | ||||
|  | ||||
|   void transfer_into(data_buffer &buffer); | ||||
|  | ||||
| public: | ||||
|   auto operator=(const data_buffer &buffer) noexcept -> packet &; | ||||
|  | ||||
|   auto operator=(data_buffer &&buffer) noexcept -> packet &; | ||||
|  | ||||
|   auto operator=(const packet &p) noexcept -> packet &; | ||||
|  | ||||
|   auto operator=(packet &&p) noexcept -> packet &; | ||||
|  | ||||
|   [[nodiscard]] auto operator[](std::size_t index) -> char & { | ||||
|     return buffer_[index]; | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto operator[](std::size_t index) const -> const char & { | ||||
|     return buffer_.at(index); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| using packet = packet; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_PACKET_HPP_ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_PACKET_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_PACKET_HPP_ | ||||
|  | ||||
| #include "types/remote.hpp" | ||||
| #include "types/repertory.hpp" | ||||
|  | ||||
| namespace repertory { | ||||
| #define DECODE_OR_RETURN(p, value)                                             \ | ||||
|   if ((ret = (p)->decode(value)) != 0)                                         \ | ||||
|   return ret | ||||
| #define DECODE_OR_IGNORE(p, value)                                             \ | ||||
|   if (ret == 0)                                                                \ | ||||
|   ret = (p)->decode(value) | ||||
|  | ||||
| class packet final { | ||||
| public: | ||||
|   using error_type = std::int32_t; | ||||
|  | ||||
| public: | ||||
|   packet() = default; | ||||
|  | ||||
|   explicit packet(data_buffer buffer) : buffer_(std::move(buffer)) {} | ||||
|  | ||||
|   explicit packet(data_buffer &&buffer) : buffer_(std::move(buffer)) {} | ||||
|  | ||||
|   packet(const packet &pkt) noexcept = default; | ||||
|  | ||||
|   packet(packet &&pkt) noexcept | ||||
|       : buffer_(std::move(pkt.buffer_)), decode_offset_(pkt.decode_offset_) {} | ||||
|  | ||||
|   ~packet() = default; | ||||
|  | ||||
| private: | ||||
|   data_buffer buffer_; | ||||
|   std::size_t decode_offset_ = 0U; | ||||
|  | ||||
| public: | ||||
|   [[nodiscard]] static auto decode_json(packet &response, json &json_data) | ||||
|       -> int; | ||||
|  | ||||
| public: | ||||
|   void clear(); | ||||
|  | ||||
|   [[nodiscard]] auto current_pointer() -> char * { | ||||
|     return (decode_offset_ < buffer_.size()) ? &buffer_[decode_offset_] | ||||
|                                              : nullptr; | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto current_pointer() const -> const char * { | ||||
|     return (decode_offset_ < buffer_.size()) ? &buffer_[decode_offset_] | ||||
|                                              : nullptr; | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::string &data) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::wstring &data) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(void *buffer, std::size_t size) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(void *&ptr) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int8_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint8_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int16_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint16_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int32_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint32_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::int64_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(std::uint64_t &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::open_flags &val) -> error_type { | ||||
|     return decode(reinterpret_cast<std::uint32_t &>(val)); | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::setattr_x &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::stat &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::statfs &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::statfs_x &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decode(remote::file_info &val) -> error_type; | ||||
|  | ||||
|   [[nodiscard]] auto decrypt(const std::string &token) -> error_type; | ||||
|  | ||||
|   void encode(const void *buffer, std::size_t size, bool should_reserve = true); | ||||
|  | ||||
|   void encode(const char *str) { | ||||
|     encode(std::string(str == nullptr ? "" : str)); | ||||
|   } | ||||
|  | ||||
|   void encode(const std::string &str); | ||||
|  | ||||
|   void encode(const wchar_t *str); | ||||
|  | ||||
|   void encode(const std::wstring &str); | ||||
|  | ||||
|   void encode(void *ptr) { | ||||
|     encode(static_cast<std::uint64_t>(reinterpret_cast<std::uintptr_t>(ptr))); | ||||
|   } | ||||
|  | ||||
|   void encode(std::int8_t val); | ||||
|  | ||||
|   void encode(std::uint8_t val); | ||||
|  | ||||
|   void encode(std::int16_t val); | ||||
|  | ||||
|   void encode(std::uint16_t val); | ||||
|  | ||||
|   void encode(std::int32_t val); | ||||
|  | ||||
|   void encode(std::uint32_t val); | ||||
|  | ||||
|   void encode(std::int64_t val); | ||||
|  | ||||
|   void encode(std::uint64_t val); | ||||
|  | ||||
|   void encode(remote::open_flags val) { | ||||
|     encode(static_cast<std::uint32_t>(val)); | ||||
|   } | ||||
|  | ||||
|   void encode(remote::setattr_x val); | ||||
|  | ||||
|   void encode(remote::stat val); | ||||
|  | ||||
|   void encode(remote::statfs val, bool should_reserve = true); | ||||
|  | ||||
|   void encode(remote::statfs_x val); | ||||
|  | ||||
|   void encode(remote::file_info val); | ||||
|  | ||||
|   void encode_top(const void *buffer, std::size_t size, | ||||
|                   bool should_reserve = true); | ||||
|  | ||||
|   void encode_top(const std::string &str); | ||||
|  | ||||
|   void encode_top(const std::wstring &str); | ||||
|  | ||||
|   void encode_top(void *ptr) { | ||||
|     encode_top( | ||||
|         static_cast<std::uint64_t>(reinterpret_cast<std::uintptr_t>(ptr))); | ||||
|   } | ||||
|  | ||||
|   void encode_top(std::int8_t val); | ||||
|  | ||||
|   void encode_top(std::uint8_t val); | ||||
|  | ||||
|   void encode_top(std::int16_t val); | ||||
|  | ||||
|   void encode_top(std::uint16_t val); | ||||
|  | ||||
|   void encode_top(std::int32_t val); | ||||
|  | ||||
|   void encode_top(std::uint32_t val); | ||||
|  | ||||
|   void encode_top(std::int64_t val); | ||||
|  | ||||
|   void encode_top(std::uint64_t val); | ||||
|  | ||||
|   void encode_top(remote::open_flags val) { | ||||
|     encode_top(static_cast<std::uint32_t>(val)); | ||||
|   } | ||||
|  | ||||
|   void encode_top(remote::setattr_x val); | ||||
|  | ||||
|   void encode_top(remote::stat val); | ||||
|  | ||||
|   void encode_top(remote::statfs val, bool should_reserve = true); | ||||
|  | ||||
|   void encode_top(remote::statfs_x val); | ||||
|  | ||||
|   void encode_top(remote::file_info val); | ||||
|  | ||||
|   void encrypt(const std::string &token); | ||||
|  | ||||
|   [[nodiscard]] auto get_size() const -> std::uint32_t { | ||||
|     return static_cast<std::uint32_t>(buffer_.size()); | ||||
|   } | ||||
|  | ||||
|   void transfer_into(data_buffer &buffer); | ||||
|  | ||||
| public: | ||||
|   auto operator=(const data_buffer &buffer) noexcept -> packet &; | ||||
|  | ||||
|   auto operator=(data_buffer &&buffer) noexcept -> packet &; | ||||
|  | ||||
|   auto operator=(const packet &pkt) noexcept -> packet &; | ||||
|  | ||||
|   auto operator=(packet &&pkt) noexcept -> packet &; | ||||
|  | ||||
|   [[nodiscard]] auto operator[](std::size_t index) -> char & { | ||||
|     return buffer_[index]; | ||||
|   } | ||||
|  | ||||
|   [[nodiscard]] auto operator[](std::size_t index) const -> const char & { | ||||
|     return buffer_.at(index); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| using packet = packet; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_PACKET_HPP_ | ||||
|   | ||||
| @@ -1,91 +1,91 @@ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_PACKET_CLIENT_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_PACKET_CLIENT_HPP_ | ||||
|  | ||||
| #include "comm/packet/packet.hpp" | ||||
|  | ||||
| using boost::asio::ip::tcp; | ||||
|  | ||||
| namespace repertory { | ||||
| class packet_client final { | ||||
| private: | ||||
|   struct client final { | ||||
|     explicit client(boost::asio::io_context &ctx) : nonce(""), socket(ctx) {} | ||||
|  | ||||
|     std::string nonce; | ||||
|     tcp::socket socket; | ||||
|   }; | ||||
|  | ||||
| public: | ||||
|   packet_client(std::string host_name_or_ip, std::uint8_t max_connections, | ||||
|                 std::uint16_t port, std::uint16_t receive_timeout, | ||||
|                 std::uint16_t send_timeout, std::string encryption_token); | ||||
|  | ||||
|   ~packet_client(); | ||||
|  | ||||
| private: | ||||
|   boost::asio::io_context io_context_; | ||||
|   const std::string host_name_or_ip_; | ||||
|   const std::uint8_t max_connections_; | ||||
|   const std::uint16_t port_; | ||||
|   const std::uint16_t receive_timeout_; | ||||
|   const std::uint16_t send_timeout_; | ||||
|   const std::string encryption_token_; | ||||
|   std::string unique_id_; | ||||
|  | ||||
|   bool allow_connections_ = true; | ||||
|   boost::asio::ip::basic_resolver<boost::asio::ip::tcp>::results_type | ||||
|       resolve_results_; | ||||
|   std::mutex clients_mutex_; | ||||
|   std::vector<std::shared_ptr<client>> clients_; | ||||
|  | ||||
| private: | ||||
|   void close(client &c) const; | ||||
|  | ||||
|   void close_all(); | ||||
|  | ||||
|   void connect(client &c); | ||||
|  | ||||
|   [[nodiscard]] auto get_client() -> std::shared_ptr<client>; | ||||
|  | ||||
|   void put_client(std::shared_ptr<client> &c); | ||||
|  | ||||
|   [[nodiscard]] auto read_packet(client &c, packet &response) | ||||
|       -> packet::error_type; | ||||
|  | ||||
|   void resolve(); | ||||
|  | ||||
| public: | ||||
|   [[nodiscard]] auto send(const std::string &method, | ||||
|                           std::uint32_t &service_flags) -> packet::error_type; | ||||
|  | ||||
|   [[nodiscard]] auto send(const std::string &method, packet &request, | ||||
|                           std::uint32_t &service_flags) -> packet::error_type; | ||||
|  | ||||
|   [[nodiscard]] auto send(const std::string &method, packet &request, | ||||
|                           packet &response, std::uint32_t &service_flags) | ||||
|       -> packet::error_type; | ||||
| }; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_PACKET_CLIENT_HPP_ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_PACKET_CLIENT_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_PACKET_CLIENT_HPP_ | ||||
|  | ||||
| #include "comm/packet/packet.hpp" | ||||
|  | ||||
| using boost::asio::ip::tcp; | ||||
|  | ||||
| namespace repertory { | ||||
| class packet_client final { | ||||
| private: | ||||
|   struct client final { | ||||
|     explicit client(boost::asio::io_context &ctx) : nonce(""), socket(ctx) {} | ||||
|  | ||||
|     std::string nonce; | ||||
|     tcp::socket socket; | ||||
|   }; | ||||
|  | ||||
| public: | ||||
|   packet_client(std::string host_name_or_ip, std::uint8_t max_connections, | ||||
|                 std::uint16_t port, std::uint16_t receive_timeout, | ||||
|                 std::uint16_t send_timeout, std::string encryption_token); | ||||
|  | ||||
|   ~packet_client(); | ||||
|  | ||||
| private: | ||||
|   boost::asio::io_context io_context_; | ||||
|   const std::string host_name_or_ip_; | ||||
|   const std::uint8_t max_connections_; | ||||
|   const std::uint16_t port_; | ||||
|   const std::uint16_t receive_timeout_; | ||||
|   const std::uint16_t send_timeout_; | ||||
|   const std::string encryption_token_; | ||||
|   std::string unique_id_; | ||||
|  | ||||
|   bool allow_connections_ = true; | ||||
|   boost::asio::ip::basic_resolver<boost::asio::ip::tcp>::results_type | ||||
|       resolve_results_; | ||||
|   std::mutex clients_mutex_; | ||||
|   std::vector<std::shared_ptr<client>> clients_; | ||||
|  | ||||
| private: | ||||
|   void close(client &c) const; | ||||
|  | ||||
|   void close_all(); | ||||
|  | ||||
|   void connect(client &c); | ||||
|  | ||||
|   [[nodiscard]] auto get_client() -> std::shared_ptr<client>; | ||||
|  | ||||
|   void put_client(std::shared_ptr<client> &c); | ||||
|  | ||||
|   [[nodiscard]] auto read_packet(client &c, packet &response) | ||||
|       -> packet::error_type; | ||||
|  | ||||
|   void resolve(); | ||||
|  | ||||
| public: | ||||
|   [[nodiscard]] auto send(const std::string &method, | ||||
|                           std::uint32_t &service_flags) -> packet::error_type; | ||||
|  | ||||
|   [[nodiscard]] auto send(const std::string &method, packet &request, | ||||
|                           std::uint32_t &service_flags) -> packet::error_type; | ||||
|  | ||||
|   [[nodiscard]] auto send(const std::string &method, packet &request, | ||||
|                           packet &response, std::uint32_t &service_flags) | ||||
|       -> packet::error_type; | ||||
| }; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_PACKET_CLIENT_HPP_ | ||||
|   | ||||
| @@ -1,92 +1,92 @@ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_PACKET_SERVER_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_PACKET_SERVER_HPP_ | ||||
|  | ||||
| #include "comm/packet/client_pool.hpp" | ||||
| #include "types/repertory.hpp" | ||||
| #include "utils/utils.hpp" | ||||
|  | ||||
| using namespace boost::asio; | ||||
| using boost::asio::ip::tcp; | ||||
|  | ||||
| namespace repertory { | ||||
| class packet_server final { | ||||
| public: | ||||
|   using closed_callback = std::function<void(const std::string &)>; | ||||
|   using message_complete_callback = client_pool::worker_complete_callback; | ||||
|   using message_handler_callback = std::function<void( | ||||
|       std::uint32_t, const std::string &, std::uint64_t, const std::string &, | ||||
|       packet *, packet &, message_complete_callback)>; | ||||
|  | ||||
| public: | ||||
|   packet_server(std::uint16_t port, std::string token, std::uint8_t pool_size, | ||||
|                 closed_callback closed, | ||||
|                 message_handler_callback message_handler); | ||||
|  | ||||
|   ~packet_server(); | ||||
|  | ||||
| private: | ||||
|   struct connection { | ||||
|     connection(boost::asio::io_service &io_service, tcp::acceptor &acceptor_) | ||||
|         : socket(io_service), acceptor(acceptor_) {} | ||||
|  | ||||
|     tcp::socket socket; | ||||
|     tcp::acceptor &acceptor; | ||||
|     data_buffer buffer; | ||||
|     std::string client_id; | ||||
|     std::string nonce; | ||||
|  | ||||
|     void generate_nonce() { nonce = utils::generate_random_string(256u); } | ||||
|   }; | ||||
|  | ||||
| private: | ||||
|   const std::string encryption_token_; | ||||
|   closed_callback closed_; | ||||
|   message_handler_callback message_handler_; | ||||
|   boost::asio::io_context io_context_; | ||||
|   std::unique_ptr<std::thread> server_thread_; | ||||
|   std::vector<std::thread> service_threads_; | ||||
|   std::recursive_mutex connection_mutex_; | ||||
|   std::unordered_map<std::string, std::uint32_t> connection_lookup_; | ||||
|  | ||||
| private: | ||||
|   void add_client(connection &c, const std::string &client_id); | ||||
|  | ||||
|   void initialize(const uint16_t &port, uint8_t pool_size); | ||||
|  | ||||
|   void listen_for_connection(tcp::acceptor &acceptor); | ||||
|  | ||||
|   void on_accept(std::shared_ptr<connection> c, boost::system::error_code ec); | ||||
|  | ||||
|   void read_header(std::shared_ptr<connection> c); | ||||
|  | ||||
|   void read_packet(std::shared_ptr<connection> c, std::uint32_t data_size); | ||||
|  | ||||
|   void remove_client(connection &c); | ||||
|  | ||||
|   void send_response(std::shared_ptr<connection> c, | ||||
|                      const packet::error_type &result, packet &response); | ||||
| }; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_PACKET_SERVER_HPP_ | ||||
| /* | ||||
|   Copyright <2018-2023> <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 INCLUDE_COMM_PACKET_PACKET_SERVER_HPP_ | ||||
| #define INCLUDE_COMM_PACKET_PACKET_SERVER_HPP_ | ||||
|  | ||||
| #include "comm/packet/client_pool.hpp" | ||||
| #include "types/repertory.hpp" | ||||
| #include "utils/utils.hpp" | ||||
|  | ||||
| using namespace boost::asio; | ||||
| using boost::asio::ip::tcp; | ||||
|  | ||||
| namespace repertory { | ||||
| class packet_server final { | ||||
| public: | ||||
|   using closed_callback = std::function<void(const std::string &)>; | ||||
|   using message_complete_callback = client_pool::worker_complete_callback; | ||||
|   using message_handler_callback = std::function<void( | ||||
|       std::uint32_t, const std::string &, std::uint64_t, const std::string &, | ||||
|       packet *, packet &, message_complete_callback)>; | ||||
|  | ||||
| public: | ||||
|   packet_server(std::uint16_t port, std::string token, std::uint8_t pool_size, | ||||
|                 closed_callback closed, | ||||
|                 message_handler_callback message_handler); | ||||
|  | ||||
|   ~packet_server(); | ||||
|  | ||||
| private: | ||||
|   struct connection { | ||||
|     connection(boost::asio::io_service &io_service, tcp::acceptor &acceptor_) | ||||
|         : socket(io_service), acceptor(acceptor_) {} | ||||
|  | ||||
|     tcp::socket socket; | ||||
|     tcp::acceptor &acceptor; | ||||
|     data_buffer buffer; | ||||
|     std::string client_id; | ||||
|     std::string nonce; | ||||
|  | ||||
|     void generate_nonce() { nonce = utils::generate_random_string(256u); } | ||||
|   }; | ||||
|  | ||||
| private: | ||||
|   const std::string encryption_token_; | ||||
|   closed_callback closed_; | ||||
|   message_handler_callback message_handler_; | ||||
|   boost::asio::io_context io_context_; | ||||
|   std::unique_ptr<std::thread> server_thread_; | ||||
|   std::vector<std::thread> service_threads_; | ||||
|   std::recursive_mutex connection_mutex_; | ||||
|   std::unordered_map<std::string, std::uint32_t> connection_lookup_; | ||||
|  | ||||
| private: | ||||
|   void add_client(connection &c, const std::string &client_id); | ||||
|  | ||||
|   void initialize(const uint16_t &port, uint8_t pool_size); | ||||
|  | ||||
|   void listen_for_connection(tcp::acceptor &acceptor); | ||||
|  | ||||
|   void on_accept(std::shared_ptr<connection> c, boost::system::error_code ec); | ||||
|  | ||||
|   void read_header(std::shared_ptr<connection> c); | ||||
|  | ||||
|   void read_packet(std::shared_ptr<connection> c, std::uint32_t data_size); | ||||
|  | ||||
|   void remove_client(connection &c); | ||||
|  | ||||
|   void send_response(std::shared_ptr<connection> c, | ||||
|                      const packet::error_type &result, packet &response); | ||||
| }; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // INCLUDE_COMM_PACKET_PACKET_SERVER_HPP_ | ||||
|   | ||||
		Reference in New Issue
	
	Block a user