#include #include #include "test_common/message_exchange.hpp" #include "test_common/test_service.hpp" #include "test_common/test_stream.hpp" using namespace async_mqtt5; BOOST_AUTO_TEST_SUITE(malformed_packet/* , *boost::unit_test::disabled()*/) BOOST_AUTO_TEST_CASE(test_malformed_publish) { using test::after; using std::chrono_literals::operator ""ms; // packets auto connect = encoders::encode_connect( "", std::nullopt, std::nullopt, 10, false, {}, std::nullopt ); auto connack = encoders::encode_connack( false, reason_codes::success.value(), {} ); auto publish = encoders::encode_publish( 1, "topic", "payload", static_cast(0b11), retain_e::yes, dup_e::no, {} ); disconnect_props dprops; dprops[prop::reason_string] = "Malformed PUBLISH received: QoS bits set to 0b11"; auto disconnect = encoders::encode_disconnect( reason_codes::malformed_packet.value(), dprops ); test::msg_exchange broker_side; error_code success {}; broker_side .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .send(publish, after(10ms)) .expect(disconnect) .complete_with(success, after(1ms)); asio::io_context ioc; auto executor = ioc.get_executor(); auto& broker = asio::make_service( ioc, executor, std::move(broker_side) ); using client_type = mqtt_client; client_type c(executor, ""); c.brokers("127.0.0.1,127.0.0.1") // to avoid reconnect backoff .run(); asio::steady_timer timer(c.get_executor()); timer.expires_after(std::chrono::seconds(2)); timer.async_wait([&](auto) { c.cancel(); }); ioc.run(); BOOST_CHECK(broker.received_all_expected()); } // does not receive the malformed pubrel BOOST_AUTO_TEST_CASE(test_malformed_pubrel, *boost::unit_test::disabled()) { using test::after; using std::chrono_literals::operator ""ms; constexpr int expected_handlers_called = 1; int handlers_called = 0; std::string topic = "topic"; std::string payload = "payload"; // packets auto connect = encoders::encode_connect( "", std::nullopt, std::nullopt, 10, false, {}, std::nullopt ); auto connack = encoders::encode_connack( false, reason_codes::success.value(), {} ); auto publish = encoders::encode_publish( 1, topic, payload, qos_e::exactly_once, retain_e::yes, dup_e::no, {} ); auto pubrec = encoders::encode_pubrec(1, reason_codes::success.value(), {}); auto pubrel = encoders::encode_pubrel(1, reason_codes::success.value(), {}); auto malformed_pubrel = encoders::encode_pubrel(1, 0x04, {}); auto pubcomp = encoders::encode_pubcomp(1, reason_codes::success.value(), {}); disconnect_props dprops; dprops[prop::reason_string] = "Malformed PUBREL received: invalid Reason Code"; auto disconnect = encoders::encode_disconnect( reason_codes::malformed_packet.value(), dprops ); test::msg_exchange broker_side; error_code success {}; broker_side .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .send(publish, after(10ms)) .expect(pubrec) .complete_with(success, after(1ms)) .reply_with(malformed_pubrel, after(2ms)) .expect(disconnect) .complete_with(success, after(1ms)) .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .send(pubrel, after(10ms)) .expect(pubcomp) .complete_with(success, after(1ms)); asio::io_context ioc; auto executor = ioc.get_executor(); auto& broker = asio::make_service( ioc, executor, std::move(broker_side) ); using client_type = mqtt_client; client_type c(executor, ""); c.brokers("127.0.0.1,127.0.0.1") // to avoid reconnect backoff .run(); c.async_receive( [&]( error_code ec, std::string rec_topic, std::string rec_payload, publish_props ) { ++handlers_called; BOOST_CHECK_MESSAGE(!ec, ec.message()); BOOST_CHECK_EQUAL(topic, rec_topic); BOOST_CHECK_EQUAL(payload, rec_payload); c.cancel(); } ); ioc.run_for(std::chrono::seconds(2)); BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called); BOOST_CHECK(broker.received_all_expected()); } BOOST_AUTO_TEST_CASE(malformed_puback) { using test::after; using std::chrono_literals::operator ""ms; constexpr int expected_handlers_called = 1; int handlers_called = 0; // packets auto connect = encoders::encode_connect( "", std::nullopt, std::nullopt, 10, false, {}, std::nullopt ); auto connack = encoders::encode_connack(false, reason_codes::success.value(), {}); auto publish = encoders::encode_publish( 1, "t", "p", qos_e::at_least_once, retain_e::no, dup_e::no, {} ); auto malformed_puback = encoders::encode_puback(1, uint8_t(0x04), {}); auto publish_dup = encoders::encode_publish( 1, "t", "p", qos_e::at_least_once, retain_e::no, dup_e::yes, {} ); auto puback = encoders::encode_puback(1, reason_codes::success.value(), {}); disconnect_props dc_props; dc_props[prop::reason_string] = "Malformed PUBACK: invalid Reason Code"; auto disconnect = encoders::encode_disconnect( reason_codes::malformed_packet.value(), dc_props ); test::msg_exchange broker_side; error_code success {}; broker_side .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .expect(publish) .complete_with(success, after(0ms)) .reply_with(malformed_puback, after(0ms)) .expect(disconnect) .complete_with(success, after(0ms)) .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .expect(publish_dup) .complete_with(success, after(0ms)) .reply_with(puback, after(0ms)); asio::io_context ioc; auto executor = ioc.get_executor(); auto& broker = asio::make_service( ioc, executor, std::move(broker_side) ); using client_type = mqtt_client; client_type c(executor, ""); c.brokers("127.0.0.1,127.0.0.1") // to avoid reconnect backoff .run(); c.async_publish( "t", "p", retain_e::no, publish_props {}, [&](error_code ec, reason_code rc, auto) { ++handlers_called; BOOST_CHECK(!ec); BOOST_CHECK_EQUAL(rc, reason_codes::success); c.cancel(); } ); ioc.run_for(std::chrono::seconds(2)); BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called); BOOST_CHECK(broker.received_all_expected()); } BOOST_AUTO_TEST_CASE(malformed_pubrec_pubcomp) { using test::after; using std::chrono_literals::operator ""ms; constexpr int expected_handlers_called = 1; int handlers_called = 0; // packets auto connect = encoders::encode_connect( "", std::nullopt, std::nullopt, 10, false, {}, std::nullopt ); auto connack = encoders::encode_connack(false, reason_codes::success.value(), {}); auto publish = encoders::encode_publish( 1, "t", "p", qos_e::exactly_once, retain_e::no, dup_e::no, {} ); auto malformed_pubrec = encoders::encode_pubrec(1, uint8_t(0x04), {}); auto publish_dup = encoders::encode_publish( 1, "t", "p", qos_e::exactly_once, retain_e::no, dup_e::yes, {} ); auto pubrec = encoders::encode_pubrec(1, reason_codes::success.value(), {}); auto pubrel = encoders::encode_pubrel(1, reason_codes::success.value(), {}); auto malformed_pubcomp = encoders::encode_pubcomp(1, uint8_t(0x04), {}); auto pubcomp = encoders::encode_pubcomp(1, reason_codes::success.value(), {}); disconnect_props dc_props; dc_props[prop::reason_string] = "Malformed PUBREC: invalid Reason Code"; auto disconnect_on_pubrec = encoders::encode_disconnect( reason_codes::malformed_packet.value(), dc_props ); dc_props[prop::reason_string] = "Malformed PUBCOMP: invalid Reason Code"; auto disconnect_on_pubcomp = encoders::encode_disconnect( reason_codes::malformed_packet.value(), dc_props ); test::msg_exchange broker_side; error_code success {}; broker_side .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .expect(publish) .complete_with(success, after(0ms)) .reply_with(malformed_pubrec, after(0ms)) .expect(disconnect_on_pubrec) .complete_with(success, after(0ms)) .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .expect(publish_dup) .complete_with(success, after(0ms)) .reply_with(pubrec, after(0ms)) .expect(pubrel) .complete_with(success, after(0ms)) .reply_with(malformed_pubcomp, after(0ms)) .expect(disconnect_on_pubcomp) .complete_with(success, after(0ms)) .expect(connect) .complete_with(success, after(0ms)) .reply_with(connack, after(0ms)) .expect(pubrel) .complete_with(success, after(0ms)) .reply_with(pubcomp, after(0ms)); asio::io_context ioc; auto executor = ioc.get_executor(); auto& broker = asio::make_service( ioc, executor, std::move(broker_side) ); using client_type = mqtt_client; client_type c(executor, ""); c.brokers("127.0.0.1,127.0.0.1") // to avoid reconnect backoff .run(); c.async_publish( "t", "p", retain_e::no, publish_props {}, [&](error_code ec, reason_code rc, auto) { ++handlers_called; BOOST_CHECK(!ec); BOOST_CHECK_EQUAL(rc, reason_codes::success); c.cancel(); } ); ioc.run_for(std::chrono::seconds(6)); BOOST_CHECK_EQUAL(handlers_called, expected_handlers_called); BOOST_CHECK(broker.received_all_expected()); } BOOST_AUTO_TEST_SUITE_END();