// // Copyright (w) 2016-2017 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) // // Official repository: https://github.com/boostorg/beast // // Test that header file is self-contained. #include #include "test.hpp" namespace boost { namespace beast { namespace websocket { class ping_test : public websocket_test_suite { public: template void doTestPing(Wrap const& w) { permessage_deflate pmd; pmd.client_enable = false; pmd.server_enable = false; // ping doTest(pmd, [&](ws_type& ws) { w.ping(ws, {}); }); // pong doTest(pmd, [&](ws_type& ws) { w.pong(ws, {}); }); // ping, already closed { echo_server es{log}; stream ws{ioc_}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); ws.close({}); try { w.ping(ws, {}); fail("", __FILE__, __LINE__); } catch(system_error const& se) { BEAST_EXPECTS( se.code() == boost::asio::error::operation_aborted, se.code().message()); } } // pong, already closed { echo_server es{log}; stream ws{ioc_}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); ws.close({}); try { w.pong(ws, {}); fail("", __FILE__, __LINE__); } catch(system_error const& se) { BEAST_EXPECTS( se.code() == boost::asio::error::operation_aborted, se.code().message()); } } } void testPing() { doTestPing(SyncClient{}); yield_to([&](yield_context yield) { doTestPing(AsyncClient{yield}); }); } void testSuspend() { // suspend on write doFailLoop([&](test::fail_counter& fc) { echo_server es{log}; boost::asio::io_context ioc; stream ws{ioc, fc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); std::size_t count = 0; ws.async_write(sbuf("Hello, world"), [&](error_code ec, std::size_t n) { ++count; if(ec) BOOST_THROW_EXCEPTION( system_error{ec}); BEAST_EXPECT(n == 12); }); BEAST_EXPECT(ws.wr_block_.is_locked()); BEAST_EXPECT(count == 0); ws.async_ping({}, [&](error_code ec) { ++count; if(ec) BOOST_THROW_EXCEPTION( system_error{ec}); }); BEAST_EXPECT(count == 0); ioc.run(); BEAST_EXPECT(count == 2); }); // suspend on close doFailLoop([&](test::fail_counter& fc) { echo_server es{log}; boost::asio::io_context ioc; stream ws{ioc, fc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); std::size_t count = 0; ws.async_close({}, [&](error_code ec) { ++count; if(ec) BOOST_THROW_EXCEPTION( system_error{ec}); }); BEAST_EXPECT(ws.wr_block_.is_locked()); BEAST_EXPECT(count == 0); ws.async_ping({}, [&](error_code ec) { ++count; if(ec != boost::asio::error::operation_aborted) BOOST_THROW_EXCEPTION( system_error{ec}); }); BEAST_EXPECT(count == 0); ioc.run(); BEAST_EXPECT(count == 2); }); // suspend on read ping + message doFailLoop([&](test::fail_counter& fc) { echo_server es{log}; boost::asio::io_context ioc; stream ws{ioc, fc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); // add a ping and message to the input ws.next_layer().append(string_view{ "\x89\x00" "\x81\x01*", 5}); std::size_t count = 0; multi_buffer b; ws.async_read(b, [&](error_code ec, std::size_t) { ++count; if(ec) BOOST_THROW_EXCEPTION( system_error{ec}); }); while(! ws.wr_block_.is_locked()) { ioc.run_one(); if(! BEAST_EXPECT(! ioc.stopped())) break; } BEAST_EXPECT(count == 0); ws.async_ping({}, [&](error_code ec) { ++count; if(ec) BOOST_THROW_EXCEPTION( system_error{ec}); }); BEAST_EXPECT(count == 0); ioc.run(); BEAST_EXPECT(count == 2); }); // suspend on read bad message doFailLoop([&](test::fail_counter& fc) { echo_server es{log}; boost::asio::io_context ioc; stream ws{ioc, fc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); // add an invalid frame to the input ws.next_layer().append(string_view{ "\x09\x00", 2}); std::size_t count = 0; multi_buffer b; ws.async_read(b, [&](error_code ec, std::size_t) { ++count; if(ec != error::bad_control_fragment) BOOST_THROW_EXCEPTION( system_error{ec}); }); while(! ws.wr_block_.is_locked()) { ioc.run_one(); if(! BEAST_EXPECT(! ioc.stopped())) break; } BEAST_EXPECT(count == 0); ws.async_ping({}, [&](error_code ec) { ++count; if(ec != boost::asio::error::operation_aborted) BOOST_THROW_EXCEPTION( system_error{ec}); }); BEAST_EXPECT(count == 0); ioc.run(); BEAST_EXPECT(count == 2); }); // suspend on read close #1 doFailLoop([&](test::fail_counter& fc) { echo_server es{log}; boost::asio::io_context ioc; stream ws{ioc, fc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); // add a close frame to the input ws.next_layer().append(string_view{ "\x88\x00", 2}); std::size_t count = 0; multi_buffer b; ws.async_read(b, [&](error_code ec, std::size_t) { ++count; if(ec != error::closed) BOOST_THROW_EXCEPTION( system_error{ec}); }); while(! ws.wr_block_.is_locked()) { ioc.run_one(); if(! BEAST_EXPECT(! ioc.stopped())) break; } BEAST_EXPECT(count == 0); ws.async_ping({}, [&](error_code ec) { ++count; if(ec != boost::asio::error::operation_aborted) BOOST_THROW_EXCEPTION( system_error{ec}); }); BEAST_EXPECT(count == 0); ioc.run(); BEAST_EXPECT(count == 2); }); // suspend on read close #2 doFailLoop([&](test::fail_counter& fc) { echo_server es{log, kind::async}; boost::asio::io_context ioc; stream ws{ioc, fc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); // Cause close to be received es.async_close(); std::size_t count = 0; multi_buffer b; ws.async_read(b, [&](error_code ec, std::size_t) { ++count; if(ec != error::closed) BOOST_THROW_EXCEPTION( system_error{ec}); }); while(! ws.wr_block_.is_locked()) { ioc.run_one(); if(! BEAST_EXPECT(! ioc.stopped())) break; } BEAST_EXPECT(count == 0); ws.async_ping({}, [&](error_code ec) { ++count; if(ec != boost::asio::error::operation_aborted) BOOST_THROW_EXCEPTION( system_error{ec}); }); BEAST_EXPECT(count == 0); ioc.run(); BEAST_EXPECT(count == 2); }); // don't ping on close doFailLoop([&](test::fail_counter& fc) { echo_server es{log}; error_code ec; boost::asio::io_context ioc; stream ws{ioc, fc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); std::size_t count = 0; ws.async_write(sbuf("*"), [&](error_code ec, std::size_t n) { ++count; if(ec) BOOST_THROW_EXCEPTION( system_error{ec}); BEAST_EXPECT(n == 1); }); BEAST_EXPECT(ws.wr_block_.is_locked()); ws.async_ping("", [&](error_code ec) { ++count; if(ec != boost::asio::error::operation_aborted) BOOST_THROW_EXCEPTION( system_error{ec}); }); ws.async_close({}, [&](error_code) { ++count; if(ec) BOOST_THROW_EXCEPTION( system_error{ec}); }); ioc.run(); BEAST_EXPECT(count == 3); }); { echo_server es{log, kind::async}; boost::asio::io_context ioc; stream ws{ioc}; ws.next_layer().connect(es.stream()); ws.handshake("localhost", "/"); // Cause close to be received es.async_close(); multi_buffer b; std::size_t count = 0; // Read a close frame. // Sends a close frame, blocking writes. ws.async_read(b, [&](error_code ec, std::size_t) { // Read should complete with error::closed ++count; BEAST_EXPECTS(ec == error::closed, ec.message()); }); if(! BEAST_EXPECT(run_until(ioc, 100, [&]{ return ws.wr_close_; }))) return; // Try to ping ws.async_ping("payload", [&](error_code ec) { // Pings after a close are aborted ++count; BEAST_EXPECTS(ec == boost::asio:: error::operation_aborted, ec.message()); // Subsequent calls to close are aborted ws.async_close({}, [&](error_code ec) { ++count; BEAST_EXPECTS(ec == boost::asio:: error::operation_aborted, ec.message()); }); }); static std::size_t constexpr limit = 100; std::size_t n; for(n = 0; n < limit; ++n) { if(count >= 3) break; ioc.run_one(); } BEAST_EXPECT(n < limit); ioc.run(); } } void testContHook() { struct handler { void operator()(error_code) {} }; stream ws{ioc_}; stream::ping_op op{ handler{}, ws, detail::opcode::ping, {}}; using boost::asio::asio_handler_is_continuation; asio_handler_is_continuation(&op); } void run() override { testPing(); testSuspend(); testContHook(); } }; BEAST_DEFINE_TESTSUITE(beast,websocket,ping); } // websocket } // beast } // boost