mirror of
https://github.com/boostorg/mqtt5.git
synced 2025-10-04 04:40:55 +02:00
Summary: - Separate tests into unit & integration - Update paths in workflows - Cut test execution time Reviewers: ivica Reviewed By: ivica Subscribers: miljen, iljazovic Differential Revision: https://repo.mireo.local/D27374
305 lines
9.1 KiB
C++
305 lines
9.1 KiB
C++
#include <boost/test/unit_test.hpp>
|
|
|
|
#include <boost/asio/bind_cancellation_slot.hpp>
|
|
#include <boost/asio/cancellation_signal.hpp>
|
|
#include <boost/asio/io_context.hpp>
|
|
#include <boost/asio/steady_timer.hpp>
|
|
|
|
#include <async_mqtt5/impl/client_service.hpp>
|
|
#include <async_mqtt5/impl/publish_send_op.hpp>
|
|
|
|
#include "test_common/test_service.hpp"
|
|
|
|
using namespace async_mqtt5;
|
|
|
|
BOOST_AUTO_TEST_SUITE(publish_send_op/*, *boost::unit_test::disabled()*/)
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(test_pid_overrun) {
|
|
constexpr int expected_handlers_called = 1;
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
using client_service_type = test::overrun_client<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(ioc.get_executor(), "");
|
|
|
|
auto handler = [&handlers_called](error_code ec, reason_code rc, puback_props) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == client::error::pid_overrun);
|
|
BOOST_CHECK_EQUAL(rc, reason_codes::empty);
|
|
};
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_least_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform(
|
|
"test", "payload", retain_e::no, {}
|
|
);
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_invalid_topic_names) {
|
|
std::vector<std::string> invalid_topics = {
|
|
"", "+", "#",
|
|
"invalid+", "invalid#", "invalid/#", "invalid/+"
|
|
};
|
|
int expected_handlers_called = static_cast<int>(invalid_topics.size());
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
using client_service_type = test::test_service<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(ioc.get_executor());
|
|
|
|
for (const auto& topic: invalid_topics) {
|
|
auto handler = [&handlers_called](error_code ec) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == client::error::invalid_topic);
|
|
};
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_most_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform(topic, "payload", retain_e::no, {});
|
|
}
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_malformed_packet) {
|
|
uint16_t topic_alias_max = 10;
|
|
std::string malformed_str = std::string { 0x01 };
|
|
|
|
connack_props cprops;
|
|
cprops[prop::topic_alias_maximum] = topic_alias_max;
|
|
|
|
publish_props malfored_response_topic_props;
|
|
malfored_response_topic_props[prop::response_topic] = "response#topic";
|
|
|
|
publish_props utf8_payload_props;
|
|
utf8_payload_props[prop::payload_format_indicator] = uint8_t(1);
|
|
|
|
publish_props invalid_user_props;
|
|
invalid_user_props[prop::user_property].push_back(malformed_str);
|
|
|
|
publish_props malformed_content_type_props;
|
|
malformed_content_type_props[prop::content_type] = malformed_str;
|
|
|
|
publish_props zero_topic_alias_props;
|
|
zero_topic_alias_props[prop::topic_alias] = uint16_t(0);
|
|
|
|
publish_props out_of_range_subid_props;
|
|
out_of_range_subid_props[prop::subscription_identifier] = 300'000'000;
|
|
|
|
|
|
std::vector<publish_props> testing_props = {
|
|
malfored_response_topic_props, utf8_payload_props,
|
|
invalid_user_props, malformed_content_type_props,
|
|
zero_topic_alias_props, out_of_range_subid_props
|
|
};
|
|
|
|
int expected_handlers_called = static_cast<int>(testing_props.size());
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
using client_service_type = test::test_service<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(ioc.get_executor(), std::move(cprops));
|
|
BOOST_ASSERT(svc_ptr->connack_property(prop::topic_alias_maximum) == topic_alias_max);
|
|
|
|
for (const auto& props: testing_props) {
|
|
auto handler = [&handlers_called](error_code ec) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == client::error::malformed_packet);
|
|
};
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_most_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform("topic", malformed_str, retain_e::no, props);
|
|
}
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_packet_too_large) {
|
|
int max_packet_sz = 10;
|
|
|
|
connack_props props;
|
|
props[prop::maximum_packet_size] = max_packet_sz;
|
|
|
|
constexpr int expected_handlers_called = 1;
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
using client_service_type = test::test_service<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(
|
|
ioc.get_executor(), std::move(props)
|
|
);
|
|
BOOST_ASSERT(svc_ptr->connack_property(prop::maximum_packet_size) == max_packet_sz);
|
|
|
|
auto handler = [&handlers_called](error_code ec, reason_code rc, puback_props) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == client::error::packet_too_large);
|
|
BOOST_CHECK_EQUAL(rc, reason_codes::empty);
|
|
};
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_least_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform(
|
|
"test", "payload", retain_e::no, {}
|
|
);
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_qos_not_supported) {
|
|
uint8_t max_qos = 0;
|
|
|
|
connack_props props;
|
|
props[prop::maximum_qos] = max_qos;
|
|
|
|
constexpr int expected_handlers_called = 1;
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
using client_service_type = test::test_service<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(
|
|
ioc.get_executor(), std::move(props)
|
|
);
|
|
BOOST_ASSERT(svc_ptr->connack_property(prop::maximum_qos) == max_qos);
|
|
|
|
auto handler = [&handlers_called](error_code ec, reason_code rc, puback_props) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == client::error::qos_not_supported);
|
|
BOOST_CHECK_EQUAL(rc, reason_codes::empty);
|
|
};
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_least_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform(
|
|
"test", "payload", retain_e::no, {}
|
|
);
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_retain_not_available) {
|
|
uint8_t retain = 0;
|
|
|
|
connack_props props;
|
|
props[prop::retain_available] = retain;
|
|
|
|
constexpr int expected_handlers_called = 1;
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
using client_service_type = test::test_service<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(
|
|
ioc.get_executor(), std::move(props)
|
|
);
|
|
BOOST_ASSERT(svc_ptr->connack_property(prop::retain_available) == retain);
|
|
|
|
auto handler = [&handlers_called](error_code ec, reason_code rc, puback_props) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == client::error::retain_not_available);
|
|
BOOST_CHECK_EQUAL(rc, reason_codes::empty);
|
|
};
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_least_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform(
|
|
"test", "payload", retain_e::yes, {}
|
|
);
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_topic_alias_maximum) {
|
|
uint16_t max_topic_alias = 10;
|
|
|
|
connack_props ta_allowed_props;
|
|
ta_allowed_props[prop::topic_alias_maximum] = max_topic_alias;
|
|
|
|
std::vector<connack_props> test_props = {
|
|
ta_allowed_props, connack_props {} /* not allowed */
|
|
};
|
|
|
|
int expected_handlers_called = static_cast<int>(test_props.size());
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
|
|
for (const auto& ca_props: test_props) {
|
|
using client_service_type = test::test_service<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(
|
|
ioc.get_executor(), ca_props
|
|
);
|
|
|
|
auto handler = [&handlers_called](error_code ec) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == client::error::topic_alias_maximum_reached);
|
|
};
|
|
|
|
publish_props props;
|
|
props[prop::topic_alias] = uint16_t(12);
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_most_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform(
|
|
"test", "payload", retain_e::yes, props
|
|
);
|
|
}
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(test_publish_cancellation) {
|
|
constexpr int expected_handlers_called = 1;
|
|
int handlers_called = 0;
|
|
|
|
asio::io_context ioc;
|
|
using client_service_type = test::test_service<asio::ip::tcp::socket>;
|
|
auto svc_ptr = std::make_shared<client_service_type>(ioc.get_executor());
|
|
asio::cancellation_signal cancel_signal;
|
|
|
|
auto h = [&handlers_called](error_code ec, reason_code rc, puback_props) {
|
|
++handlers_called;
|
|
BOOST_CHECK(ec == asio::error::operation_aborted);
|
|
BOOST_CHECK_EQUAL(rc, reason_codes::empty);
|
|
};
|
|
|
|
auto handler = asio::bind_cancellation_slot(cancel_signal.slot(), std::move(h));
|
|
|
|
asio::steady_timer timer(ioc.get_executor());
|
|
timer.expires_after(std::chrono::milliseconds(60));
|
|
timer.async_wait(
|
|
[&cancel_signal](error_code) {
|
|
cancel_signal.emit(asio::cancellation_type::terminal);
|
|
}
|
|
);
|
|
|
|
detail::publish_send_op<
|
|
client_service_type, decltype(handler), qos_e::at_least_once
|
|
> { svc_ptr, std::move(handler) }
|
|
.perform(
|
|
"test", "payload", retain_e::no, {}
|
|
);
|
|
|
|
ioc.run_for(std::chrono::milliseconds(500));
|
|
BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called);
|
|
}
|
|
|
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|