//------------------------------------------------------------------------------ /* This file is part of clio: https://github.com/XRPLF/clio Copyright (c) 2024, the clio developers. Permission to use, copy, modify, and distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ //============================================================================== #pragma once #include #include #include #include #include #include namespace util { /** * @brief CoroutineGroup is a helper class to manage a group of coroutines. It allows to spawn multiple coroutines and * wait for all of them to finish. * @note This class is safe to use from multiple threads. */ class CoroutineGroup { boost::asio::steady_timer timer_; std::optional maxChildren_; std::atomic_size_t childrenCounter_{0}; public: /** * @brief Construct a new Coroutine Group object * * @param yield The yield context to use for the internal timer * @param maxChildren The maximum number of coroutines that can be spawned at the same time. If not provided, there * is no limit */ CoroutineGroup(boost::asio::yield_context yield, std::optional maxChildren = std::nullopt); /** * @brief Destroy the Coroutine Group object * * @note asyncWait() must be called before the object is destroyed */ ~CoroutineGroup(); /** * @brief Spawn a new coroutine in the group * * @param yield The yield context to use for the coroutine (it should be the same as the one used in the * constructor) * @param fn The function to execute * @return true If the coroutine was spawned successfully. false if the maximum number of coroutines has been * reached */ bool spawn(boost::asio::yield_context yield, std::function fn); /** * @brief Register a foreign coroutine this group should wait for. * @note A foreign coroutine is still counted as a child one, i.e. calling this method increases the size of the * group. * * @param yield The yield context owning the coroutine group. * @return A callback to call on foreign coroutine completes or std::nullopt if the group is already full. */ std::optional> registerForeign(boost::asio::yield_context yield); /** * @brief Wait for all the coroutines in the group to finish * * @note This method must be called before the object is destroyed * * @param yield The yield context to use for the internal timer */ void asyncWait(boost::asio::yield_context yield); /** * @brief Get the number of coroutines in the group * * @return size_t The number of coroutines in the group */ size_t size() const; /** * @brief Check if the group is full * * @return true If the group is full false otherwise */ bool isFull() const; private: void onCoroutineCompleted(); }; } // namespace util