mirror of https://gitee.com/bigwinds/arangodb
189 lines
6.1 KiB
C++
189 lines
6.1 KiB
C++
////////////////////////////////////////////////////////////////////////////////
|
|
/// DISCLAIMER
|
|
///
|
|
/// Copyright 2016 ArangoDB GmbH, Cologne, Germany
|
|
///
|
|
/// Licensed under the Apache License, Version 2.0 (the "License");
|
|
/// you may not use this file except in compliance with the License.
|
|
/// You may obtain a copy of the License at
|
|
///
|
|
/// http://www.apache.org/licenses/LICENSE-2.0
|
|
///
|
|
/// Unless required by applicable law or agreed to in writing, software
|
|
/// distributed under the License is distributed on an "AS IS" BASIS,
|
|
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
/// See the License for the specific language governing permissions and
|
|
/// limitations under the License.
|
|
///
|
|
/// Copyright holder is ArangoDB GmbH, Cologne, Germany
|
|
///
|
|
/// @author Dr. Frank Celler
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#ifndef ARANGOD_SCHEDULER_SOCKET_H
|
|
#define ARANGOD_SCHEDULER_SOCKET_H 1
|
|
|
|
#include "Basics/Common.h"
|
|
|
|
#include <boost/asio/basic_stream_socket.hpp>
|
|
#include <boost/asio/serial_port_service.hpp>
|
|
#include <boost/asio/ssl.hpp>
|
|
|
|
#include "Basics/StringBuffer.h"
|
|
#include "Basics/asio-helper.h"
|
|
#include "Logger/Logger.h"
|
|
|
|
namespace arangodb {
|
|
|
|
typedef std::function<void(const boost::system::error_code& ec,
|
|
std::size_t transferred)>
|
|
AsyncHandler;
|
|
|
|
namespace socketcommon {
|
|
template <typename T>
|
|
bool doSslHandshake(T& socket) {
|
|
boost::system::error_code ec;
|
|
|
|
uint64_t tries = 0;
|
|
double start = 0.0;
|
|
|
|
while (true) {
|
|
ec.assign(boost::system::errc::success,
|
|
boost::system::generic_category());
|
|
socket.handshake(
|
|
boost::asio::ssl::stream_base::handshake_type::server, ec);
|
|
|
|
if (ec.value() != boost::asio::error::would_block) {
|
|
break;
|
|
}
|
|
|
|
// got error EWOULDBLOCK and need to try again
|
|
++tries;
|
|
|
|
// following is a helpless fix for connections hanging in the handshake
|
|
// phase forever. we've seen this happening when the underlying peer
|
|
// connection was closed during the handshake.
|
|
// with the helpless fix, handshakes will be aborted it they take longer
|
|
// than x seconds. a proper fix is to make the handshake run asynchronously
|
|
// and somehow signal it that the connection got closed. apart from that
|
|
// running it asynchronously will not block the scheduler thread as it
|
|
// does now. anyway, even the helpless fix allows self-healing of busy
|
|
// scheduler threads after a network failure
|
|
if (tries == 1) {
|
|
// capture start time of handshake
|
|
start = TRI_microtime();
|
|
} else if (tries % 50 == 0) {
|
|
// check if we have spent more than x seconds handshaking and then abort
|
|
TRI_ASSERT(start != 0.0);
|
|
|
|
if (TRI_microtime() - start >= 3) {
|
|
ec.assign(boost::asio::error::connection_reset,
|
|
boost::system::generic_category());
|
|
LOG_TOPIC(DEBUG, Logger::COMMUNICATION) << "forcefully shutting down connection after wait time";
|
|
break;
|
|
} else {
|
|
usleep(10000);
|
|
}
|
|
}
|
|
|
|
// next iteration
|
|
}
|
|
|
|
if (ec) {
|
|
LOG_TOPIC(ERR, Logger::COMMUNICATION)
|
|
<< "unable to perform ssl handshake: " << ec.message() << " : "
|
|
<< ec.value();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <typename T>
|
|
size_t doWrite(T& socket, basics::StringBuffer* buffer,
|
|
boost::system::error_code& ec) {
|
|
return socket.write_some(
|
|
boost::asio::buffer(buffer->begin(), buffer->length()), ec);
|
|
}
|
|
template <typename T>
|
|
void doAsyncWrite(T& socket, boost::asio::mutable_buffers_1 const& buffer,
|
|
AsyncHandler const& handler) {
|
|
return boost::asio::async_write(socket, buffer, handler);
|
|
}
|
|
template <typename T>
|
|
size_t doRead(T& socket, boost::asio::mutable_buffers_1 const& buffer,
|
|
boost::system::error_code& ec) {
|
|
return socket.read_some(buffer, ec);
|
|
}
|
|
template <typename T>
|
|
void doAsyncRead(T& socket, boost::asio::mutable_buffers_1 const& buffer,
|
|
AsyncHandler const& handler) {
|
|
return socket.async_read_some(buffer, handler);
|
|
}
|
|
}
|
|
|
|
class Socket {
|
|
public:
|
|
Socket(boost::asio::io_service& ioService,
|
|
boost::asio::ssl::context&& context, bool encrypted)
|
|
: _ioService(ioService),
|
|
_context(std::move(context)),
|
|
_encrypted(encrypted) {}
|
|
Socket(Socket const& that) = delete;
|
|
Socket(Socket&& that) = delete;
|
|
virtual ~Socket() {}
|
|
|
|
virtual void setNonBlocking(bool) = 0;
|
|
virtual std::string peerAddress() = 0;
|
|
virtual int peerPort() = 0;
|
|
bool handshake();
|
|
virtual size_t write(basics::StringBuffer* buffer,
|
|
boost::system::error_code& ec) = 0;
|
|
virtual void asyncWrite(boost::asio::mutable_buffers_1 const& buffer,
|
|
AsyncHandler const& handler) = 0;
|
|
virtual size_t read(boost::asio::mutable_buffers_1 const& buffer,
|
|
boost::system::error_code& ec) = 0;
|
|
virtual std::size_t available(boost::system::error_code& ec) = 0;
|
|
virtual void asyncRead(boost::asio::mutable_buffers_1 const& buffer,
|
|
AsyncHandler const& handler) = 0;
|
|
|
|
virtual void shutdownAndClose(boost::system::error_code& ec, bool closeSend, bool closeReceive) {
|
|
if (closeSend) {
|
|
this->shutdownSend(ec);
|
|
if (ec && ec != boost::asio::error::not_connected) {
|
|
LOG_TOPIC(DEBUG, Logger::COMMUNICATION)
|
|
<< "shutdown send stream failed with: " << ec.message();
|
|
}
|
|
}
|
|
|
|
if (closeReceive) {
|
|
this->shutdownReceive(ec);
|
|
if (ec && ec != boost::asio::error::not_connected) {
|
|
LOG_TOPIC(DEBUG, Logger::COMMUNICATION)
|
|
<< "shutdown receive stream failed with: " << ec.message();
|
|
}
|
|
}
|
|
|
|
this->close(ec);
|
|
if (ec && ec != boost::asio::error::not_connected) {
|
|
LOG_TOPIC(DEBUG, Logger::COMMUNICATION)
|
|
<< "closing stream failed with: " << ec.message();
|
|
}
|
|
}
|
|
|
|
protected:
|
|
virtual bool sslHandshake() = 0;
|
|
virtual void shutdownReceive(boost::system::error_code& ec) = 0;
|
|
virtual void shutdownSend(boost::system::error_code& ec) = 0;
|
|
virtual void close(boost::system::error_code& ec) = 0;
|
|
|
|
public:
|
|
boost::asio::io_service& _ioService;
|
|
boost::asio::ssl::context _context;
|
|
|
|
bool _encrypted;
|
|
bool _handshakeDone = false;
|
|
};
|
|
}
|
|
|
|
#endif
|