forked from boostorg/mqtt5
Summary: related to T13767 T13767 Reviewers: ivica Reviewed By: ivica Subscribers: miljen, iljazovic Differential Revision: https://repo.mireo.local/D29686
299 lines
7.9 KiB
C++
299 lines
7.9 KiB
C++
//
|
|
// Copyright (c) 2023-2024 Ivica Siladic, Bruno Iljazovic, Korina Simicevic
|
|
//
|
|
// Distributed under the Boost Software License, Version 1.0.
|
|
// (See accompanying file LICENSE or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
//
|
|
|
|
#ifndef ASYNC_MQTT5_DISCONNECT_OP_HPP
|
|
#define ASYNC_MQTT5_DISCONNECT_OP_HPP
|
|
|
|
#include <boost/asio/consign.hpp>
|
|
#include <boost/asio/deferred.hpp>
|
|
#include <boost/asio/prepend.hpp>
|
|
#include <boost/asio/steady_timer.hpp>
|
|
#include <boost/asio/experimental/parallel_group.hpp>
|
|
|
|
#include <async_mqtt5/types.hpp>
|
|
|
|
#include <async_mqtt5/detail/cancellable_handler.hpp>
|
|
#include <async_mqtt5/detail/control_packet.hpp>
|
|
#include <async_mqtt5/detail/internal_types.hpp>
|
|
#include <async_mqtt5/detail/topic_validation.hpp>
|
|
#include <async_mqtt5/detail/utf8_mqtt.hpp>
|
|
|
|
#include <async_mqtt5/impl/codecs/message_encoders.hpp>
|
|
|
|
namespace async_mqtt5::detail {
|
|
|
|
namespace asio = boost::asio;
|
|
|
|
template <
|
|
typename ClientService,
|
|
typename DisconnectContext
|
|
>
|
|
class disconnect_op {
|
|
using client_service = ClientService;
|
|
|
|
struct on_disconnect {};
|
|
|
|
std::shared_ptr<client_service> _svc_ptr;
|
|
DisconnectContext _context;
|
|
|
|
using handler_type = cancellable_handler<
|
|
asio::any_completion_handler<void (error_code)>,
|
|
typename ClientService::executor_type
|
|
>;
|
|
handler_type _handler;
|
|
|
|
public:
|
|
template <typename Handler>
|
|
disconnect_op(
|
|
const std::shared_ptr<client_service>& svc_ptr,
|
|
DisconnectContext&& context, Handler&& handler
|
|
) :
|
|
_svc_ptr(svc_ptr), _context(std::move(context)),
|
|
_handler(std::move(handler), _svc_ptr->get_executor())
|
|
{
|
|
auto slot = asio::get_associated_cancellation_slot(_handler);
|
|
if (slot.is_connected())
|
|
slot.assign([&svc = *_svc_ptr](asio::cancellation_type_t) {
|
|
svc.cancel();
|
|
});
|
|
}
|
|
|
|
disconnect_op(disconnect_op&&) = default;
|
|
disconnect_op(const disconnect_op&) = delete;
|
|
|
|
using executor_type = asio::associated_executor_t<handler_type>;
|
|
executor_type get_executor() const noexcept {
|
|
return asio::get_associated_executor(_handler);
|
|
}
|
|
|
|
using allocator_type = asio::associated_allocator_t<handler_type>;
|
|
allocator_type get_allocator() const noexcept {
|
|
return asio::get_associated_allocator(_handler);
|
|
}
|
|
|
|
void perform() {
|
|
error_code ec = validate_disconnect(_context.props);
|
|
if (ec)
|
|
return complete_post(ec);
|
|
|
|
auto disconnect = control_packet<allocator_type>::of(
|
|
no_pid, get_allocator(),
|
|
encoders::encode_disconnect,
|
|
static_cast<uint8_t>(_context.reason_code), _context.props
|
|
);
|
|
|
|
auto max_packet_size = _svc_ptr->connack_property(prop::maximum_packet_size)
|
|
.value_or(default_max_send_size);
|
|
if (disconnect.size() > max_packet_size)
|
|
// drop properties
|
|
return send_disconnect(control_packet<allocator_type>::of(
|
|
no_pid, get_allocator(),
|
|
encoders::encode_disconnect,
|
|
static_cast<uint8_t>(_context.reason_code), disconnect_props {}
|
|
));
|
|
|
|
send_disconnect(std::move(disconnect));
|
|
}
|
|
|
|
void send_disconnect(control_packet<allocator_type> disconnect) {
|
|
auto wire_data = disconnect.wire_data();
|
|
_svc_ptr->async_send(
|
|
wire_data,
|
|
no_serial, send_flag::terminal,
|
|
asio::prepend(
|
|
std::move(*this),
|
|
on_disconnect {}, std::move(disconnect)
|
|
)
|
|
);
|
|
}
|
|
|
|
void operator()(
|
|
on_disconnect,
|
|
control_packet<allocator_type> disconnect, error_code ec
|
|
) {
|
|
// The connection must be closed even
|
|
// if we failed to send the DISCONNECT packet
|
|
// with Reason Code of 0x80 or greater.
|
|
|
|
if (
|
|
ec == asio::error::operation_aborted ||
|
|
ec == asio::error::no_recovery
|
|
)
|
|
return complete(asio::error::operation_aborted);
|
|
|
|
if (ec == asio::error::try_again) {
|
|
if (_context.terminal)
|
|
return send_disconnect(std::move(disconnect));
|
|
return complete(error_code {});
|
|
}
|
|
|
|
if (_context.terminal) {
|
|
_svc_ptr->cancel();
|
|
return complete(error_code {});
|
|
}
|
|
|
|
_svc_ptr->close_stream();
|
|
_svc_ptr->open_stream();
|
|
|
|
complete(error_code {});
|
|
}
|
|
|
|
private:
|
|
static error_code validate_disconnect(const disconnect_props& props) {
|
|
const auto& reason_string = props[prop::reason_string];
|
|
if (
|
|
reason_string &&
|
|
validate_mqtt_utf8(*reason_string) != validation_result::valid
|
|
)
|
|
return client::error::malformed_packet;
|
|
|
|
const auto& user_properties = props[prop::user_property];
|
|
for (const auto& user_property: user_properties)
|
|
if (!is_valid_string_pair(user_property))
|
|
return client::error::malformed_packet;
|
|
return error_code {};
|
|
}
|
|
|
|
void complete(error_code ec) {
|
|
_handler.complete(ec);
|
|
}
|
|
|
|
void complete_post(error_code ec) {
|
|
_handler.complete_post(ec);
|
|
}
|
|
};
|
|
|
|
template <typename ClientService, typename Handler>
|
|
class terminal_disconnect_op {
|
|
using client_service = ClientService;
|
|
|
|
static constexpr uint8_t seconds = 5;
|
|
|
|
std::shared_ptr<client_service> _svc_ptr;
|
|
std::unique_ptr<asio::steady_timer> _timer;
|
|
|
|
using handler_type = cancellable_handler<
|
|
Handler, typename ClientService::executor_type
|
|
>;
|
|
handler_type _handler;
|
|
|
|
public:
|
|
terminal_disconnect_op(
|
|
const std::shared_ptr<client_service>& svc_ptr,
|
|
Handler&& handler
|
|
) :
|
|
_svc_ptr(svc_ptr),
|
|
_timer(new asio::steady_timer(_svc_ptr->get_executor())),
|
|
_handler(std::move(handler), _svc_ptr->get_executor())
|
|
{}
|
|
|
|
terminal_disconnect_op(terminal_disconnect_op&&) = default;
|
|
terminal_disconnect_op(const terminal_disconnect_op&) = delete;
|
|
|
|
using executor_type = asio::associated_executor_t<handler_type>;
|
|
executor_type get_executor() const noexcept {
|
|
return asio::get_associated_executor(_handler);
|
|
}
|
|
|
|
using allocator_type = asio::associated_allocator_t<handler_type>;
|
|
allocator_type get_allocator() const noexcept {
|
|
return asio::get_associated_allocator(_handler);
|
|
}
|
|
|
|
using cancellation_slot_type = asio::associated_cancellation_slot_t<handler_type>;
|
|
cancellation_slot_type get_cancellation_slot() const noexcept {
|
|
return asio::get_associated_cancellation_slot(_handler);
|
|
}
|
|
|
|
template <typename DisconnectContext>
|
|
void perform(DisconnectContext&& context) {
|
|
namespace asioex = boost::asio::experimental;
|
|
|
|
auto init_disconnect = [](
|
|
auto handler, disconnect_ctx ctx,
|
|
const std::shared_ptr<ClientService>& svc_ptr
|
|
) {
|
|
disconnect_op {
|
|
svc_ptr, std::move(ctx), std::move(handler)
|
|
}.perform();
|
|
};
|
|
|
|
_timer->expires_after(std::chrono::seconds(seconds));
|
|
|
|
auto timed_disconnect = asioex::make_parallel_group(
|
|
asio::async_initiate<const asio::deferred_t, void (error_code)>(
|
|
init_disconnect, asio::deferred,
|
|
std::forward<DisconnectContext>(context), _svc_ptr
|
|
),
|
|
_timer->async_wait(asio::deferred)
|
|
);
|
|
|
|
timed_disconnect.async_wait(
|
|
asioex::wait_for_one(), asio::prepend(std::move(*this))
|
|
);
|
|
}
|
|
|
|
void operator()(
|
|
std::array<std::size_t, 2> /* ord */,
|
|
error_code disconnect_ec, error_code /* timer_ec */
|
|
) {
|
|
_handler.complete(disconnect_ec);
|
|
}
|
|
};
|
|
|
|
template <typename ClientService, typename CompletionToken>
|
|
decltype(auto) async_disconnect(
|
|
disconnect_rc_e reason_code, const disconnect_props& props,
|
|
const std::shared_ptr<ClientService>& svc_ptr,
|
|
CompletionToken&& token
|
|
) {
|
|
using Signature = void (error_code);
|
|
|
|
auto initiation = [](
|
|
auto handler, disconnect_ctx ctx,
|
|
const std::shared_ptr<ClientService>& svc_ptr
|
|
) {
|
|
disconnect_op {
|
|
svc_ptr, std::move(ctx), std::move(handler)
|
|
}.perform();
|
|
};
|
|
|
|
return asio::async_initiate<CompletionToken, Signature>(
|
|
initiation, token,
|
|
disconnect_ctx { reason_code, props, false },
|
|
svc_ptr
|
|
);
|
|
}
|
|
|
|
template <typename ClientService, typename CompletionToken>
|
|
decltype(auto) async_terminal_disconnect(
|
|
disconnect_rc_e reason_code, const disconnect_props& props,
|
|
const std::shared_ptr<ClientService>& svc_ptr,
|
|
CompletionToken&& token
|
|
) {
|
|
using Signature = void (error_code);
|
|
|
|
auto initiation = [](
|
|
auto handler, disconnect_ctx ctx,
|
|
const std::shared_ptr<ClientService>& svc_ptr
|
|
) {
|
|
terminal_disconnect_op {
|
|
svc_ptr, std::move(handler)
|
|
}.perform(std::move(ctx));
|
|
};
|
|
|
|
return asio::async_initiate<CompletionToken, Signature>(
|
|
initiation, token,
|
|
disconnect_ctx { reason_code, props, true },
|
|
svc_ptr
|
|
);
|
|
}
|
|
|
|
} // end namespace async_mqtt5::detail
|
|
|
|
#endif // !ASYNC_MQTT5_DISCONNECT_HPP
|