2018-09-23 17:14:25 +01:00
|
|
|
#include <catch.hpp>
|
2018-04-03 12:02:08 +02:00
|
|
|
#include <zmq.hpp>
|
2019-03-30 11:04:24 +00:00
|
|
|
#ifdef ZMQ_CPP11
|
|
|
|
#include <future>
|
|
|
|
#endif
|
2018-04-03 12:02:08 +02:00
|
|
|
|
2019-04-02 14:01:10 +00:00
|
|
|
#if (__cplusplus >= 201703L)
|
|
|
|
static_assert(std::is_nothrow_swappable<zmq::socket_t>::value,
|
|
|
|
"socket_t should be nothrow swappable");
|
|
|
|
#endif
|
|
|
|
|
2019-04-14 12:21:30 +00:00
|
|
|
TEST_CASE("socket default ctor", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::socket_t socket;
|
|
|
|
}
|
|
|
|
|
2018-09-23 17:14:25 +01:00
|
|
|
TEST_CASE("socket create destroy", "[socket]")
|
2018-04-03 12:02:08 +02:00
|
|
|
{
|
|
|
|
zmq::context_t context;
|
2018-05-12 17:28:28 +01:00
|
|
|
zmq::socket_t socket(context, ZMQ_ROUTER);
|
2018-04-03 12:02:08 +02:00
|
|
|
}
|
|
|
|
|
2018-04-03 12:04:23 +02:00
|
|
|
#ifdef ZMQ_CPP11
|
2019-04-14 12:21:30 +00:00
|
|
|
TEST_CASE("socket create assign", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t socket(context, ZMQ_ROUTER);
|
2019-04-07 21:10:04 +00:00
|
|
|
CHECK(static_cast<bool>(socket));
|
|
|
|
CHECK(socket.handle() != nullptr);
|
2019-04-14 12:21:30 +00:00
|
|
|
socket = {};
|
2019-04-07 21:10:04 +00:00
|
|
|
CHECK(!static_cast<bool>(socket));
|
|
|
|
CHECK(socket.handle() == nullptr);
|
2019-04-14 12:21:30 +00:00
|
|
|
}
|
|
|
|
|
2018-09-23 17:14:25 +01:00
|
|
|
TEST_CASE("socket create by enum and destroy", "[socket]")
|
2018-04-03 12:04:23 +02:00
|
|
|
{
|
|
|
|
zmq::context_t context;
|
2018-05-12 17:28:28 +01:00
|
|
|
zmq::socket_t socket(context, zmq::socket_type::router);
|
2018-04-03 12:04:23 +02:00
|
|
|
}
|
2019-04-02 14:01:10 +00:00
|
|
|
|
|
|
|
TEST_CASE("socket swap", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t socket1(context, zmq::socket_type::router);
|
|
|
|
zmq::socket_t socket2(context, zmq::socket_type::dealer);
|
|
|
|
using std::swap;
|
|
|
|
swap(socket1, socket2);
|
|
|
|
}
|
2019-05-09 22:33:49 +00:00
|
|
|
|
|
|
|
TEST_CASE("socket flags", "[socket]")
|
|
|
|
{
|
|
|
|
CHECK((zmq::recv_flags::dontwait | zmq::recv_flags::none)
|
|
|
|
== static_cast<zmq::recv_flags>(ZMQ_DONTWAIT | 0));
|
|
|
|
CHECK((zmq::recv_flags::dontwait & zmq::recv_flags::none)
|
|
|
|
== static_cast<zmq::recv_flags>(ZMQ_DONTWAIT & 0));
|
|
|
|
CHECK((zmq::recv_flags::dontwait ^ zmq::recv_flags::none)
|
|
|
|
== static_cast<zmq::recv_flags>(ZMQ_DONTWAIT ^ 0));
|
|
|
|
CHECK(~zmq::recv_flags::dontwait == static_cast<zmq::recv_flags>(~ZMQ_DONTWAIT));
|
|
|
|
|
|
|
|
CHECK((zmq::send_flags::dontwait | zmq::send_flags::sndmore)
|
|
|
|
== static_cast<zmq::send_flags>(ZMQ_DONTWAIT | ZMQ_SNDMORE));
|
|
|
|
CHECK((zmq::send_flags::dontwait & zmq::send_flags::sndmore)
|
|
|
|
== static_cast<zmq::send_flags>(ZMQ_DONTWAIT & ZMQ_SNDMORE));
|
|
|
|
CHECK((zmq::send_flags::dontwait ^ zmq::send_flags::sndmore)
|
|
|
|
== static_cast<zmq::send_flags>(ZMQ_DONTWAIT ^ ZMQ_SNDMORE));
|
|
|
|
CHECK(~zmq::send_flags::dontwait == static_cast<zmq::send_flags>(~ZMQ_DONTWAIT));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("socket readme example", "[socket]")
|
2019-04-22 21:15:51 +00:00
|
|
|
{
|
2019-05-08 21:37:14 +00:00
|
|
|
zmq::context_t ctx;
|
2019-04-22 21:15:51 +00:00
|
|
|
zmq::socket_t sock(ctx, zmq::socket_type::push);
|
|
|
|
sock.bind("inproc://test");
|
|
|
|
const std::string m = "Hello, world";
|
|
|
|
sock.send(zmq::buffer(m), zmq::send_flags::dontwait);
|
|
|
|
}
|
2018-04-03 12:04:23 +02:00
|
|
|
#endif
|
2018-08-21 13:36:01 +02:00
|
|
|
|
2018-09-23 17:14:25 +01:00
|
|
|
TEST_CASE("socket sends and receives const buffer", "[socket]")
|
2018-08-21 13:36:01 +02:00
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t sender(context, ZMQ_PAIR);
|
|
|
|
zmq::socket_t receiver(context, ZMQ_PAIR);
|
|
|
|
receiver.bind("inproc://test");
|
|
|
|
sender.connect("inproc://test");
|
2019-04-22 21:15:51 +00:00
|
|
|
const char* str = "Hi";
|
|
|
|
|
|
|
|
#ifdef ZMQ_CPP11
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(2 == *sender.send(zmq::buffer(str, 2)));
|
2019-04-22 21:15:51 +00:00
|
|
|
char buf[2];
|
|
|
|
const auto res = receiver.recv(zmq::buffer(buf));
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res);
|
|
|
|
CHECK(!res->truncated());
|
|
|
|
CHECK(2 == res->size);
|
2019-04-22 21:15:51 +00:00
|
|
|
#else
|
|
|
|
CHECK(2 == sender.send(str, 2));
|
2018-08-21 13:36:01 +02:00
|
|
|
char buf[2];
|
2018-09-23 17:14:25 +01:00
|
|
|
CHECK(2 == receiver.recv(buf, 2));
|
2019-04-22 21:15:51 +00:00
|
|
|
#endif
|
|
|
|
CHECK(0 == memcmp(buf, str, 2));
|
2018-08-21 13:36:01 +02:00
|
|
|
}
|
2019-03-30 11:04:24 +00:00
|
|
|
|
|
|
|
#ifdef ZMQ_CPP11
|
2019-04-22 21:15:51 +00:00
|
|
|
|
|
|
|
TEST_CASE("socket send none sndmore", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::router);
|
|
|
|
s.bind("inproc://test");
|
|
|
|
|
|
|
|
std::vector<char> buf(4);
|
|
|
|
auto res = s.send(zmq::buffer(buf), zmq::send_flags::sndmore);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res);
|
|
|
|
CHECK(*res == buf.size());
|
2019-04-22 21:15:51 +00:00
|
|
|
res = s.send(zmq::buffer(buf));
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res);
|
|
|
|
CHECK(*res == buf.size());
|
2019-04-22 21:15:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("socket send dontwait", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::push);
|
|
|
|
s.bind("inproc://test");
|
|
|
|
|
|
|
|
std::vector<char> buf(4);
|
|
|
|
auto res = s.send(zmq::buffer(buf), zmq::send_flags::dontwait);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(!res);
|
2019-04-22 21:15:51 +00:00
|
|
|
res = s.send(zmq::buffer(buf),
|
|
|
|
zmq::send_flags::dontwait | zmq::send_flags::sndmore);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(!res);
|
2019-04-22 21:15:51 +00:00
|
|
|
|
|
|
|
zmq::message_t msg;
|
|
|
|
auto resm = s.send(msg, zmq::send_flags::dontwait);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(!resm);
|
2019-04-22 21:15:51 +00:00
|
|
|
CHECK(msg.size() == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("socket send exception", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::pull);
|
|
|
|
s.bind("inproc://test");
|
|
|
|
|
|
|
|
std::vector<char> buf(4);
|
|
|
|
CHECK_THROWS_AS(s.send(zmq::buffer(buf)), const zmq::error_t &);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("socket recv none", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::pair);
|
|
|
|
zmq::socket_t s2(context, zmq::socket_type::pair);
|
|
|
|
s2.bind("inproc://test");
|
|
|
|
s.connect("inproc://test");
|
|
|
|
|
|
|
|
std::vector<char> sbuf(4);
|
|
|
|
const auto res_send = s2.send(zmq::buffer(sbuf));
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res_send);
|
|
|
|
CHECK(res_send.has_value());
|
2019-04-22 21:15:51 +00:00
|
|
|
|
|
|
|
std::vector<char> buf(2);
|
|
|
|
const auto res = s.recv(zmq::buffer(buf));
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res.has_value());
|
|
|
|
CHECK(res->truncated());
|
|
|
|
CHECK(res->untruncated_size == sbuf.size());
|
|
|
|
CHECK(res->size == buf.size());
|
2019-04-22 21:15:51 +00:00
|
|
|
|
|
|
|
const auto res_send2 = s2.send(zmq::buffer(sbuf));
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res_send2.has_value());
|
2019-04-22 21:15:51 +00:00
|
|
|
std::vector<char> buf2(10);
|
|
|
|
const auto res2 = s.recv(zmq::buffer(buf2));
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res2.has_value());
|
|
|
|
CHECK(!res2->truncated());
|
|
|
|
CHECK(res2->untruncated_size == sbuf.size());
|
|
|
|
CHECK(res2->size == sbuf.size());
|
2019-04-22 21:15:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("socket send recv message_t", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::pair);
|
|
|
|
zmq::socket_t s2(context, zmq::socket_type::pair);
|
|
|
|
s2.bind("inproc://test");
|
|
|
|
s.connect("inproc://test");
|
|
|
|
|
2019-05-15 16:49:22 +00:00
|
|
|
zmq::message_t smsg(10);
|
2019-04-22 21:15:51 +00:00
|
|
|
const auto res_send = s2.send(smsg, zmq::send_flags::none);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res_send);
|
|
|
|
CHECK(*res_send == 10);
|
2019-04-22 21:15:51 +00:00
|
|
|
CHECK(smsg.size() == 0);
|
|
|
|
|
|
|
|
zmq::message_t rmsg;
|
|
|
|
const auto res = s.recv(rmsg);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(res);
|
|
|
|
CHECK(*res == 10);
|
|
|
|
CHECK(res.value() == 10);
|
|
|
|
CHECK(rmsg.size() == *res);
|
2019-04-22 21:15:51 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 13:44:40 +00:00
|
|
|
TEST_CASE("socket send recv message_t by pointer", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::pair);
|
|
|
|
zmq::socket_t s2(context, zmq::socket_type::pair);
|
|
|
|
s2.bind("inproc://test");
|
|
|
|
s.connect("inproc://test");
|
|
|
|
|
|
|
|
zmq::message_t smsg(size_t{10});
|
|
|
|
const auto res_send = s2.send(smsg, zmq::send_flags::none);
|
|
|
|
CHECK(res_send);
|
|
|
|
CHECK(*res_send == 10);
|
|
|
|
CHECK(smsg.size() == 0);
|
|
|
|
|
|
|
|
zmq::message_t rmsg;
|
|
|
|
const bool res = s.recv(&rmsg);
|
|
|
|
CHECK(res);
|
|
|
|
}
|
|
|
|
|
2019-04-22 21:15:51 +00:00
|
|
|
TEST_CASE("socket recv dontwait", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::pull);
|
|
|
|
s.bind("inproc://test");
|
|
|
|
|
|
|
|
std::vector<char> buf(4);
|
|
|
|
constexpr auto flags = zmq::recv_flags::none | zmq::recv_flags::dontwait;
|
|
|
|
auto res = s.recv(zmq::buffer(buf), flags);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(!res);
|
2019-04-22 21:15:51 +00:00
|
|
|
|
|
|
|
zmq::message_t msg;
|
|
|
|
auto resm = s.recv(msg, flags);
|
2019-05-10 14:22:04 +00:00
|
|
|
CHECK(!resm);
|
|
|
|
CHECK_THROWS_AS(resm.value(), const std::exception &);
|
2019-04-22 21:15:51 +00:00
|
|
|
CHECK(msg.size() == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("socket recv exception", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t s(context, zmq::socket_type::push);
|
|
|
|
s.bind("inproc://test");
|
|
|
|
|
|
|
|
std::vector<char> buf(4);
|
|
|
|
CHECK_THROWS_AS(s.recv(zmq::buffer(buf)), const zmq::error_t &);
|
|
|
|
}
|
|
|
|
|
2019-03-30 11:04:24 +00:00
|
|
|
TEST_CASE("socket proxy", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t front(context, ZMQ_ROUTER);
|
|
|
|
zmq::socket_t back(context, ZMQ_ROUTER);
|
|
|
|
zmq::socket_t capture(context, ZMQ_DEALER);
|
|
|
|
front.bind("inproc://test1");
|
|
|
|
back.bind("inproc://test2");
|
|
|
|
capture.bind("inproc://test3");
|
|
|
|
auto f = std::async(std::launch::async, [&]() {
|
|
|
|
auto s1 = std::move(front);
|
|
|
|
auto s2 = std::move(back);
|
|
|
|
auto s3 = std::move(capture);
|
|
|
|
try
|
|
|
|
{
|
2019-04-07 21:10:04 +00:00
|
|
|
zmq::proxy(s1, s2, zmq::socket_ref(s3));
|
2019-03-30 11:04:24 +00:00
|
|
|
}
|
|
|
|
catch (const zmq::error_t& e)
|
|
|
|
{
|
|
|
|
return e.num() == ETERM;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
context.close();
|
|
|
|
CHECK(f.get());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE("socket proxy steerable", "[socket]")
|
|
|
|
{
|
|
|
|
zmq::context_t context;
|
|
|
|
zmq::socket_t front(context, ZMQ_ROUTER);
|
|
|
|
zmq::socket_t back(context, ZMQ_ROUTER);
|
|
|
|
zmq::socket_t control(context, ZMQ_SUB);
|
|
|
|
front.bind("inproc://test1");
|
|
|
|
back.bind("inproc://test2");
|
|
|
|
control.connect("inproc://test3");
|
|
|
|
auto f = std::async(std::launch::async, [&]() {
|
|
|
|
auto s1 = std::move(front);
|
|
|
|
auto s2 = std::move(back);
|
|
|
|
auto s3 = std::move(control);
|
|
|
|
try
|
|
|
|
{
|
2019-04-07 21:10:04 +00:00
|
|
|
zmq::proxy_steerable(s1, s2, zmq::socket_ref(), s3);
|
2019-03-30 11:04:24 +00:00
|
|
|
}
|
|
|
|
catch (const zmq::error_t& e)
|
|
|
|
{
|
|
|
|
return e.num() == ETERM;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
context.close();
|
|
|
|
CHECK(f.get());
|
|
|
|
}
|
|
|
|
#endif
|