1 #pragma once 2 3 #include "logging.hpp" 4 5 #include <boost/asio/buffer.hpp> 6 #include <boost/asio/connect_pipe.hpp> 7 #include <boost/asio/io_context.hpp> 8 #include <boost/asio/readable_pipe.hpp> 9 #include <boost/asio/writable_pipe.hpp> 10 #include <boost/asio/write.hpp> 11 12 #include <array> 13 #include <string> 14 15 // Wrapper for boost::async_pipe ensuring proper pipe cleanup 16 class CredentialsPipe 17 { 18 public: 19 explicit CredentialsPipe(boost::asio::io_context& io) : impl(io), read(io) 20 { 21 boost::system::error_code ec; 22 boost::asio::connect_pipe(read, impl, ec); 23 if (ec) 24 { 25 BMCWEB_LOG_CRITICAL("Failed to connect pipe {}", ec.what()); 26 } 27 } 28 29 CredentialsPipe(const CredentialsPipe&) = delete; 30 CredentialsPipe(CredentialsPipe&&) = delete; 31 CredentialsPipe& operator=(const CredentialsPipe&) = delete; 32 CredentialsPipe& operator=(CredentialsPipe&&) = delete; 33 34 ~CredentialsPipe() 35 { 36 explicit_bzero(user.data(), user.capacity()); 37 explicit_bzero(pass.data(), pass.capacity()); 38 } 39 40 int releaseFd() 41 { 42 return read.release(); 43 } 44 45 template <typename WriteHandler> 46 void asyncWrite(std::string&& username, std::string&& password, 47 WriteHandler&& handler) 48 { 49 user = std::move(username); 50 pass = std::move(password); 51 52 // Add +1 to ensure that the null terminator is included. 53 std::array<boost::asio::const_buffer, 2> buffer{ 54 {{user.data(), user.size() + 1}, {pass.data(), pass.size() + 1}}}; 55 boost::asio::async_write(impl, buffer, 56 std::forward<WriteHandler>(handler)); 57 } 58 59 boost::asio::writable_pipe impl; 60 boost::asio::readable_pipe read; 61 62 private: 63 std::string user; 64 std::string pass; 65 }; 66