2019-08-06 08:51:23 -05:00
|
|
|
/*
|
|
|
|
* ZeroTier One - Network Virtualization Everywhere
|
|
|
|
* Copyright (C) 2011-2019 ZeroTier, Inc. https://www.zerotier.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 3 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* --
|
|
|
|
*
|
|
|
|
* You can be released from the requirements of the license by purchasing
|
|
|
|
* a commercial license. Buying such a license is mandatory as soon as you
|
|
|
|
* develop commercial closed-source software that incorporates or links
|
|
|
|
* directly against ZeroTier software without disclosing the source code
|
|
|
|
* of your own application.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2019-03-05 15:11:50 -08:00
|
|
|
#include "RabbitMQ.hpp"
|
|
|
|
|
2019-03-11 11:16:44 -07:00
|
|
|
#ifdef ZT_CONTROLLER_USE_LIBPQ
|
|
|
|
|
2019-03-05 15:11:50 -08:00
|
|
|
#include <amqp.h>
|
|
|
|
#include <amqp_tcp_socket.h>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
namespace ZeroTier
|
|
|
|
{
|
|
|
|
|
|
|
|
RabbitMQ::RabbitMQ(MQConfig *cfg, const char *queueName)
|
2019-08-06 07:51:50 -05:00
|
|
|
: _mqc(cfg)
|
|
|
|
, _qName(queueName)
|
|
|
|
, _socket(NULL)
|
|
|
|
, _status(0)
|
2019-03-05 15:11:50 -08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
RabbitMQ::~RabbitMQ()
|
|
|
|
{
|
2019-08-06 07:51:50 -05:00
|
|
|
amqp_channel_close(_conn, _channel, AMQP_REPLY_SUCCESS);
|
|
|
|
amqp_connection_close(_conn, AMQP_REPLY_SUCCESS);
|
|
|
|
amqp_destroy_connection(_conn);
|
2019-03-05 15:11:50 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void RabbitMQ::init()
|
|
|
|
{
|
2019-08-06 07:51:50 -05:00
|
|
|
struct timeval tval;
|
|
|
|
memset(&tval, 0, sizeof(struct timeval));
|
|
|
|
tval.tv_sec = 5;
|
2019-03-05 15:11:50 -08:00
|
|
|
|
2019-08-06 07:51:50 -05:00
|
|
|
fprintf(stderr, "Initializing RabbitMQ %s\n", _qName);
|
|
|
|
_conn = amqp_new_connection();
|
|
|
|
_socket = amqp_tcp_socket_new(_conn);
|
|
|
|
if (!_socket) {
|
|
|
|
throw std::runtime_error("Can't create socket for RabbitMQ");
|
|
|
|
}
|
|
|
|
|
|
|
|
_status = amqp_socket_open_noblock(_socket, _mqc->host, _mqc->port, &tval);
|
|
|
|
if (_status) {
|
|
|
|
throw std::runtime_error("Can't connect to RabbitMQ");
|
|
|
|
}
|
|
|
|
|
|
|
|
amqp_rpc_reply_t r = amqp_login(_conn, "/", 0, 131072, 0, AMQP_SASL_METHOD_PLAIN,
|
|
|
|
_mqc->username, _mqc->password);
|
|
|
|
if (r.reply_type != AMQP_RESPONSE_NORMAL) {
|
|
|
|
throw std::runtime_error("RabbitMQ Login Error");
|
|
|
|
}
|
2019-03-05 15:11:50 -08:00
|
|
|
|
2019-08-06 07:51:50 -05:00
|
|
|
static int chan = 0;
|
2019-03-08 10:29:36 -08:00
|
|
|
{
|
|
|
|
Mutex::Lock l(_chan_m);
|
2019-08-06 07:51:50 -05:00
|
|
|
_channel = ++chan;
|
|
|
|
}
|
|
|
|
amqp_channel_open(_conn, _channel);
|
|
|
|
r = amqp_get_rpc_reply(_conn);
|
|
|
|
if(r.reply_type != AMQP_RESPONSE_NORMAL) {
|
|
|
|
throw std::runtime_error("Error opening communication channel");
|
|
|
|
}
|
|
|
|
|
|
|
|
_q = amqp_queue_declare(_conn, _channel, amqp_cstring_bytes(_qName), 0, 0, 0, 0, amqp_empty_table);
|
|
|
|
r = amqp_get_rpc_reply(_conn);
|
|
|
|
if (r.reply_type != AMQP_RESPONSE_NORMAL) {
|
|
|
|
throw std::runtime_error("Error declaring queue " + std::string(_qName));
|
2019-03-08 10:29:36 -08:00
|
|
|
}
|
2019-03-05 15:11:50 -08:00
|
|
|
|
2019-08-06 07:51:50 -05:00
|
|
|
amqp_basic_consume(_conn, _channel, amqp_cstring_bytes(_qName), amqp_empty_bytes, 0, 1, 0, amqp_empty_table);
|
|
|
|
r = amqp_get_rpc_reply(_conn);
|
|
|
|
if (r.reply_type != AMQP_RESPONSE_NORMAL) {
|
|
|
|
throw std::runtime_error("Error consuming queue " + std::string(_qName));
|
|
|
|
}
|
|
|
|
fprintf(stderr, "RabbitMQ Init OK %s\n", _qName);
|
2019-03-05 15:11:50 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string RabbitMQ::consume()
|
|
|
|
{
|
2019-08-06 07:51:50 -05:00
|
|
|
amqp_rpc_reply_t res;
|
|
|
|
amqp_envelope_t envelope;
|
|
|
|
amqp_maybe_release_buffers(_conn);
|
2019-03-05 15:11:50 -08:00
|
|
|
|
2019-08-06 07:51:50 -05:00
|
|
|
struct timeval timeout;
|
|
|
|
timeout.tv_sec = 1;
|
|
|
|
timeout.tv_usec = 0;
|
2019-04-18 14:57:06 -07:00
|
|
|
|
2019-08-06 07:51:50 -05:00
|
|
|
res = amqp_consume_message(_conn, &envelope, &timeout, 0);
|
|
|
|
if (res.reply_type != AMQP_RESPONSE_NORMAL) {
|
|
|
|
if (res.reply_type == AMQP_RESPONSE_LIBRARY_EXCEPTION && res.library_error == AMQP_STATUS_TIMEOUT) {
|
|
|
|
// timeout waiting for message. Return empty string
|
|
|
|
return "";
|
|
|
|
} else {
|
|
|
|
throw std::runtime_error("Error getting message");
|
|
|
|
}
|
|
|
|
}
|
2019-03-05 15:11:50 -08:00
|
|
|
|
2019-08-06 07:51:50 -05:00
|
|
|
std::string msg(
|
|
|
|
(const char*)envelope.message.body.bytes,
|
|
|
|
envelope.message.body.len
|
|
|
|
);
|
|
|
|
amqp_destroy_envelope(&envelope);
|
|
|
|
return msg;
|
2019-03-05 15:11:50 -08:00
|
|
|
}
|
|
|
|
|
2019-03-08 10:29:36 -08:00
|
|
|
}
|
2019-03-11 11:16:44 -07:00
|
|
|
|
|
|
|
#endif // ZT_CONTROLLER_USE_LIBPQ
|