Back to home page

EIC code displayed by LXR

 
 

    


Warning, file /include/boost/mysql/any_connection.hpp was not indexed or was modified since last indexation (in which case cross-reference links may be missing, inaccurate or erroneous).

0001 //
0002 // Copyright (c) 2019-2024 Ruben Perez Hidalgo (rubenperez038 at gmail dot com)
0003 //
0004 // Distributed under the Boost Software License, Version 1.0. (See accompanying
0005 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
0006 //
0007 
0008 #ifndef BOOST_MYSQL_ANY_CONNECTION_HPP
0009 #define BOOST_MYSQL_ANY_CONNECTION_HPP
0010 
0011 #include <boost/mysql/any_address.hpp>
0012 #include <boost/mysql/character_set.hpp>
0013 #include <boost/mysql/connect_params.hpp>
0014 #include <boost/mysql/defaults.hpp>
0015 #include <boost/mysql/diagnostics.hpp>
0016 #include <boost/mysql/error_code.hpp>
0017 #include <boost/mysql/execution_state.hpp>
0018 #include <boost/mysql/metadata_mode.hpp>
0019 #include <boost/mysql/rows_view.hpp>
0020 #include <boost/mysql/statement.hpp>
0021 #include <boost/mysql/string_view.hpp>
0022 
0023 #include <boost/mysql/detail/access.hpp>
0024 #include <boost/mysql/detail/algo_params.hpp>
0025 #include <boost/mysql/detail/config.hpp>
0026 #include <boost/mysql/detail/connect_params_helpers.hpp>
0027 #include <boost/mysql/detail/connection_impl.hpp>
0028 #include <boost/mysql/detail/engine.hpp>
0029 #include <boost/mysql/detail/execution_concepts.hpp>
0030 #include <boost/mysql/detail/ssl_fwd.hpp>
0031 #include <boost/mysql/detail/throw_on_error_loc.hpp>
0032 
0033 #include <boost/asio/any_io_executor.hpp>
0034 #include <boost/assert.hpp>
0035 #include <boost/system/result.hpp>
0036 
0037 #include <cstddef>
0038 #include <memory>
0039 #include <type_traits>
0040 #include <utility>
0041 #include <vector>
0042 
0043 namespace boost {
0044 namespace mysql {
0045 
0046 // Forward declarations
0047 template <class... StaticRow>
0048 class static_execution_state;
0049 
0050 class pipeline_request;
0051 class stage_response;
0052 
0053 /**
0054  * \brief (EXPERIMENTAL) Configuration parameters that can be passed to \ref any_connection's constructor.
0055  *
0056  * \par Experimental
0057  * This part of the API is experimental, and may change in successive
0058  * releases without previous notice.
0059  */
0060 struct any_connection_params
0061 {
0062     /**
0063      * \brief An external SSL context containing options to configure TLS.
0064      * \details
0065      * Relevant only for SSL connections (those that result on \ref
0066      * any_connection::uses_ssl returning `true`).
0067      * \n
0068      * If the connection is configured to use TLS, an internal `asio::ssl::stream`
0069      * object will be created. If this member is set to a non-null value,
0070      * this internal object will be initialized using the passed context.
0071      * This is the only way to configure TLS options in `any_connection`.
0072      * \n
0073      * If the connection is configured to use TLS and this member is `nullptr`,
0074      * an internal `asio::ssl::context` object with suitable default options
0075      * will be created.
0076      *
0077      * \par Object lifetimes
0078      * If set to non-null, the pointee object must be kept alive until
0079      * all \ref any_connection objects constructed from `*this` are destroyed.
0080      */
0081     asio::ssl::context* ssl_context{};
0082 
0083     /**
0084      * \brief The initial size of the connection's buffer, in bytes.
0085      * \details A bigger read buffer can increase the number of rows
0086      * returned by \ref any_connection::read_some_rows.
0087      */
0088     std::size_t initial_buffer_size{default_initial_read_buffer_size};
0089 
0090     /**
0091      * \brief The maximum size of the connection's buffer, in bytes (64MB by default).
0092      * \details
0093      * Attempting to read or write a protocol packet bigger than this size
0094      * will fail with a \ref client_errc::max_buffer_size_exceeded error.
0095      * \n
0096      * This effectively means: \n
0097      *   - Each request sent to the server must be smaller than this value.
0098      *   - Each individual row received from the server must be smaller than this value.
0099      *     Note that when using `execute` or `async_execute`, results objects may
0100      *     allocate memory beyond this limit if the total number of rows is high.
0101      * \n
0102      * If you need to send or receive larger packets, you may need to adjust
0103      * your server's <a
0104      * href="https://dev.mysql.com/doc/refman/8.4/en/server-system-variables.html#sysvar_max_allowed_packet">`max_allowed_packet`</a>
0105      * system variable, too.
0106      */
0107     std::size_t max_buffer_size{0x4000000};
0108 };
0109 
0110 /**
0111  * \brief (EXPERIMENTAL) A type-erased connection to a MySQL server.
0112  * \details
0113  * Represents a connection to a MySQL server. Compared to \ref connection, this class:
0114  * \n
0115  * \li Is type-erased. The type of the connection doesn't depend on the transport being used.
0116  *     Supported transports include plaintext TCP, SSL over TCP and UNIX domain sockets.
0117  * \li Is easier to connect, as \ref connect and \ref async_connect handle hostname resolution.
0118  * \li Can always be re-connected after being used or encountering an error.
0119  * \li Doesn't support default completion tokens.
0120  * \n
0121  * Provides a level of performance similar to \ref connection.
0122  * \n
0123  * This is a move-only type.
0124  * \n
0125  * \par Thread safety
0126  * Distinct objects: safe. \n
0127  * Shared objects: unsafe. \n
0128  * This class is <b>not thread-safe</b>: for a single object, if you
0129  * call its member functions concurrently from separate threads, you will get a race condition.
0130  *
0131  * \par Experimental
0132  * This part of the API is experimental, and may change in successive
0133  * releases without previous notice.
0134  */
0135 class any_connection
0136 {
0137     detail::connection_impl impl_;
0138 
0139 #ifndef BOOST_MYSQL_DOXYGEN
0140     friend struct detail::access;
0141 #endif
0142 
0143     BOOST_MYSQL_DECL
0144     static std::unique_ptr<detail::engine> create_engine(asio::any_io_executor ex, asio::ssl::context* ctx);
0145 
0146     // Used by tests
0147     any_connection(
0148         std::size_t initial_buffer_size,
0149         std::size_t max_buffer_size,
0150         std::unique_ptr<detail::engine> eng
0151     )
0152         : impl_(initial_buffer_size, max_buffer_size, std::move(eng))
0153     {
0154     }
0155 
0156 public:
0157     /**
0158      * \brief Constructs a connection object from an executor and an optional set of parameters.
0159      * \details
0160      * The resulting connection has `this->get_executor() == ex`. Any internally required I/O objects
0161      * will be constructed using this executor.
0162      * \n
0163      * You can configure extra parameters, like the SSL context and buffer sizes, by passing
0164      * an \ref any_connection_params object to this constructor.
0165      */
0166     any_connection(boost::asio::any_io_executor ex, any_connection_params params = {})
0167         : any_connection(
0168               params.initial_buffer_size,
0169               params.max_buffer_size,
0170               create_engine(std::move(ex), params.ssl_context)
0171           )
0172     {
0173     }
0174 
0175     /**
0176      * \brief Constructs a connection object from an execution context and an optional set of parameters.
0177      * \details
0178      * The resulting connection has `this->get_executor() == ctx.get_executor()`.
0179      * Any internally required I/O objects will be constructed using this executor.
0180      * \n
0181      * You can configure extra parameters, like the SSL context and buffer sizes, by passing
0182      * an \ref any_connection_params object to this constructor.
0183      * \n
0184      * This function participates in overload resolution only if `ExecutionContext`
0185      * satisfies the `ExecutionContext` requirements imposed by Boost.Asio.
0186      */
0187     template <
0188         class ExecutionContext
0189 #ifndef BOOST_MYSQL_DOXYGEN
0190         ,
0191         class = typename std::enable_if<std::is_convertible<
0192             decltype(std::declval<ExecutionContext&>().get_executor()),
0193             asio::any_io_executor>::value>::type
0194 #endif
0195         >
0196     any_connection(ExecutionContext& ctx, any_connection_params params = {})
0197         : any_connection(ctx.get_executor(), params)
0198     {
0199     }
0200 
0201     /**
0202      * \brief Move constructor.
0203      */
0204     any_connection(any_connection&& other) = default;
0205 
0206     /**
0207      * \brief Move assignment.
0208      */
0209     any_connection& operator=(any_connection&& rhs) = default;
0210 
0211 #ifndef BOOST_MYSQL_DOXYGEN
0212     any_connection(const any_connection&) = delete;
0213     any_connection& operator=(const any_connection&) = delete;
0214 #endif
0215 
0216     /**
0217      * \brief Destructor.
0218      * \details
0219      * Closes the connection at the transport layer (by closing any underlying socket objects).
0220      * If you require a clean close, call \ref close or \ref async_close before the connection
0221      * is destroyed.
0222      */
0223     ~any_connection() = default;
0224 
0225     /// The executor type associated to this object.
0226     using executor_type = asio::any_io_executor;
0227 
0228     /**
0229      * \brief Retrieves the executor associated to this object.
0230      * \par Exception safety
0231      * No-throw guarantee.
0232      */
0233     executor_type get_executor() noexcept { return impl_.get_engine().get_executor(); }
0234 
0235     /**
0236      * \brief Returns whether the connection negotiated the use of SSL or not.
0237      * \details
0238      * This function can be used to determine whether you are using a SSL
0239      * connection or not when using SSL negotiation.
0240      * \n
0241      * This function always returns `false`
0242      * for connections that haven't been established yet. If the connection establishment fails,
0243      * the return value is undefined.
0244      *
0245      * \par Exception safety
0246      * No-throw guarantee.
0247      */
0248     bool uses_ssl() const noexcept { return impl_.ssl_active(); }
0249 
0250     /**
0251      * \brief Returns whether backslashes are being treated as escape sequences.
0252      * \details
0253      * By default, the server treats backslashes in string values as escape characters.
0254      * This behavior can be disabled by activating the <a
0255      *   href="https://dev.mysql.com/doc/refman/8.0/en/sql-mode.html#sqlmode_no_backslash_escapes">`NO_BACKSLASH_ESCAPES`</a>
0256      * SQL mode.
0257      * \n
0258      * Every time an operation involving server communication completes, the server reports whether
0259      * this mode was activated or not as part of the response. Connections store this information
0260      * and make it available through this function.
0261      * \n
0262      * \li If backslash are treated like escape characters, returns `true`.
0263      * \li If `NO_BACKSLASH_ESCAPES` has been activated, returns `false`.
0264      * \li If connection establishment hasn't happened yet, returns `true`.
0265      * \li Calling this function while an async operation that changes backslash behavior
0266      *     is outstanding may return `true` or `false`.
0267      * \n
0268      * This function does not involve server communication.
0269      *
0270      * \par Exception safety
0271      * No-throw guarantee.
0272      */
0273     bool backslash_escapes() const noexcept { return impl_.backslash_escapes(); }
0274 
0275     /**
0276      * \brief Returns the character set used by this connection.
0277      * \details
0278      * Connections attempt to keep track of the current character set.
0279      * Deficiencies in the protocol can cause the character set to be unknown, though.
0280      * When the character set is known, this function returns
0281      * the character set currently in use. Otherwise, returns \ref client_errc::unknown_character_set.
0282      * \n
0283      * The following functions can modify the return value of this function: \n
0284      *   \li Prior to connection, the character set is always unknown.
0285      *   \li \ref connect and \ref async_connect may set the current character set
0286      *       to a known value, depending on the requested collation.
0287      *   \li \ref set_character_set always and \ref async_set_character_set always
0288      *       set the current character set to the passed value.
0289      *   \li \ref reset_connection and \ref async_reset_connection always makes the current character
0290      *       unknown.
0291      *
0292      * \par Avoid changing the character set directly
0293      * If you change the connection's character set directly using SQL statements
0294      * like `"SET NAMES utf8mb4"`, the client has no way to track this change,
0295      * and this function will return incorrect results.
0296      *
0297      * \par Errors
0298      * \li \ref client_errc::unknown_character_set if the current character set is unknown.
0299      *
0300      * \par Exception safety
0301      * No-throw guarantee.
0302      */
0303     system::result<character_set> current_character_set() const noexcept
0304     {
0305         return impl_.current_character_set();
0306     }
0307 
0308     /**
0309      * \brief Returns format options suitable to format SQL according to the current connection configuation.
0310      * \details
0311      * If the current character set is known (as given by \ref current_character_set), returns
0312      * a value suitable to be passed to SQL formatting functions. Otherwise, returns an error.
0313      *
0314      * \par Errors
0315      * \li \ref client_errc::unknown_character_set if the current character set is unknown.
0316      *
0317      * \par Exception safety
0318      * No-throw guarantee.
0319      */
0320     system::result<format_options> format_opts() const noexcept
0321     {
0322         auto res = current_character_set();
0323         if (res.has_error())
0324             return res.error();
0325         return format_options{res.value(), backslash_escapes()};
0326     }
0327 
0328     /// \copydoc connection::meta_mode
0329     metadata_mode meta_mode() const noexcept { return impl_.meta_mode(); }
0330 
0331     /// \copydoc connection::set_meta_mode
0332     void set_meta_mode(metadata_mode v) noexcept { impl_.set_meta_mode(v); }
0333 
0334     /**
0335      * \brief Establishes a connection to a MySQL server.
0336      * \details
0337      * This function performs the following:
0338      * \n
0339      * \li If a connection has already been established (by a previous call to \ref connect
0340      *     or \ref async_connect), closes it at the transport layer (by closing any underlying socket)
0341      *     and discards any protocol state associated to it. (If you require
0342      *     a clean close, call \ref close or \ref async_close before using this function).
0343      * \li If the connection is configured to use TCP (`params.server_address.type() ==
0344      *     address_type::host_and_port`), resolves the passed hostname to a set of endpoints. An empty
0345      *     hostname is equivalent to `"localhost"`.
0346      * \li Establishes the physical connection (performing the
0347      *     TCP or UNIX socket connect).
0348      * \li Performs the MySQL handshake to establish a session. If the
0349      *     connection is configured to use TLS, the TLS handshake is performed as part of this step.
0350      * \li If any of the above steps fail, the TCP or UNIX socket connection is closed.
0351      * \n
0352      * You can configure some options using the \ref connect_params struct.
0353      * \n
0354      * The decision to use TLS or not is performed using the following:
0355      * \n
0356      * \li If the transport is not TCP (`params.server_address.type() != address_type::host_and_port`),
0357      *     the connection will never use TLS.
0358      * \li If the transport is TCP, and `params.ssl == ssl_mode::disable`, the connection will not use TLS.
0359      * \li If the transport is TCP, and `params.ssl == ssl_mode::enable`, the connection will use TLS
0360      *     only if the server supports it.
0361      * \li If the transport is TCP, and `params.ssl == ssl_mode::require`, the connection will always use TLS.
0362      *     If the server doesn't support it, this function will fail with \ref
0363      *     client_errc::server_doesnt_support_ssl.
0364      * \n
0365      * If `params.connection_collation` is within a set of well-known collations, this function
0366      * sets the current character set, such that \ref current_character_set returns a non-null value.
0367      * The default collation (`utf8mb4_general_ci`) is the only one guaranteed to be in the set of well-known
0368      * collations.
0369      */
0370     void connect(const connect_params& params, error_code& ec, diagnostics& diag)
0371     {
0372         impl_.connect_v2(params, ec, diag);
0373     }
0374 
0375     /// \copydoc connect
0376     void connect(const connect_params& params)
0377     {
0378         error_code err;
0379         diagnostics diag;
0380         connect(params, err, diag);
0381         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0382     }
0383 
0384     /**
0385      * \copydoc connect
0386      *
0387      * \par Object lifetimes
0388      * params needs to be kept alive until the operation completes, as no
0389      * copies will be made by the library.
0390      *
0391      * \par Handler signature
0392      * The handler signature for this operation is `void(boost::mysql::error_code)`.
0393      */
0394     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
0395     auto async_connect(const connect_params& params, diagnostics& diag, CompletionToken&& token)
0396         BOOST_MYSQL_RETURN_TYPE(detail::async_connect_v2_t<CompletionToken&&>)
0397     {
0398         return impl_.async_connect_v2(params, diag, std::forward<CompletionToken>(token));
0399     }
0400 
0401     /// \copydoc async_connect
0402     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
0403     auto async_connect(const connect_params& params, CompletionToken&& token)
0404         BOOST_MYSQL_RETURN_TYPE(detail::async_connect_v2_t<CompletionToken&&>)
0405     {
0406         return async_connect(params, impl_.shared_diag(), std::forward<CompletionToken>(token));
0407     }
0408 
0409     /// \copydoc connection::execute
0410     template <BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest, BOOST_MYSQL_RESULTS_TYPE ResultsType>
0411     void execute(const ExecutionRequest& req, ResultsType& result, error_code& err, diagnostics& diag)
0412     {
0413         impl_.execute(req, result, err, diag);
0414     }
0415 
0416     /// \copydoc execute
0417     template <BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest, BOOST_MYSQL_RESULTS_TYPE ResultsType>
0418     void execute(const ExecutionRequest& req, ResultsType& result)
0419     {
0420         error_code err;
0421         diagnostics diag;
0422         execute(req, result, err, diag);
0423         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0424     }
0425 
0426     /// \copydoc connection::async_execute
0427     template <
0428         BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest,
0429         BOOST_MYSQL_RESULTS_TYPE ResultsType,
0430         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0431     auto async_execute(ExecutionRequest&& req, ResultsType& result, CompletionToken&& token)
0432         BOOST_MYSQL_RETURN_TYPE(detail::async_execute_t<ExecutionRequest&&, ResultsType, CompletionToken&&>)
0433     {
0434         return async_execute(
0435             std::forward<ExecutionRequest>(req),
0436             result,
0437             impl_.shared_diag(),
0438             std::forward<CompletionToken>(token)
0439         );
0440     }
0441 
0442     /// \copydoc async_execute
0443     template <
0444         BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest,
0445         BOOST_MYSQL_RESULTS_TYPE ResultsType,
0446         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0447     auto async_execute(
0448         ExecutionRequest&& req,
0449         ResultsType& result,
0450         diagnostics& diag,
0451         CompletionToken&& token
0452     ) BOOST_MYSQL_RETURN_TYPE(detail::async_execute_t<ExecutionRequest&&, ResultsType, CompletionToken&&>)
0453     {
0454         return impl_.async_execute(
0455             std::forward<ExecutionRequest>(req),
0456             result,
0457             diag,
0458             std::forward<CompletionToken>(token)
0459         );
0460     }
0461 
0462     /// \copydoc connection::start_execution
0463     template <
0464         BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest,
0465         BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType>
0466     void start_execution(
0467         const ExecutionRequest& req,
0468         ExecutionStateType& st,
0469         error_code& err,
0470         diagnostics& diag
0471     )
0472     {
0473         impl_.start_execution(req, st, err, diag);
0474     }
0475 
0476     /// \copydoc start_execution
0477     template <
0478         BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest,
0479         BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType>
0480     void start_execution(const ExecutionRequest& req, ExecutionStateType& st)
0481     {
0482         error_code err;
0483         diagnostics diag;
0484         start_execution(req, st, err, diag);
0485         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0486     }
0487 
0488     /// \copydoc connection::async_start_execution
0489     template <
0490         BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest,
0491         BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType,
0492         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0493     auto async_start_execution(ExecutionRequest&& req, ExecutionStateType& st, CompletionToken&& token)
0494         BOOST_MYSQL_RETURN_TYPE(detail::async_start_execution_t<
0495                                 ExecutionRequest&&,
0496                                 ExecutionStateType,
0497                                 CompletionToken&&>)
0498     {
0499         return async_start_execution(
0500             std::forward<ExecutionRequest>(req),
0501             st,
0502             impl_.shared_diag(),
0503             std::forward<CompletionToken>(token)
0504         );
0505     }
0506 
0507     /// \copydoc async_start_execution
0508     template <
0509         BOOST_MYSQL_EXECUTION_REQUEST ExecutionRequest,
0510         BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType,
0511         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0512     auto async_start_execution(
0513         ExecutionRequest&& req,
0514         ExecutionStateType& st,
0515         diagnostics& diag,
0516         CompletionToken&& token
0517     )
0518         BOOST_MYSQL_RETURN_TYPE(detail::async_start_execution_t<
0519                                 ExecutionRequest&&,
0520                                 ExecutionStateType,
0521                                 CompletionToken&&>)
0522     {
0523         return impl_.async_start_execution(
0524             std::forward<ExecutionRequest>(req),
0525             st,
0526             diag,
0527             std::forward<CompletionToken>(token)
0528         );
0529     }
0530 
0531     /// \copydoc connection::prepare_statement
0532     statement prepare_statement(string_view stmt, error_code& err, diagnostics& diag)
0533     {
0534         return impl_.run(detail::prepare_statement_algo_params{&diag, stmt}, err);
0535     }
0536 
0537     /// \copydoc prepare_statement
0538     statement prepare_statement(string_view stmt)
0539     {
0540         error_code err;
0541         diagnostics diag;
0542         statement res = prepare_statement(stmt, err, diag);
0543         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0544         return res;
0545     }
0546 
0547     /// \copydoc connection::async_prepare_statement
0548     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code, ::boost::mysql::statement))
0549                   CompletionToken>
0550     auto async_prepare_statement(string_view stmt, CompletionToken&& token)
0551         BOOST_MYSQL_RETURN_TYPE(detail::async_prepare_statement_t<CompletionToken&&>)
0552     {
0553         return async_prepare_statement(stmt, impl_.shared_diag(), std::forward<CompletionToken>(token));
0554     }
0555 
0556     /// \copydoc async_prepare_statement
0557     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code, ::boost::mysql::statement))
0558                   CompletionToken>
0559     auto async_prepare_statement(string_view stmt, diagnostics& diag, CompletionToken&& token)
0560         BOOST_MYSQL_RETURN_TYPE(detail::async_prepare_statement_t<CompletionToken&&>)
0561     {
0562         return impl_.async_run(
0563             detail::prepare_statement_algo_params{&diag, stmt},
0564             std::forward<CompletionToken>(token)
0565         );
0566     }
0567 
0568     /// \copydoc connection::close_statement
0569     void close_statement(const statement& stmt, error_code& err, diagnostics& diag)
0570     {
0571         impl_.run(impl_.make_params_close_statement(stmt, diag), err);
0572     }
0573 
0574     /// \copydoc close_statement
0575     void close_statement(const statement& stmt)
0576     {
0577         error_code err;
0578         diagnostics diag;
0579         close_statement(stmt, err, diag);
0580         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0581     }
0582 
0583     /// \copydoc connection::async_close_statement
0584     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0585     auto async_close_statement(const statement& stmt, CompletionToken&& token)
0586         BOOST_MYSQL_RETURN_TYPE(detail::async_close_statement_t<CompletionToken&&>)
0587     {
0588         return async_close_statement(stmt, impl_.shared_diag(), std::forward<CompletionToken>(token));
0589     }
0590 
0591     /// \copydoc async_close_statement
0592     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0593     auto async_close_statement(const statement& stmt, diagnostics& diag, CompletionToken&& token)
0594         BOOST_MYSQL_RETURN_TYPE(detail::async_close_statement_t<CompletionToken&&>)
0595     {
0596         return impl_.async_run(
0597             impl_.make_params_close_statement(stmt, diag),
0598             std::forward<CompletionToken>(token)
0599         );
0600     }
0601 
0602     /// \copydoc connection::read_some_rows
0603     rows_view read_some_rows(execution_state& st, error_code& err, diagnostics& diag)
0604     {
0605         return impl_.run(impl_.make_params_read_some_rows(st, diag), err);
0606     }
0607 
0608     /// \copydoc read_some_rows(execution_state&,error_code&,diagnostics&)
0609     rows_view read_some_rows(execution_state& st)
0610     {
0611         error_code err;
0612         diagnostics diag;
0613         rows_view res = read_some_rows(st, err, diag);
0614         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0615         return res;
0616     }
0617 
0618     /// \copydoc connection::async_read_some_rows(execution_state&,CompletionToken&&)
0619     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code, ::boost::mysql::rows_view))
0620                   CompletionToken>
0621     auto async_read_some_rows(execution_state& st, CompletionToken&& token)
0622         BOOST_MYSQL_RETURN_TYPE(detail::async_read_some_rows_dynamic_t<CompletionToken&&>)
0623     {
0624         return async_read_some_rows(st, impl_.shared_diag(), std::forward<CompletionToken>(token));
0625     }
0626 
0627     /// \copydoc async_read_some_rows(execution_state&,CompletionToken&&)
0628     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code, ::boost::mysql::rows_view))
0629                   CompletionToken>
0630     auto async_read_some_rows(execution_state& st, diagnostics& diag, CompletionToken&& token)
0631         BOOST_MYSQL_RETURN_TYPE(detail::async_read_some_rows_dynamic_t<CompletionToken&&>)
0632     {
0633         return impl_.async_run(
0634             impl_.make_params_read_some_rows(st, diag),
0635             std::forward<CompletionToken>(token)
0636         );
0637     }
0638 
0639 #ifdef BOOST_MYSQL_CXX14
0640 
0641     /**
0642      * \brief Reads a batch of rows.
0643      * \details
0644      * Reads a batch of rows of unspecified size into the storage given by `output`.
0645      * At most `output.size()` rows will be read. If the operation represented by `st`
0646      * has still rows to read, and `output.size() > 0`, at least one row will be read.
0647      * \n
0648      * Returns the number of read rows.
0649      * \n
0650      * If there are no more rows, or `st.should_read_rows() == false`, this function is a no-op and returns
0651      * zero.
0652      * \n
0653      * The number of rows that will be read depends on the connection's buffer size. The bigger the buffer,
0654      * the greater the batch size (up to a maximum). You can set the initial buffer size in the
0655      * constructor. The buffer may be grown bigger by other read operations, if required.
0656      * \n
0657      * Rows read by this function are owning objects, and don't hold any reference to
0658      * the connection's internal buffers (contrary what happens with the dynamic interface's counterpart).
0659      * \n
0660      * The type `SpanElementType` must be the underlying row type for one of the types in the
0661      * `StaticRow` parameter pack (i.e., one of the types in `underlying_row_t<StaticRow>...`).
0662      * The type must match the resultset that is currently being processed by `st`. For instance,
0663      * given `static_execution_state<T1, T2>`, when reading rows for the second resultset, `SpanElementType`
0664      * must exactly be `underlying_row_t<T2>`. If this is not the case, a runtime error will be issued.
0665      * \n
0666      * This function can report schema mismatches.
0667      */
0668     template <class SpanElementType, class... StaticRow>
0669     std::size_t read_some_rows(
0670         static_execution_state<StaticRow...>& st,
0671         span<SpanElementType> output,
0672         error_code& err,
0673         diagnostics& diag
0674     )
0675     {
0676         return impl_.run(impl_.make_params_read_some_rows_static(st, output, diag), err);
0677     }
0678 
0679     /**
0680      * \brief Reads a batch of rows.
0681      * \details
0682      * Reads a batch of rows of unspecified size into the storage given by `output`.
0683      * At most `output.size()` rows will be read. If the operation represented by `st`
0684      * has still rows to read, and `output.size() > 0`, at least one row will be read.
0685      * \n
0686      * Returns the number of read rows.
0687      * \n
0688      * If there are no more rows, or `st.should_read_rows() == false`, this function is a no-op and returns
0689      * zero.
0690      * \n
0691      * The number of rows that will be read depends on the connection's buffer size. The bigger the buffer,
0692      * the greater the batch size (up to a maximum). You can set the initial buffer size in the
0693      * constructor. The buffer may be grown bigger by other read operations, if required.
0694      * \n
0695      * Rows read by this function are owning objects, and don't hold any reference to
0696      * the connection's internal buffers (contrary what happens with the dynamic interface's counterpart).
0697      * \n
0698      * The type `SpanElementType` must be the underlying row type for one of the types in the
0699      * `StaticRow` parameter pack (i.e., one of the types in `underlying_row_t<StaticRow>...`).
0700      * The type must match the resultset that is currently being processed by `st`. For instance,
0701      * given `static_execution_state<T1, T2>`, when reading rows for the second resultset, `SpanElementType`
0702      * must exactly be `underlying_row_t<T2>`. If this is not the case, a runtime error will be issued.
0703      * \n
0704      * This function can report schema mismatches.
0705      */
0706     template <class SpanElementType, class... StaticRow>
0707     std::size_t read_some_rows(static_execution_state<StaticRow...>& st, span<SpanElementType> output)
0708     {
0709         error_code err;
0710         diagnostics diag;
0711         std::size_t res = read_some_rows(st, output, err, diag);
0712         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0713         return res;
0714     }
0715 
0716     /**
0717      * \brief Reads a batch of rows.
0718      * \details
0719      * Reads a batch of rows of unspecified size into the storage given by `output`.
0720      * At most `output.size()` rows will be read. If the operation represented by `st`
0721      * has still rows to read, and `output.size() > 0`, at least one row will be read.
0722      * \n
0723      * Returns the number of read rows.
0724      * \n
0725      * If there are no more rows, or `st.should_read_rows() == false`, this function is a no-op and returns
0726      * zero.
0727      * \n
0728      * The number of rows that will be read depends on the connection's buffer size. The bigger the buffer,
0729      * the greater the batch size (up to a maximum). You can set the initial buffer size in the
0730      * constructor. The buffer may be grown bigger by other read operations, if required.
0731      * \n
0732      * Rows read by this function are owning objects, and don't hold any reference to
0733      * the connection's internal buffers (contrary what happens with the dynamic interface's counterpart).
0734      * \n
0735      * The type `SpanElementType` must be the underlying row type for one of the types in the
0736      * `StaticRow` parameter pack (i.e., one of the types in `underlying_row_t<StaticRow>...`).
0737      * The type must match the resultset that is currently being processed by `st`. For instance,
0738      * given `static_execution_state<T1, T2>`, when reading rows for the second resultset, `SpanElementType`
0739      * must exactly be `underlying_row_t<T2>`. If this is not the case, a runtime error will be issued.
0740      * \n
0741      * This function can report schema mismatches.
0742      *
0743      * \par Handler signature
0744      * The handler signature for this operation is
0745      * `void(boost::mysql::error_code, std::size_t)`.
0746      *
0747      * \par Object lifetimes
0748      * The storage that `output` references must be kept alive until the operation completes.
0749      */
0750     template <
0751         class SpanElementType,
0752         class... StaticRow,
0753         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code, std::size_t))
0754             CompletionToken BOOST_ASIO_DEFAULT_COMPLETION_TOKEN_TYPE(executor_type)>
0755     auto async_read_some_rows(
0756         static_execution_state<StaticRow...>& st,
0757         span<SpanElementType> output,
0758         CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN(executor_type)
0759     )
0760     {
0761         return async_read_some_rows(st, output, impl_.shared_diag(), std::forward<CompletionToken>(token));
0762     }
0763 
0764     /**
0765      * \brief Reads a batch of rows.
0766      * \details
0767      * Reads a batch of rows of unspecified size into the storage given by `output`.
0768      * At most `output.size()` rows will be read. If the operation represented by `st`
0769      * has still rows to read, and `output.size() > 0`, at least one row will be read.
0770      * \n
0771      * Returns the number of read rows.
0772      * \n
0773      * If there are no more rows, or `st.should_read_rows() == false`, this function is a no-op and returns
0774      * zero.
0775      * \n
0776      * The number of rows that will be read depends on the connection's buffer size. The bigger the buffer,
0777      * the greater the batch size (up to a maximum). You can set the initial buffer size in the
0778      * constructor. The buffer may be grown bigger by other read operations, if required.
0779      * \n
0780      * Rows read by this function are owning objects, and don't hold any reference to
0781      * the connection's internal buffers (contrary what happens with the dynamic interface's counterpart).
0782      * \n
0783      * The type `SpanElementType` must be the underlying row type for one of the types in the
0784      * `StaticRow` parameter pack (i.e., one of the types in `underlying_row_t<StaticRow>...`).
0785      * The type must match the resultset that is currently being processed by `st`. For instance,
0786      * given `static_execution_state<T1, T2>`, when reading rows for the second resultset, `SpanElementType`
0787      * must exactly be `underlying_row_t<T2>`. If this is not the case, a runtime error will be issued.
0788      * \n
0789      * This function can report schema mismatches.
0790      *
0791      * \par Handler signature
0792      * The handler signature for this operation is
0793      * `void(boost::mysql::error_code, std::size_t)`.
0794      *
0795      * \par Object lifetimes
0796      * The storage that `output` references must be kept alive until the operation completes.
0797      */
0798     template <
0799         class SpanElementType,
0800         class... StaticRow,
0801         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code, std::size_t))
0802             CompletionToken BOOST_ASIO_DEFAULT_COMPLETION_TOKEN_TYPE(executor_type)>
0803     auto async_read_some_rows(
0804         static_execution_state<StaticRow...>& st,
0805         span<SpanElementType> output,
0806         diagnostics& diag,
0807         CompletionToken&& token BOOST_ASIO_DEFAULT_COMPLETION_TOKEN(executor_type)
0808     )
0809     {
0810         return impl_.async_run(
0811             impl_.make_params_read_some_rows_static(st, output, diag),
0812             std::forward<CompletionToken>(token)
0813         );
0814     }
0815 #endif
0816 
0817     /// \copydoc connection::read_resultset_head
0818     template <BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType>
0819     void read_resultset_head(ExecutionStateType& st, error_code& err, diagnostics& diag)
0820     {
0821         return impl_.run(impl_.make_params_read_resultset_head(st, diag), err);
0822     }
0823 
0824     /// \copydoc read_resultset_head
0825     template <BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType>
0826     void read_resultset_head(ExecutionStateType& st)
0827     {
0828         error_code err;
0829         diagnostics diag;
0830         read_resultset_head(st, err, diag);
0831         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0832     }
0833 
0834     /// \copydoc connection::async_read_resultset_head
0835     template <
0836         BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType,
0837         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0838     auto async_read_resultset_head(ExecutionStateType& st, CompletionToken&& token)
0839         BOOST_MYSQL_RETURN_TYPE(detail::async_read_resultset_head_t<CompletionToken&&>)
0840     {
0841         return async_read_resultset_head(st, impl_.shared_diag(), std::forward<CompletionToken>(token));
0842     }
0843 
0844     /// \copydoc async_read_resultset_head
0845     template <
0846         BOOST_MYSQL_EXECUTION_STATE_TYPE ExecutionStateType,
0847         BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0848     auto async_read_resultset_head(ExecutionStateType& st, diagnostics& diag, CompletionToken&& token)
0849         BOOST_MYSQL_RETURN_TYPE(detail::async_read_resultset_head_t<CompletionToken&&>)
0850     {
0851         return impl_.async_run(
0852             impl_.make_params_read_resultset_head(st, diag),
0853             std::forward<CompletionToken>(token)
0854         );
0855     }
0856 
0857     /**
0858      * \brief Sets the connection's character set, as per SET NAMES.
0859      * \details
0860      * Sets the connection's character set by running a
0861      * <a href="https://dev.mysql.com/doc/refman/8.0/en/set-names.html">`SET NAMES`</a>
0862      * SQL statement, using the passed \ref character_set::name as the charset name to set.
0863      * \n
0864      * This function will also update the value returned by \ref current_character_set, so
0865      * prefer using this function over raw SQL statements.
0866      * \n
0867      * If the server was unable to set the character set to the requested value (e.g. because
0868      * the server does not support the requested charset), this function will fail,
0869      * as opposed to how \ref connect behaves when an unsupported collation is passed.
0870      * This is a limitation of MySQL servers.
0871      * \n
0872      * You need to perform connection establishment for this function to succeed, since it
0873      * involves communicating with the server.
0874      *
0875      * \par Object lifetimes
0876      * `charset` will be copied as required, and does not need to be kept alive.
0877      */
0878     void set_character_set(const character_set& charset, error_code& err, diagnostics& diag)
0879     {
0880         impl_.run(impl_.make_params_set_character_set(charset, diag), err);
0881     }
0882 
0883     /// \copydoc set_character_set
0884     void set_character_set(const character_set& charset)
0885     {
0886         error_code err;
0887         diagnostics diag;
0888         set_character_set(charset, err, diag);
0889         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0890     }
0891 
0892     /**
0893      * \copydoc set_character_set
0894      * \details
0895      * \n
0896      * \par Handler signature
0897      * The handler signature for this operation is `void(boost::mysql::error_code)`.
0898      */
0899     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0900     auto async_set_character_set(const character_set& charset, CompletionToken&& token)
0901         BOOST_MYSQL_RETURN_TYPE(detail::async_set_character_set_t<CompletionToken&&>)
0902     {
0903         return async_set_character_set(charset, impl_.shared_diag(), std::forward<CompletionToken>(token));
0904     }
0905 
0906     /// \copydoc async_set_character_set
0907     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0908     auto async_set_character_set(const character_set& charset, diagnostics& diag, CompletionToken&& token)
0909         BOOST_MYSQL_RETURN_TYPE(detail::async_set_character_set_t<CompletionToken&&>)
0910     {
0911         return impl_.async_run(
0912             impl_.make_params_set_character_set(charset, diag),
0913             std::forward<CompletionToken>(token)
0914         );
0915     }
0916 
0917     /// \copydoc connection::ping
0918     void ping(error_code& err, diagnostics& diag) { impl_.run(impl_.make_params_ping(diag), err); }
0919 
0920     /// \copydoc ping
0921     void ping()
0922     {
0923         error_code err;
0924         diagnostics diag;
0925         ping(err, diag);
0926         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0927     }
0928 
0929     /// \copydoc connection::async_ping
0930     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0931     auto async_ping(CompletionToken&& token) BOOST_MYSQL_RETURN_TYPE(detail::async_ping_t<CompletionToken&&>)
0932     {
0933         return async_ping(impl_.shared_diag(), std::forward<CompletionToken>(token));
0934     }
0935 
0936     /// \copydoc async_ping
0937     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
0938     auto async_ping(diagnostics& diag, CompletionToken&& token)
0939         BOOST_MYSQL_RETURN_TYPE(detail::async_ping_t<CompletionToken&&>)
0940     {
0941         return impl_.async_run(impl_.make_params_ping(diag), std::forward<CompletionToken>(token));
0942     }
0943 
0944     /**
0945      * \brief Resets server-side session state, like variables and prepared statements.
0946      * \details
0947      * Resets all server-side state for the current session:
0948      * \n
0949      *   \li Rolls back any active transactions and resets autocommit mode.
0950      *   \li Releases all table locks.
0951      *   \li Drops all temporary tables.
0952      *   \li Resets all session system variables to their default values (including the ones set by `SET
0953      *       NAMES`) and clears all user-defined variables.
0954      *   \li Closes all prepared statements.
0955      * \n
0956      * A full reference on the affected session state can be found
0957      * <a href="https://dev.mysql.com/doc/c-api/8.0/en/mysql-reset-connection.html">here</a>.
0958      * \n
0959      * \n
0960      * This function will not reset the current physical connection and won't cause re-authentication.
0961      * It is faster than closing and re-opening a connection.
0962      * \n
0963      * The connection must be connected and authenticated before calling this function.
0964      * This function involves communication with the server, and thus may fail.
0965      *
0966      * \par Warning on character sets
0967      * This function will restore the connection's character set and collation **to the server's default**,
0968      * and not to the one specified during connection establishment. Some servers have `latin1` as their
0969      * default character set, which is not usually what you want. Since there is no way to know this
0970      * character set, \ref current_character_set will return `nullptr` after the operation succeeds.
0971      * We recommend always using \ref set_character_set or \ref async_set_character_set after calling this
0972      * function.
0973      * \n
0974      * You can find the character set that your server will use after the reset by running:
0975      * \code
0976      * "SELECT @@global.character_set_client, @@global.character_set_results;"
0977      * \endcode
0978      */
0979     void reset_connection(error_code& err, diagnostics& diag)
0980     {
0981         impl_.run(impl_.make_params_reset_connection(diag), err);
0982     }
0983 
0984     /// \copydoc reset_connection
0985     void reset_connection()
0986     {
0987         error_code err;
0988         diagnostics diag;
0989         reset_connection(err, diag);
0990         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
0991     }
0992 
0993     /**
0994      * \copydoc reset_connection
0995      * \details
0996      * \n
0997      * \par Handler signature
0998      * The handler signature for this operation is `void(boost::mysql::error_code)`.
0999      */
1000     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
1001     auto async_reset_connection(CompletionToken&& token)
1002         BOOST_MYSQL_RETURN_TYPE(detail::async_reset_connection_t<CompletionToken&&>)
1003     {
1004         return async_reset_connection(impl_.shared_diag(), std::forward<CompletionToken>(token));
1005     }
1006 
1007     /// \copydoc async_reset_connection
1008     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(::boost::mysql::error_code)) CompletionToken>
1009     auto async_reset_connection(diagnostics& diag, CompletionToken&& token)
1010         BOOST_MYSQL_RETURN_TYPE(detail::async_reset_connection_t<CompletionToken&&>)
1011     {
1012         return impl_.async_run(
1013             impl_.make_params_reset_connection(diag),
1014             std::forward<CompletionToken>(token)
1015         );
1016     }
1017 
1018     /**
1019      * \brief Cleanly closes the connection to the server.
1020      * \details
1021      * This function does the following:
1022      * \n
1023      * \li Sends a quit request. This is required by the MySQL protocol, to inform
1024      *     the server that we're closing the connection gracefully.
1025      * \li If the connection is using TLS (`this->uses_ssl() == true`), performs
1026      *     the TLS shutdown.
1027      * \li Closes the transport-level connection (the TCP or UNIX socket).
1028      * \n
1029      * Since this function involves writing a message to the server, it can fail.
1030      * Only use this function if you know that the connection is healthy and you want
1031      * to cleanly close it.
1032      * \n
1033      * If you don't call this function, the destructor or successive connects will
1034      * perform a transport-layer close. This doesn't cause any resource leaks, but may
1035      * cause warnings to be written to the server logs.
1036      */
1037     void close(error_code& err, diagnostics& diag)
1038     {
1039         this->impl_.run(this->impl_.make_params_close(diag), err);
1040     }
1041 
1042     /// \copydoc close
1043     void close()
1044     {
1045         error_code err;
1046         diagnostics diag;
1047         close(err, diag);
1048         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
1049     }
1050 
1051     /**
1052      * \copydoc close
1053      * \details
1054      * \par Handler signature
1055      * The handler signature for this operation is `void(boost::mysql::error_code)`.
1056      */
1057     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
1058     auto async_close(CompletionToken&& token)
1059         BOOST_MYSQL_RETURN_TYPE(detail::async_close_connection_t<CompletionToken&&>)
1060     {
1061         return async_close(impl_.shared_diag(), std::forward<CompletionToken>(token));
1062     }
1063 
1064     /// \copydoc async_close
1065     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
1066     auto async_close(diagnostics& diag, CompletionToken&& token)
1067         BOOST_MYSQL_RETURN_TYPE(detail::async_close_connection_t<CompletionToken&&>)
1068     {
1069         return this->impl_.async_run(
1070             this->impl_.make_params_close(diag),
1071             std::forward<CompletionToken>(token)
1072         );
1073     }
1074 
1075     /**
1076      * \brief Runs a set of pipelined requests.
1077      * \details
1078      * Runs the pipeline described by `req` and stores its response in `res`.
1079      * After the operation completes, `res` will have as many elements as stages
1080      * were in `req`, even if the operation fails.
1081      * \n
1082      * Request stages are seen by the server as a series of unrelated requests.
1083      * As a consequence, all stages are always run, even if previous stages fail.
1084      * \n
1085      * If all stages succeed, the operation completes successfully. Thus, there is no need to check
1086      * the per-stage error code in `res` if this operation completed successfully.
1087      * \n
1088      * If any stage fails with a non-fatal error (as per \ref is_fatal_error), the result of the operation
1089      * is the first encountered error. You can check which stages succeeded and which ones didn't by
1090      * inspecting each stage in `res`.
1091      * \n
1092      * If any stage fails with a fatal error, the result of the operation is the fatal error.
1093      * Successive stages will be marked as failed with the fatal error. The server may or may
1094      * not have processed such stages.
1095      */
1096     void run_pipeline(
1097         const pipeline_request& req,
1098         std::vector<stage_response>& res,
1099         error_code& err,
1100         diagnostics& diag
1101     )
1102     {
1103         impl_.run(impl_.make_params_pipeline(req, res, diag), err);
1104     }
1105 
1106     /// \copydoc run_pipeline
1107     void run_pipeline(const pipeline_request& req, std::vector<stage_response>& res)
1108     {
1109         error_code err;
1110         diagnostics diag;
1111         run_pipeline(req, res, err, diag);
1112         detail::throw_on_error_loc(err, diag, BOOST_CURRENT_LOCATION);
1113     }
1114 
1115     /**
1116      * \copydoc run_pipeline
1117      * \details
1118      * \par Handler signature
1119      * The handler signature for this operation is `void(boost::mysql::error_code)`.
1120      *
1121      * \par Object lifetimes
1122      * The request and response objects must be kept alive and should not be modified
1123      * until the operation completes.
1124      */
1125     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
1126     auto async_run_pipeline(
1127         const pipeline_request& req,
1128         std::vector<stage_response>& res,
1129         CompletionToken&& token
1130     ) BOOST_MYSQL_RETURN_TYPE(detail::async_run_pipeline_t<CompletionToken&&>)
1131     {
1132         return async_run_pipeline(req, res, impl_.shared_diag(), std::forward<CompletionToken>(token));
1133     }
1134 
1135     /// \copydoc async_run_pipeline
1136     template <BOOST_ASIO_COMPLETION_TOKEN_FOR(void(error_code)) CompletionToken>
1137     auto async_run_pipeline(
1138         const pipeline_request& req,
1139         std::vector<stage_response>& res,
1140         diagnostics& diag,
1141         CompletionToken&& token
1142     ) BOOST_MYSQL_RETURN_TYPE(detail::async_run_pipeline_t<CompletionToken&&>)
1143     {
1144         return this->impl_.async_run(
1145             impl_.make_params_pipeline(req, res, diag),
1146             std::forward<CompletionToken>(token)
1147         );
1148     }
1149 };
1150 
1151 }  // namespace mysql
1152 }  // namespace boost
1153 
1154 #ifdef BOOST_MYSQL_HEADER_ONLY
1155 #include <boost/mysql/impl/any_connection.ipp>
1156 #endif
1157 
1158 #endif