// // Copyright (c) 2013-2016 Vinnie Falco (vinnie dot falco at gmail dot com) // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #ifndef BEAST_WEBSOCKET_IMPL_PING_IPP #define BEAST_WEBSOCKET_IMPL_PING_IPP #include #include #include #include #include #include namespace beast { namespace websocket { //------------------------------------------------------------------------------ // write a ping frame // template template class stream::ping_op { struct data : op { bool cont; stream& ws; detail::frame_streambuf fb; int state = 0; data(Handler& handler, stream& ws_, opcode op_, ping_data const& payload) : cont(beast_asio_helpers:: is_continuation(handler)) , ws(ws_) { using boost::asio::buffer; using boost::asio::buffer_copy; ws.template write_ping< static_streambuf>(fb, op_, payload); } }; handler_ptr d_; public: ping_op(ping_op&&) = default; ping_op(ping_op const&) = default; template ping_op(DeducedHandler&& h, stream& ws, Args&&... args) : d_(std::forward(h), ws, std::forward(args)...) { (*this)(error_code{}, false); } void operator()() { (*this)(error_code{}); } void operator()(error_code ec, std::size_t); void operator()(error_code ec, bool again = true); friend void* asio_handler_allocate( std::size_t size, ping_op* op) { return beast_asio_helpers:: allocate(size, op->d_.handler()); } friend void asio_handler_deallocate( void* p, std::size_t size, ping_op* op) { return beast_asio_helpers:: deallocate(p, size, op->d_.handler()); } friend bool asio_handler_is_continuation(ping_op* op) { return op->d_->cont; } template friend void asio_handler_invoke(Function&& f, ping_op* op) { return beast_asio_helpers:: invoke(f, op->d_.handler()); } }; template template void stream::ping_op:: operator()(error_code ec, std::size_t) { auto& d = *d_; if(ec) d.ws.failed_ = true; (*this)(ec); } template template void stream:: ping_op:: operator()(error_code ec, bool again) { auto& d = *d_; d.cont = d.cont || again; if(ec) goto upcall; for(;;) { switch(d.state) { case 0: if(d.ws.wr_block_) { // suspend d.state = 2; d.ws.wr_op_.template emplace< ping_op>(std::move(*this)); return; } if(d.ws.failed_ || d.ws.wr_close_) { // call handler d.state = 99; d.ws.get_io_service().post( bind_handler(std::move(*this), boost::asio::error::operation_aborted)); return; } // fall through case 1: // send ping frame d.state = 99; BOOST_ASSERT(! d.ws.wr_block_); d.ws.wr_block_ = &d; boost::asio::async_write(d.ws.stream_, d.fb.data(), std::move(*this)); return; case 2: d.state = 3; d.ws.get_io_service().post( bind_handler(std::move(*this), ec)); return; case 3: if(d.ws.failed_ || d.ws.wr_close_) { // call handler ec = boost::asio::error::operation_aborted; goto upcall; } d.state = 1; break; case 99: goto upcall; } } upcall: if(d.ws.wr_block_ == &d) d.ws.wr_block_ = nullptr; d.ws.rd_op_.maybe_invoke(); d_.invoke(ec); } template template typename async_completion< WriteHandler, void(error_code)>::result_type stream:: async_ping(ping_data const& payload, WriteHandler&& handler) { static_assert(is_AsyncStream::value, "AsyncStream requirements requirements not met"); beast::async_completion< WriteHandler, void(error_code) > completion(handler); ping_op{ completion.handler, *this, opcode::ping, payload}; return completion.result.get(); } template template typename async_completion< WriteHandler, void(error_code)>::result_type stream:: async_pong(ping_data const& payload, WriteHandler&& handler) { static_assert(is_AsyncStream::value, "AsyncStream requirements requirements not met"); beast::async_completion< WriteHandler, void(error_code) > completion(handler); ping_op{ completion.handler, *this, opcode::pong, payload}; return completion.result.get(); } template void stream:: ping(ping_data const& payload) { error_code ec; ping(payload, ec); if(ec) throw system_error{ec}; } template void stream:: ping(ping_data const& payload, error_code& ec) { detail::frame_streambuf db; write_ping( db, opcode::ping, payload); boost::asio::write(stream_, db.data(), ec); } template void stream:: pong(ping_data const& payload) { error_code ec; pong(payload, ec); if(ec) throw system_error{ec}; } template void stream:: pong(ping_data const& payload, error_code& ec) { detail::frame_streambuf db; write_ping( db, opcode::pong, payload); boost::asio::write(stream_, db.data(), ec); } //------------------------------------------------------------------------------ } // websocket } // beast #endif