channel.hpp 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. //
  2. // Copyright (c) 2019-2023 Ruben Perez Hidalgo (rubenperez038 at gmail dot com)
  3. //
  4. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. #ifndef BOOST_MYSQL_DETAIL_CHANNEL_CHANNEL_HPP
  8. #define BOOST_MYSQL_DETAIL_CHANNEL_CHANNEL_HPP
  9. #include <boost/mysql/diagnostics.hpp>
  10. #include <boost/mysql/error_code.hpp>
  11. #include <boost/mysql/field_view.hpp>
  12. #include <boost/mysql/metadata_mode.hpp>
  13. #include <boost/mysql/detail/auxiliar/bytestring.hpp>
  14. #include <boost/mysql/detail/channel/disableable_ssl_stream.hpp>
  15. #include <boost/mysql/detail/channel/message_reader.hpp>
  16. #include <boost/mysql/detail/channel/message_writer.hpp>
  17. #include <boost/mysql/detail/protocol/capabilities.hpp>
  18. #include <boost/mysql/detail/protocol/db_flavor.hpp>
  19. #include <boost/asio/async_result.hpp>
  20. #include <boost/asio/buffer.hpp>
  21. #include <cstddef>
  22. #include <utility>
  23. #include <vector>
  24. namespace boost {
  25. namespace mysql {
  26. namespace detail {
  27. // Implements the message layer of the MySQL protocol
  28. class channel_base
  29. {
  30. db_flavor flavor_{db_flavor::mysql};
  31. capabilities current_caps_;
  32. std::uint8_t shared_sequence_number_{}; // for async ops
  33. bytestring shared_buff_; // for async ops
  34. diagnostics shared_diag_; // for async ops
  35. std::vector<field_view> shared_fields_; // for read_some ops
  36. metadata_mode meta_mode_{metadata_mode::minimal};
  37. protected:
  38. message_reader reader_;
  39. message_writer writer_;
  40. public:
  41. channel_base(std::size_t read_buffer_size) : reader_(read_buffer_size) {}
  42. // Reading
  43. const std::uint8_t* buffer_first() const noexcept { return reader_.buffer_first(); }
  44. bool has_read_messages() const noexcept { return reader_.has_message(); }
  45. boost::asio::const_buffer next_read_message(std::uint8_t& seqnum, error_code& err) noexcept
  46. {
  47. return reader_.get_next_message(seqnum, err);
  48. }
  49. // Exposed for the sake of testing
  50. std::size_t read_buffer_size() const noexcept { return reader_.buffer().size(); }
  51. // Capabilities
  52. capabilities current_capabilities() const noexcept { return current_caps_; }
  53. void set_current_capabilities(capabilities value) noexcept { current_caps_ = value; }
  54. // DB flavor
  55. db_flavor flavor() const noexcept { return flavor_; }
  56. void set_flavor(db_flavor v) noexcept { flavor_ = v; }
  57. void reset()
  58. {
  59. flavor_ = db_flavor::mysql;
  60. current_caps_ = capabilities();
  61. reset_sequence_number();
  62. // Metadata mode does not get reset on handshake
  63. }
  64. // Internal buffer, diagnostics and sequence_number to help async ops
  65. const bytestring& shared_buffer() const noexcept { return shared_buff_; }
  66. bytestring& shared_buffer() noexcept { return shared_buff_; }
  67. diagnostics& shared_diag() noexcept { return shared_diag_; }
  68. std::uint8_t& shared_sequence_number() noexcept { return shared_sequence_number_; }
  69. std::uint8_t& reset_sequence_number() noexcept { return shared_sequence_number_ = 0; }
  70. std::vector<field_view>& shared_fields() noexcept { return shared_fields_; }
  71. const std::vector<field_view>& shared_fields() const noexcept { return shared_fields_; }
  72. // Metadata mode
  73. metadata_mode meta_mode() const noexcept { return meta_mode_; }
  74. void set_meta_mode(metadata_mode v) noexcept { meta_mode_ = v; }
  75. };
  76. template <class Stream>
  77. class channel : public channel_base
  78. {
  79. disableable_ssl_stream<Stream> stream_;
  80. public:
  81. template <class... Args>
  82. channel(std::size_t read_buffer_size, Args&&... args)
  83. : channel_base(read_buffer_size), stream_(std::forward<Args>(args)...)
  84. {
  85. }
  86. // Executor
  87. using executor_type = typename Stream::executor_type;
  88. executor_type get_executor() { return stream_.get_executor(); }
  89. // Reading
  90. void read_some(error_code& code, bool keep_messages = false)
  91. {
  92. return reader_.read_some(stream_, code, keep_messages);
  93. }
  94. template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
  95. BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, void(error_code))
  96. async_read_some(CompletionToken&& token, bool keep_messages = false)
  97. {
  98. return reader_.async_read_some(stream_, std::forward<CompletionToken>(token), keep_messages);
  99. }
  100. boost::asio::const_buffer read_one(std::uint8_t& seqnum, error_code& ec, bool keep_messages = false)
  101. {
  102. return reader_.read_one(stream_, seqnum, ec, keep_messages);
  103. }
  104. template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code, ::boost::asio::const_buffer)) CompletionToken>
  105. BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, void(error_code, ::boost::asio::const_buffer))
  106. async_read_one(std::uint8_t& seqnum, CompletionToken&& token, bool keep_messages = false)
  107. {
  108. return reader_.async_read_one(stream_, seqnum, std::forward<CompletionToken>(token), keep_messages);
  109. }
  110. // Writing
  111. void write(boost::asio::const_buffer buffer, std::uint8_t& seqnum, error_code& code)
  112. {
  113. writer_.write(stream_, buffer, seqnum, code);
  114. }
  115. template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
  116. BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, void(error_code))
  117. async_write(boost::asio::const_buffer buffer, std::uint8_t& seqnum, CompletionToken&& token)
  118. {
  119. return writer_.async_write(stream_, buffer, seqnum, std::forward<CompletionToken>(token));
  120. }
  121. void write(const bytestring& buffer, std::uint8_t& seqnum, error_code& code)
  122. {
  123. write(boost::asio::buffer(buffer), seqnum, code);
  124. }
  125. template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
  126. BOOST_ASIO_INITFN_AUTO_RESULT_TYPE(CompletionToken, void(error_code))
  127. async_write(const bytestring& buffer, std::uint8_t& seqnum, CompletionToken&& token)
  128. {
  129. return async_write(boost::asio::buffer(buffer), seqnum, std::forward<CompletionToken>(token));
  130. }
  131. // SSL
  132. bool ssl_active() const noexcept { return stream_.ssl_active(); }
  133. void set_ssl_active(bool v) noexcept { stream_.set_ssl_active(v); }
  134. // Closing (only available for sockets)
  135. error_code close()
  136. {
  137. error_code err;
  138. lowest_layer().shutdown(lowest_layer_type::shutdown_both, err);
  139. lowest_layer().close(err);
  140. return err;
  141. }
  142. // Getting the underlying stream
  143. using stream_type = disableable_ssl_stream<Stream>;
  144. stream_type& stream() noexcept { return stream_; }
  145. const stream_type& stream() const noexcept { return stream_; }
  146. using lowest_layer_type = typename stream_type::lowest_layer_type;
  147. lowest_layer_type& lowest_layer() noexcept { return stream_.lowest_layer(); }
  148. void reset()
  149. {
  150. channel_base::reset();
  151. stream_.reset();
  152. }
  153. };
  154. } // namespace detail
  155. } // namespace mysql
  156. } // namespace boost
  157. #endif