2
0
mirror of https://github.com/boostorg/asio.git synced 2026-01-19 16:12:09 +00:00
Files
asio/test/spawn.cpp
Christopher Kohlhoff 1afbc5c12b Update copyright notices.
2025-03-04 22:57:26 +11:00

598 lines
11 KiB
C++

//
// spawn.cpp
// ~~~~~~~~~
//
// Copyright (c) 2003-2025 Christopher M. Kohlhoff (chris at kohlhoff dot com)
// Copyright (c) 2024 Casey Bodley (cbodley at redhat 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)
//
// Disable autolinking for unit tests.
#if !defined(BOOST_ALL_NO_LIB)
#define BOOST_ALL_NO_LIB 1
#endif // !defined(BOOST_ALL_NO_LIB)
// Test that header file is self-contained.
#include <boost/asio/spawn.hpp>
#include "unit_test.hpp"
#if defined(BOOST_ASIO_HAS_BOOST_CONTEXT_FIBER)
#include <memory>
#include <stdexcept>
#include "archetypes/async_ops.hpp"
#include <boost/asio/any_completion_handler.hpp>
#include <boost/asio/bind_cancellation_slot.hpp>
#include <boost/asio/deferred.hpp>
#include <boost/asio/io_context.hpp>
#include <boost/asio/steady_timer.hpp>
void void_returning_coroutine(boost::asio::yield_context)
{
}
int int_returning_coroutine(boost::asio::yield_context)
{
return 42;
}
void test_spawn_with_any_completion_handler()
{
boost::asio::io_context ctx;
bool called = false;
boost::asio::spawn(ctx, void_returning_coroutine,
boost::asio::any_completion_handler<void(std::exception_ptr)>(
[&](std::exception_ptr)
{
called = true;
}));
BOOST_ASIO_CHECK(!called);
ctx.run();
BOOST_ASIO_CHECK(called);
int result = 0;
boost::asio::spawn(ctx, int_returning_coroutine,
boost::asio::any_completion_handler<void(std::exception_ptr, int)>(
[&](std::exception_ptr, int i)
{
result = i;
}));
BOOST_ASIO_CHECK(result == 0);
ctx.restart();
ctx.run();
BOOST_ASIO_CHECK(result == 42);
}
void test_spawn_deferred()
{
boost::asio::io_context ctx;
{
bool called = false;
auto fn = boost::asio::spawn(ctx, void_returning_coroutine, boost::asio::deferred);
fn([&](std::exception_ptr)
{
called = true;
});
BOOST_ASIO_CHECK(!called);
ctx.poll();
BOOST_ASIO_CHECK(ctx.stopped());
BOOST_ASIO_CHECK(called);
}
{
int result = 0;
auto fn = boost::asio::spawn(ctx, int_returning_coroutine, boost::asio::deferred);
fn([&](std::exception_ptr, int i)
{
result = i;
});
BOOST_ASIO_CHECK(result == 0);
ctx.restart();
ctx.poll();
BOOST_ASIO_CHECK(ctx.stopped());
BOOST_ASIO_CHECK(result == 42);
}
}
void sleeping_coroutine(boost::asio::yield_context yield)
{
boost::asio::steady_timer timer(yield.get_executor(),
boost::asio::steady_timer::time_point::max());
timer.async_wait(yield);
}
void test_spawn_cancel()
{
boost::asio::cancellation_signal sig;
boost::asio::io_context ctx;
std::exception_ptr result = nullptr;
bool called = false;
boost::asio::spawn(ctx, sleeping_coroutine,
boost::asio::bind_cancellation_slot(sig.slot(),
[&](std::exception_ptr e)
{
result = e;
called = true;
}));
ctx.poll();
BOOST_ASIO_CHECK(!ctx.stopped());
BOOST_ASIO_CHECK(!called);
BOOST_ASIO_CHECK(result == nullptr);
sig.emit(boost::asio::cancellation_type::all);
ctx.poll();
BOOST_ASIO_CHECK(ctx.stopped());
BOOST_ASIO_CHECK(called);
BOOST_ASIO_CHECK(result != nullptr);
try
{
std::rethrow_exception(result);
}
catch (const boost::system::system_error& e)
{
BOOST_ASIO_CHECK(e.code() == boost::asio::error::operation_aborted);
}
catch (...)
{
BOOST_ASIO_ERROR("expected system_error");
}
}
void throwing_coroutine(boost::asio::yield_context)
{
throw std::runtime_error("oops");
}
void test_spawn_exception()
{
boost::asio::io_context ctx;
std::exception_ptr result = nullptr;
bool called = false;
boost::asio::spawn(ctx, throwing_coroutine,
[&](std::exception_ptr e)
{
result = e;
called = true;
});
ctx.poll();
BOOST_ASIO_CHECK(ctx.stopped());
BOOST_ASIO_CHECK(called);
BOOST_ASIO_CHECK(result != nullptr);
try
{
std::rethrow_exception(result);
}
catch (const std::runtime_error&)
{
// ok
}
catch (...)
{
BOOST_ASIO_ERROR("expected runtime_error");
}
}
std::unique_ptr<int> factory_coroutine(boost::asio::yield_context)
{
return std::unique_ptr<int>(new int(42));
}
void test_spawn_return_move_only()
{
boost::asio::io_context ctx;
std::unique_ptr<int> result;
bool called = false;
boost::asio::spawn(ctx, factory_coroutine,
[&](std::exception_ptr, std::unique_ptr<int> r)
{
result = std::move(r);
called = true;
});
ctx.poll();
BOOST_ASIO_CHECK(ctx.stopped());
BOOST_ASIO_CHECK(called);
BOOST_ASIO_CHECK(result);
BOOST_ASIO_CHECK(*result == 42);
}
int coroutine_using_async_ops_0(boost::asio::yield_context yield)
{
using namespace archetypes;
try
{
async_op_0(yield);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
async_op_ec_0(true, yield);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
async_op_ec_0(false, yield);
BOOST_ASIO_CHECK(false);
}
catch (boost::system::system_error& e)
{
BOOST_ASIO_CHECK(e.code() == boost::asio::error::operation_aborted);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
async_op_ex_0(true, yield);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
async_op_ex_0(false, yield);
BOOST_ASIO_CHECK(false);
}
catch (std::exception& e)
{
BOOST_ASIO_CHECK(e.what() == std::string("blah"));
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
return 123;
}
int coroutine_using_async_ops_1(boost::asio::yield_context yield)
{
using namespace archetypes;
try
{
int i = async_op_1(yield);
BOOST_ASIO_CHECK(i == 42);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i = async_op_ec_1(true, yield);
BOOST_ASIO_CHECK(i == 42);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i = async_op_ec_1(false, yield);
BOOST_ASIO_CHECK(false);
(void)i;
}
catch (boost::system::system_error& e)
{
BOOST_ASIO_CHECK(e.code() == boost::asio::error::operation_aborted);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i = async_op_ex_1(true, yield);
BOOST_ASIO_CHECK(i == 42);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i = async_op_ex_1(false, yield);
BOOST_ASIO_CHECK(false);
(void)i;
}
catch (std::exception& e)
{
BOOST_ASIO_CHECK(e.what() == std::string("blah"));
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
return 234;
}
int coroutine_using_async_ops_2(boost::asio::yield_context yield)
{
using namespace archetypes;
try
{
int i;
double d;
std::tie(i, d) = async_op_2(yield);
BOOST_ASIO_CHECK(i == 42);
BOOST_ASIO_CHECK(d == 2.0);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i;
double d;
std::tie(i, d) = async_op_ec_2(true, yield);
BOOST_ASIO_CHECK(i == 42);
BOOST_ASIO_CHECK(d == 2.0);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
std::tuple<int, double> t = async_op_ec_2(false, yield);
BOOST_ASIO_CHECK(false);
(void)t;
}
catch (boost::system::system_error& e)
{
BOOST_ASIO_CHECK(e.code() == boost::asio::error::operation_aborted);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i;
double d;
std::tie(i, d) = async_op_ex_2(true, yield);
BOOST_ASIO_CHECK(i == 42);
BOOST_ASIO_CHECK(d == 2.0);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
std::tuple<int, double> t = async_op_ex_2(false, yield);
BOOST_ASIO_CHECK(false);
(void)t;
}
catch (std::exception& e)
{
BOOST_ASIO_CHECK(e.what() == std::string("blah"));
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
return 345;
}
int coroutine_using_async_ops_3(boost::asio::yield_context yield)
{
using namespace archetypes;
try
{
int i;
double d;
char c;
std::tie(i, d, c) = async_op_3(yield);
BOOST_ASIO_CHECK(i == 42);
BOOST_ASIO_CHECK(d == 2.0);
BOOST_ASIO_CHECK(c == 'a');
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i;
double d;
char c;
std::tie(i, d, c) = async_op_ec_3(true, yield);
BOOST_ASIO_CHECK(i == 42);
BOOST_ASIO_CHECK(d == 2.0);
BOOST_ASIO_CHECK(c == 'a');
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
std::tuple<int, double, char> t = async_op_ec_3(false, yield);
BOOST_ASIO_CHECK(false);
(void)t;
}
catch (boost::system::system_error& e)
{
BOOST_ASIO_CHECK(e.code() == boost::asio::error::operation_aborted);
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
int i;
double d;
char c;
std::tie(i, d, c) = async_op_ex_3(true, yield);
BOOST_ASIO_CHECK(i == 42);
BOOST_ASIO_CHECK(d == 2.0);
BOOST_ASIO_CHECK(c == 'a');
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
try
{
std::tuple<int, double, char> t = async_op_ex_3(false, yield);
BOOST_ASIO_CHECK(false);
(void)t;
}
catch (std::exception& e)
{
BOOST_ASIO_CHECK(e.what() == std::string("blah"));
}
catch (...)
{
BOOST_ASIO_CHECK(false);
}
return 456;
}
void test_spawn_async_ops()
{
boost::asio::io_context ctx;
bool called = false;
boost::asio::spawn(ctx, coroutine_using_async_ops_0,
[&](std::exception_ptr, int i)
{
BOOST_ASIO_CHECK(i == 123);
called = true;
});
BOOST_ASIO_CHECK(!called);
ctx.run();
BOOST_ASIO_CHECK(called);
called = false;
boost::asio::spawn(ctx, coroutine_using_async_ops_1,
[&](std::exception_ptr, int i)
{
BOOST_ASIO_CHECK(i == 234);
called = true;
});
BOOST_ASIO_CHECK(!called);
ctx.restart();
ctx.run();
BOOST_ASIO_CHECK(called);
called = false;
boost::asio::spawn(ctx, coroutine_using_async_ops_2,
[&](std::exception_ptr, int i)
{
BOOST_ASIO_CHECK(i == 345);
called = true;
});
BOOST_ASIO_CHECK(!called);
ctx.restart();
ctx.run();
BOOST_ASIO_CHECK(called);
called = false;
boost::asio::spawn(ctx, coroutine_using_async_ops_3,
[&](std::exception_ptr, int i)
{
BOOST_ASIO_CHECK(i == 456);
called = true;
});
BOOST_ASIO_CHECK(!called);
ctx.restart();
ctx.run();
BOOST_ASIO_CHECK(called);
}
BOOST_ASIO_TEST_SUITE
(
"spawn",
BOOST_ASIO_TEST_CASE(test_spawn_with_any_completion_handler)
BOOST_ASIO_TEST_CASE(test_spawn_deferred)
BOOST_ASIO_TEST_CASE(test_spawn_cancel)
BOOST_ASIO_TEST_CASE(test_spawn_exception)
BOOST_ASIO_TEST_CASE(test_spawn_return_move_only)
BOOST_ASIO_TEST_CASE(test_spawn_async_ops)
)
#else // defined(BOOST_ASIO_HAS_BOOST_CONTEXT_FIBER)
BOOST_ASIO_TEST_SUITE
(
"spawn",
BOOST_ASIO_TEST_CASE(null_test)
)
#endif // defined(BOOST_ASIO_HAS_BOOST_CONTEXT_FIBER)