C++ 关闭Boost ASIO SSL套接字的最佳方法是什么?

C++ What is the best way to shutdown a Boost ASIO SSL socket?

本文关键字:最佳 方法 是什么 套接字 关闭 Boost ASIO SSL C++      更新时间:2023-10-16

我有一个使用Boost ASIO SSL套接字的客户端应用程序。 我遇到了连接错误,使套接字处于不可用状态,直到我重新启动应用程序。 首先,我使用以下代码发出连接请求:

void apiClient::connect(boost::asio::ip::tcp::resolver::iterator endpoint_list)
{
boost::asio::ip::tcp::resolver::iterator endpoint_iter = endpoint_list;
boost::asio::ip::tcp::endpoint endpoint = *endpoint_iter;
if (!ssl_socket_->lowest_layer().is_open()) {
ssl_socket_->lowest_layer().async_connect(endpoint,
boost::bind(&apiClient::handle_connect, this,
boost::asio::placeholders::error, ++endpoint_iter));
}
}

定期我遇到返回给连接处理程序的错误,ec.message 中的错误是 [连接尝试失败,因为连接方在一段时间后没有正确响应,或者建立的连接失败,因为连接的主机无法响应] 错误代码 [10060]

这是我的连接处理程序:

void apiClient::handle_connect(const boost::system::error_code& ec, 
boost::asio::ip::tcp::resolver::iterator endpoint_list)
{
bool found_error(false);
if (!ec) {
ssl_socket_->async_handshake(boost::asio::ssl::stream_base::client,
boost::bind(&apiClient::send_request, this, 
boost::asio::placeholders::error));
}
else if (endpoint_list != boost::asio::ip::tcp::resolver::iterator()) {
std::wstring sMsg = boost::str(boost::wformat(L"API server connection failure to host [%s]. Trying next endpoint") % s2ws(endpoint_list->host_name()));
LogMsg(sMsg);
found_error = true;
epList = endpoint_list;
}
else {
std::wstring sMsg = boost::str(boost::wformat(L"API server connection failure.  Error msg [%s] Error code [%d]") % s2ws(ec.message()) % ec.value());
LogMsg(sMsg);
found_error = true;
epList = start_endpoint_list;
}
if (found_error) {
requeue_request(SHUTDOWN_CONN);
}
}

在我的requeue_request函数中,我检查请求遇到的错误数,如果超过阈值,我会丢弃请求并继续前进。 如果未超过阈值,则在尝试重新连接并再次发送消息之前关闭套接字连接。 我的理解是 emplace 方法重新创建套接字实例。

void apiClient::reconnect(SHUTDOWN_CONN_ACTION action)
{
if (action == SHUTDOWN_CONN) {
shutdownSocket();
}
ssl_socket_.emplace(*io_context_, *ctx_);
connect_timer.expires_from_now(boost::posix_time::milliseconds(3000));
connect_timer.async_wait(boost::bind(&apiClient::handleConnectTimer, this, boost::asio::placeholders::error));
}
void CNXGHotKeyAPIClient::shutdownSocket()
{
boost::system::error_code ec;
ssl_socket_->next_layer().cancel(ec);
ssl_socket_->shutdown(ec);
ssl_socket_->lowest_layer().shutdown(boost::asio::ip::tcp::socket::shutdown_both, ec);
ssl_socket_->lowest_layer().close(ec);
}

这个设计对我不起作用。 当我遇到任何连接错误时,我的应用程序会重试连接 3 次,它失败并出现相同的错误 3 次,套接字处于我无法使用的状态,直到我重新启动我的应用程序。 我正在连接的服务器正在尊重数百个 来自不使用 Boost 的 python 应用程序的连接请求,因此我认为服务器没有问题。

我试图找到使用async_shutdown的例子,但我找不到任何对我有帮助的东西。 此链接中的示例 安全断开 asio SSL 套接字的正确方法是什么?

使用以下我无法工作的代码。 示例代码:

boost::system::error_code ec;
ssl_socket.cancel(ec);
ssl_socket.async_shutdown([](...) { ssl_socket.close(); };

ssl_socket.cancel(( 在我的 Boost 版本 (v1.66( 中甚至不存在,我能找到的最接近的东西是 ssl_socket.next_layer((.cancel(ec(

我的问题是; 在遇到错误后,如何关闭套接字,以免它处于不可用状态? 当我的应用程序遇到连接错误时,很多时候我无法重新连接,直到我重新启动应用程序。 该应用程序仍在运行并为用户操作提供服务,因此我知道它没有崩溃,但套接字处于不可用状态。 感谢您的任何帮助

  1. 您无法关闭 SSL 套接字。你必须关闭它。
  2. 连接套接字
  3. 后,即使连接尝试失败,也无法重新连接套接字。您必须关闭它并创建一个新文件。