mirror of https://github.com/Icinga/icinga2.git
411 lines
10 KiB
C++
411 lines
10 KiB
C++
/******************************************************************************
|
|
* Icinga 2 *
|
|
* Copyright (C) 2012-2018 Icinga Development Team (https://www.icinga.com/) *
|
|
* *
|
|
* This program is free software; you can redistribute it and/or *
|
|
* modify it under the terms of the GNU General Public License *
|
|
* as published by the Free Software Foundation; either version 2 *
|
|
* of the License, or (at your option) any later version. *
|
|
* *
|
|
* This program is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU General Public License *
|
|
* along with this program; if not, write to the Free Software Foundation *
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
******************************************************************************/
|
|
|
|
#ifndef REDISCONNECTION_H
|
|
#define REDISCONNECTION_H
|
|
|
|
#include <boost/asio/spawn.hpp>
|
|
#include <boost/asio/buffer.hpp>
|
|
#include <boost/asio/buffered_stream.hpp>
|
|
#include <boost/asio/io_context.hpp>
|
|
#include <boost/asio/io_context_strand.hpp>
|
|
#include <boost/asio/ip/tcp.hpp>
|
|
#include <boost/asio/local/stream_protocol.hpp>
|
|
#include <boost/asio/read.hpp>
|
|
#include <boost/asio/read_until.hpp>
|
|
#include <boost/asio/streambuf.hpp>
|
|
#include <boost/asio/write.hpp>
|
|
#include <boost/lexical_cast.hpp>
|
|
#include <boost/utility/string_view.hpp>
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
#include <future>
|
|
#include <memory>
|
|
#include <queue>
|
|
#include <stdexcept>
|
|
#include <utility>
|
|
#include <vector>
|
|
#include "base/array.hpp"
|
|
#include "base/atomic.hpp"
|
|
#include "base/io-engine.hpp"
|
|
#include "base/object.hpp"
|
|
#include "base/string.hpp"
|
|
#include "base/value.hpp"
|
|
|
|
namespace icinga
|
|
{
|
|
/**
|
|
* An Async Redis connection.
|
|
*
|
|
* @ingroup icingadb
|
|
*/
|
|
class RedisConnection final : public Object
|
|
{
|
|
public:
|
|
DECLARE_PTR_TYPEDEFS(RedisConnection);
|
|
|
|
typedef std::vector<String> Query;
|
|
typedef std::vector<Query> Queries;
|
|
typedef Value Reply;
|
|
typedef std::vector<Reply> Replies;
|
|
|
|
RedisConnection(const String host, const int port, const String path, const String password = "", const int db = 0);
|
|
|
|
void Start();
|
|
|
|
bool IsConnected();
|
|
|
|
void FireAndForgetQuery(Query query, bool highPrio = false);
|
|
void FireAndForgetQueries(Queries queries, bool highPrio = false);
|
|
|
|
Reply GetResultOfQuery(Query query, bool highPrio = false);
|
|
Replies GetResultsOfQueries(Queries queries, bool highPrio = false);
|
|
|
|
private:
|
|
enum class ResponseAction : unsigned char
|
|
{
|
|
Ignore, Deliver, DeliverBulk
|
|
};
|
|
|
|
struct FutureResponseAction
|
|
{
|
|
size_t Amount;
|
|
ResponseAction Action;
|
|
};
|
|
|
|
struct WriteQueueItem
|
|
{
|
|
std::shared_ptr<Query> FireAndForgetQuery;
|
|
std::shared_ptr<Queries> FireAndForgetQueries;
|
|
std::shared_ptr<std::pair<Query, std::promise<Reply>>> GetResultOfQuery;
|
|
std::shared_ptr<std::pair<Queries, std::promise<Replies>>> GetResultsOfQueries;
|
|
};
|
|
|
|
typedef boost::asio::ip::tcp Tcp;
|
|
typedef boost::asio::local::stream_protocol Unix;
|
|
|
|
typedef boost::asio::buffered_stream<Tcp::socket> TcpConn;
|
|
typedef boost::asio::buffered_stream<Unix::socket> UnixConn;
|
|
|
|
template<class AsyncReadStream>
|
|
static Value ReadRESP(AsyncReadStream& stream, boost::asio::yield_context& yc);
|
|
|
|
template<class AsyncReadStream>
|
|
static std::vector<char> ReadLine(AsyncReadStream& stream, boost::asio::yield_context& yc, size_t hint = 0);
|
|
|
|
template<class AsyncWriteStream>
|
|
static void WriteRESP(AsyncWriteStream& stream, const Query& query, boost::asio::yield_context& yc);
|
|
|
|
template<class AsyncWriteStream>
|
|
static void WriteInt(AsyncWriteStream& stream, intmax_t i, boost::asio::yield_context& yc);
|
|
|
|
RedisConnection(boost::asio::io_context& io, String host, int port, String path, String password, int db);
|
|
|
|
void Connect(boost::asio::yield_context& yc);
|
|
void ReadLoop(boost::asio::yield_context& yc);
|
|
void WriteLoop(boost::asio::yield_context& yc);
|
|
void WriteItem(boost::asio::yield_context& yc, WriteQueueItem item);
|
|
Reply ReadOne(boost::asio::yield_context& yc);
|
|
void WriteOne(Query& query, boost::asio::yield_context& yc);
|
|
|
|
template<class StreamPtr>
|
|
Reply ReadOne(StreamPtr& stream, boost::asio::yield_context& yc);
|
|
|
|
template<class StreamPtr>
|
|
void WriteOne(StreamPtr& stream, Query& query, boost::asio::yield_context& yc);
|
|
|
|
String m_Path;
|
|
String m_Host;
|
|
int m_Port;
|
|
String m_Password;
|
|
int m_DbIndex;
|
|
|
|
boost::asio::io_context::strand m_Strand;
|
|
std::shared_ptr<TcpConn> m_TcpConn;
|
|
std::shared_ptr<UnixConn> m_UnixConn;
|
|
Atomic<bool> m_Connecting, m_Connected, m_Started;
|
|
|
|
struct {
|
|
std::queue<WriteQueueItem> Writes, HighPrioWrites;
|
|
std::queue<std::promise<Reply>> ReplyPromises;
|
|
std::queue<std::promise<Replies>> RepliesPromises;
|
|
std::queue<FutureResponseAction> FutureResponseActions;
|
|
} m_Queues;
|
|
|
|
AsioConditionVariable m_QueuedWrites, m_QueuedReads;
|
|
};
|
|
|
|
class RedisError final : public Object
|
|
{
|
|
public:
|
|
DECLARE_PTR_TYPEDEFS(RedisError);
|
|
|
|
inline RedisError(String message) : m_Message(std::move(message))
|
|
{
|
|
}
|
|
|
|
inline const String& GetMessage()
|
|
{
|
|
return m_Message;
|
|
}
|
|
|
|
private:
|
|
String m_Message;
|
|
};
|
|
|
|
class RedisDisconnected : public std::runtime_error
|
|
{
|
|
public:
|
|
inline RedisDisconnected() : runtime_error("")
|
|
{
|
|
}
|
|
};
|
|
|
|
class RedisProtocolError : public std::runtime_error
|
|
{
|
|
protected:
|
|
inline RedisProtocolError() : runtime_error("")
|
|
{
|
|
}
|
|
};
|
|
|
|
class BadRedisType : public RedisProtocolError
|
|
{
|
|
public:
|
|
inline BadRedisType(char type) : m_What{type, 0}
|
|
{
|
|
}
|
|
|
|
virtual const char * what() const noexcept override
|
|
{
|
|
return m_What;
|
|
}
|
|
|
|
private:
|
|
char m_What[2];
|
|
};
|
|
|
|
class BadRedisInt : public RedisProtocolError
|
|
{
|
|
public:
|
|
inline BadRedisInt(std::vector<char> intStr) : m_What(std::move(intStr))
|
|
{
|
|
m_What.emplace_back(0);
|
|
}
|
|
|
|
virtual const char * what() const noexcept override
|
|
{
|
|
return m_What.data();
|
|
}
|
|
|
|
private:
|
|
std::vector<char> m_What;
|
|
};
|
|
|
|
template<class StreamPtr>
|
|
RedisConnection::Reply RedisConnection::ReadOne(StreamPtr& stream, boost::asio::yield_context& yc)
|
|
{
|
|
if (!stream) {
|
|
throw RedisDisconnected();
|
|
}
|
|
|
|
auto strm (stream);
|
|
|
|
try {
|
|
return ReadRESP(*strm, yc);
|
|
} catch (const boost::coroutines::detail::forced_unwind&) {
|
|
throw;
|
|
} catch (...) {
|
|
if (m_Connecting.exchange(false)) {
|
|
m_Connected.store(false);
|
|
stream = nullptr;
|
|
}
|
|
throw;
|
|
}
|
|
}
|
|
|
|
template<class StreamPtr>
|
|
void RedisConnection::WriteOne(StreamPtr& stream, RedisConnection::Query& query, boost::asio::yield_context& yc)
|
|
{
|
|
if (!stream) {
|
|
throw RedisDisconnected();
|
|
}
|
|
|
|
auto strm (stream);
|
|
|
|
try {
|
|
WriteRESP(*strm, query, yc);
|
|
strm->async_flush(yc);
|
|
} catch (const boost::coroutines::detail::forced_unwind&) {
|
|
throw;
|
|
} catch (...) {
|
|
if (m_Connecting.exchange(false)) {
|
|
m_Connected.store(false);
|
|
stream = nullptr;
|
|
}
|
|
throw;
|
|
}
|
|
}
|
|
|
|
template<class AsyncReadStream>
|
|
Value RedisConnection::ReadRESP(AsyncReadStream& stream, boost::asio::yield_context& yc)
|
|
{
|
|
namespace asio = boost::asio;
|
|
|
|
char type = 0;
|
|
asio::async_read(stream, asio::mutable_buffer(&type, 1), yc);
|
|
|
|
switch (type) {
|
|
case '+':
|
|
{
|
|
auto buf (ReadLine(stream, yc));
|
|
return String(buf.begin(), buf.end());
|
|
}
|
|
case '-':
|
|
{
|
|
auto buf (ReadLine(stream, yc));
|
|
return new RedisError(String(buf.begin(), buf.end()));
|
|
}
|
|
case ':':
|
|
{
|
|
auto buf (ReadLine(stream, yc, 21));
|
|
intmax_t i = 0;
|
|
|
|
try {
|
|
i = boost::lexical_cast<intmax_t>(boost::string_view(buf.data(), buf.size()));
|
|
} catch (...) {
|
|
throw BadRedisInt(std::move(buf));
|
|
}
|
|
|
|
return (double)i;
|
|
}
|
|
case '$':
|
|
{
|
|
auto buf (ReadLine(stream, yc, 21));
|
|
intmax_t i = 0;
|
|
|
|
try {
|
|
i = boost::lexical_cast<intmax_t>(boost::string_view(buf.data(), buf.size()));
|
|
} catch (...) {
|
|
throw BadRedisInt(std::move(buf));
|
|
}
|
|
|
|
if (i < 0) {
|
|
return Value();
|
|
}
|
|
|
|
buf.clear();
|
|
buf.insert(buf.end(), i, 0);
|
|
asio::async_read(stream, asio::mutable_buffer(buf.data(), buf.size()), yc);
|
|
|
|
{
|
|
char crlf[2];
|
|
asio::async_read(stream, asio::mutable_buffer(crlf, 2), yc);
|
|
}
|
|
|
|
return String(buf.begin(), buf.end());
|
|
}
|
|
case '*':
|
|
{
|
|
auto buf (ReadLine(stream, yc, 21));
|
|
intmax_t i = 0;
|
|
|
|
try {
|
|
i = boost::lexical_cast<intmax_t>(boost::string_view(buf.data(), buf.size()));
|
|
} catch (...) {
|
|
throw BadRedisInt(std::move(buf));
|
|
}
|
|
|
|
Array::Ptr arr = new Array();
|
|
|
|
if (i < 0) {
|
|
i = 0;
|
|
}
|
|
|
|
arr->Reserve(i);
|
|
|
|
for (; i; --i) {
|
|
arr->Add(ReadRESP(stream, yc));
|
|
}
|
|
|
|
return arr;
|
|
}
|
|
default:
|
|
throw BadRedisType(type);
|
|
}
|
|
}
|
|
|
|
template<class AsyncReadStream>
|
|
std::vector<char> RedisConnection::ReadLine(AsyncReadStream& stream, boost::asio::yield_context& yc, size_t hint)
|
|
{
|
|
namespace asio = boost::asio;
|
|
|
|
std::vector<char> line;
|
|
line.reserve(hint);
|
|
|
|
char next = 0;
|
|
asio::mutable_buffer buf (&next, 1);
|
|
|
|
for (;;) {
|
|
asio::async_read(stream, buf, yc);
|
|
|
|
if (next == '\r') {
|
|
asio::async_read(stream, buf, yc);
|
|
return std::move(line);
|
|
}
|
|
|
|
line.emplace_back(next);
|
|
}
|
|
}
|
|
|
|
template<class AsyncWriteStream>
|
|
void RedisConnection::WriteRESP(AsyncWriteStream& stream, const Query& query, boost::asio::yield_context& yc)
|
|
{
|
|
namespace asio = boost::asio;
|
|
|
|
asio::async_write(stream, asio::const_buffer("*", 1), yc);
|
|
WriteInt(stream, query.size(), yc);
|
|
asio::async_write(stream, asio::const_buffer("\r\n", 2), yc);
|
|
|
|
for (auto& arg : query) {
|
|
asio::async_write(stream, asio::const_buffer("$", 1), yc);
|
|
WriteInt(stream, arg.GetLength(), yc);
|
|
asio::async_write(stream, asio::const_buffer("\r\n", 2), yc);
|
|
asio::async_write(stream, asio::const_buffer(arg.CStr(), arg.GetLength()), yc);
|
|
asio::async_write(stream, asio::const_buffer("\r\n", 2), yc);
|
|
}
|
|
}
|
|
|
|
template<class AsyncWriteStream>
|
|
void RedisConnection::WriteInt(AsyncWriteStream& stream, intmax_t i, boost::asio::yield_context& yc)
|
|
{
|
|
namespace asio = boost::asio;
|
|
|
|
char buf[21] = {};
|
|
sprintf(buf, "%jd", i);
|
|
|
|
asio::async_write(stream, asio::const_buffer(buf, strlen(buf)), yc);
|
|
}
|
|
|
|
}
|
|
|
|
#endif //REDISCONNECTION_H
|