2023-12-07 09:32:34 +01:00
|
|
|
#include <boost/test/unit_test.hpp>
|
|
|
|
|
2024-01-04 15:31:06 +01:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
2023-12-07 09:32:34 +01:00
|
|
|
#include <boost/asio/ip/tcp.hpp>
|
|
|
|
#include <boost/asio/ssl/stream.hpp>
|
|
|
|
#include <boost/asio/io_context.hpp>
|
|
|
|
|
|
|
|
#include <boost/beast/websocket/stream.hpp>
|
|
|
|
|
2024-01-04 15:31:06 +01:00
|
|
|
#include <boost/type_traits/remove_cv_ref.hpp>
|
|
|
|
|
2023-12-07 09:32:34 +01:00
|
|
|
#include <async_mqtt5/detail/any_authenticator.hpp>
|
|
|
|
#include <async_mqtt5/detail/async_traits.hpp>
|
|
|
|
|
|
|
|
#include <async_mqtt5/mqtt_client.hpp>
|
|
|
|
|
|
|
|
using namespace async_mqtt5;
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_SUITE(traits/*, *boost::unit_test::disabled()*/)
|
|
|
|
|
2024-01-18 10:37:13 +01:00
|
|
|
struct good_authenticator {
|
2023-12-07 09:32:34 +01:00
|
|
|
good_authenticator() = default;
|
|
|
|
|
|
|
|
template <typename CompletionToken>
|
|
|
|
decltype(auto) async_auth(
|
|
|
|
auth_step_e step, std::string data,
|
|
|
|
CompletionToken&& token
|
|
|
|
) {
|
|
|
|
using error_code = boost::system::error_code;
|
|
|
|
using Signature = void(error_code, std::string);
|
|
|
|
|
|
|
|
auto initiate = [](auto, auth_step_e, std::string) {};
|
|
|
|
|
|
|
|
return asio::async_initiate<CompletionToken, Signature>(
|
|
|
|
initiate, token, step, std::move(data)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string_view method() const {
|
|
|
|
return "method";
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-01-18 10:37:13 +01:00
|
|
|
struct bad_authenticator {
|
2023-12-07 09:32:34 +01:00
|
|
|
bad_authenticator() = default;
|
|
|
|
|
2024-01-04 15:31:06 +01:00
|
|
|
void async_auth(std::string /* data */) {}
|
2023-12-07 09:32:34 +01:00
|
|
|
|
|
|
|
std::string_view method() const {
|
|
|
|
return "method";
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(is_authenticator) {
|
|
|
|
BOOST_STATIC_ASSERT(detail::is_authenticator<good_authenticator>);
|
|
|
|
BOOST_STATIC_ASSERT(!detail::is_authenticator<bad_authenticator>);
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace asio = boost::asio;
|
|
|
|
namespace beast = boost::beast;
|
|
|
|
|
|
|
|
using tcp_layer = asio::ip::tcp::socket;
|
|
|
|
using tls_layer = asio::ssl::stream<asio::ip::tcp::socket>;
|
|
|
|
using websocket_tcp_layer = beast::websocket::stream<tcp_layer>;
|
|
|
|
using websocket_tls_layer = beast::websocket::stream<tls_layer>;
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(async_traits) {
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_next_layer<tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_next_layer<tls_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_next_layer<websocket_tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_next_layer<websocket_tls_layer>);
|
|
|
|
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_tls_layer<tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_tls_layer<tls_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_tls_layer<websocket_tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_tls_layer<websocket_tls_layer>);
|
|
|
|
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_ws_handshake<tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_ws_handshake<tls_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_ws_handshake<websocket_tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_ws_handshake<websocket_tls_layer>);
|
|
|
|
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_tls_handshake<tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(detail::has_tls_handshake<tls_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_tls_handshake<websocket_tcp_layer>);
|
|
|
|
BOOST_STATIC_ASSERT(!detail::has_tls_handshake<websocket_tls_layer>);
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(client_functions) {
|
|
|
|
asio::io_context ioc;
|
|
|
|
|
|
|
|
mqtt_client<tcp_layer> tcp_client(ioc, "");
|
|
|
|
tcp_client.authenticator(good_authenticator());
|
2023-12-22 09:17:45 +01:00
|
|
|
|
2024-01-04 15:31:06 +01:00
|
|
|
connack_props ca_props;
|
|
|
|
ca_props.visit([&tcp_client](const auto& p, auto&) -> bool {
|
|
|
|
using ptype = boost::remove_cv_ref_t<decltype(p)>;
|
2024-01-05 13:37:20 +01:00
|
|
|
[[maybe_unused]] prop::value_type_t<ptype::value> value = tcp_client.connack_property(p);
|
2024-01-04 15:31:06 +01:00
|
|
|
return true;
|
|
|
|
});
|
|
|
|
|
|
|
|
connack_props ret_ca_props = tcp_client.connack_properties();
|
|
|
|
|
|
|
|
connect_props co_props;
|
|
|
|
co_props[prop::maximum_packet_size] = 1234;
|
|
|
|
tcp_client.connect_properties(std::move(co_props));
|
|
|
|
|
|
|
|
tcp_client.connect_property(prop::session_expiry_interval, 40);
|
|
|
|
tcp_client.connect_property(prop::receive_maximum, int16_t(10123));
|
|
|
|
tcp_client.connect_property(prop::maximum_packet_size, 103);
|
|
|
|
tcp_client.connect_property(prop::topic_alias_maximum, uint16_t(12345));
|
|
|
|
tcp_client.connect_property(prop::request_response_information, uint8_t(1));
|
|
|
|
tcp_client.connect_property(prop::request_problem_information, uint8_t(0));
|
|
|
|
tcp_client.connect_property(prop::user_property, std::vector<std::string> { "prop", "prop" });
|
|
|
|
tcp_client.connect_property(prop::authentication_method, "method");
|
|
|
|
tcp_client.connect_property(prop::authentication_data, "data");
|
2023-12-07 09:32:34 +01:00
|
|
|
|
|
|
|
asio::ssl::context ctx(asio::ssl::context::tls_client);
|
|
|
|
mqtt_client<
|
|
|
|
tls_layer, asio::ssl::context
|
|
|
|
> tls_client(ioc.get_executor(), "", std::move(ctx));
|
|
|
|
tls_client.tls_context();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_SUITE_END();
|