| 
									
										
										
										
											2023-10-05 13:59:32 +02:00
										 |  |  | #ifndef ASYNC_MQTT5_PING_OP_HPP
 | 
					
						
							|  |  |  | #define ASYNC_MQTT5_PING_OP_HPP
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <chrono>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <boost/asio/consign.hpp>
 | 
					
						
							| 
									
										
										
										
											2023-10-06 11:51:04 +02:00
										 |  |  | #include <boost/asio/prepend.hpp>
 | 
					
						
							| 
									
										
										
										
											2023-10-05 13:59:32 +02:00
										 |  |  | #include <boost/asio/recycling_allocator.hpp>
 | 
					
						
							|  |  |  | #include <boost/asio/steady_timer.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <async_mqtt5/detail/control_packet.hpp>
 | 
					
						
							| 
									
										
										
										
											2023-10-06 11:51:04 +02:00
										 |  |  | #include <async_mqtt5/detail/internal_types.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-07 09:32:34 +01:00
										 |  |  | #include <async_mqtt5/impl/codecs/message_encoders.hpp>
 | 
					
						
							| 
									
										
										
										
											2023-10-05 13:59:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace async_mqtt5::detail { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace asio = boost::asio; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename ClientService> | 
					
						
							|  |  |  | class ping_op { | 
					
						
							|  |  |  | 	using client_service = ClientService; | 
					
						
							|  |  |  | 	struct on_timer {}; | 
					
						
							|  |  |  | 	struct on_pingreq {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	static constexpr auto ping_interval = std::chrono::seconds(5); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	std::shared_ptr<client_service> _svc_ptr; | 
					
						
							|  |  |  | 	std::unique_ptr<asio::steady_timer> _ping_timer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2023-12-07 09:32:34 +01:00
										 |  |  | 	ping_op(const std::shared_ptr<client_service>& svc_ptr) : | 
					
						
							| 
									
										
										
										
											2023-10-05 13:59:32 +02:00
										 |  |  | 		_svc_ptr(svc_ptr), | 
					
						
							|  |  |  | 		_ping_timer(new asio::steady_timer(svc_ptr->get_executor())) | 
					
						
							|  |  |  | 	{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ping_op(ping_op&&) noexcept = default; | 
					
						
							|  |  |  | 	ping_op(const ping_op&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	using executor_type = typename client_service::executor_type; | 
					
						
							|  |  |  | 	executor_type get_executor() const noexcept { | 
					
						
							|  |  |  | 		return _svc_ptr->get_executor(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	using allocator_type = asio::recycling_allocator<void>; | 
					
						
							|  |  |  | 	allocator_type get_allocator() const noexcept { | 
					
						
							|  |  |  | 		return allocator_type {}; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	using cancellation_slot_type = asio::cancellation_slot; | 
					
						
							|  |  |  | 	asio::cancellation_slot get_cancellation_slot() const noexcept { | 
					
						
							|  |  |  | 		return _svc_ptr->_cancel_ping.slot(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void perform(duration from_now) { | 
					
						
							|  |  |  | 		_ping_timer->expires_from_now(from_now); | 
					
						
							|  |  |  | 		_ping_timer->async_wait( | 
					
						
							|  |  |  | 			asio::prepend(std::move(*this), on_timer {}) | 
					
						
							|  |  |  | 		); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void operator()(on_timer, error_code ec) { | 
					
						
							|  |  |  | 		get_cancellation_slot().clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (ec == asio::error::operation_aborted || !_svc_ptr->is_open()) | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		auto pingreq = control_packet<allocator_type>::of( | 
					
						
							|  |  |  | 			no_pid, get_allocator(), encoders::encode_pingreq | 
					
						
							|  |  |  | 		); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-22 10:07:20 +01:00
										 |  |  | 		auto wire_data = pingreq.wire_data(); | 
					
						
							| 
									
										
										
										
											2023-10-05 13:59:32 +02:00
										 |  |  | 		_svc_ptr->async_send( | 
					
						
							|  |  |  | 			wire_data, | 
					
						
							|  |  |  | 			no_serial, send_flag::none, | 
					
						
							|  |  |  | 			asio::consign( | 
					
						
							|  |  |  | 				asio::prepend(std::move(*this), on_pingreq {}), | 
					
						
							|  |  |  | 				std::move(pingreq) | 
					
						
							|  |  |  | 			) | 
					
						
							|  |  |  | 		); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void operator()(on_pingreq, error_code ec) { | 
					
						
							|  |  |  | 		get_cancellation_slot().clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!ec || ec == asio::error::try_again) | 
					
						
							|  |  |  | 			perform(ping_interval - std::chrono::seconds(1)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // end namespace async_mqtt5::detail
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif // !ASYNC_MQTT5_PING_OP_HPP
 |