mirror of
https://github.com/XRPLF/clio.git
synced 2026-06-04 17:26:49 +00:00
172 lines
3.9 KiB
C++
172 lines
3.9 KiB
C++
#pragma once
|
|
|
|
#include <boost/asio/error.hpp>
|
|
#include <boost/asio/io_context.hpp>
|
|
#include <boost/asio/steady_timer.hpp>
|
|
#include <boost/asio/strand.hpp>
|
|
|
|
#include <atomic>
|
|
#include <chrono>
|
|
#include <cstddef>
|
|
#include <memory>
|
|
|
|
namespace util {
|
|
|
|
/**
|
|
* @brief Interface for retry strategies
|
|
*/
|
|
class RetryStrategy {
|
|
std::chrono::steady_clock::duration initialDelay_;
|
|
std::chrono::steady_clock::duration delay_;
|
|
|
|
public:
|
|
/**
|
|
* @brief Construct a new Retry Strategy object
|
|
*
|
|
* @param delay The initial delay value
|
|
*/
|
|
RetryStrategy(std::chrono::steady_clock::duration delay);
|
|
virtual ~RetryStrategy() = default;
|
|
|
|
/**
|
|
* @return The current delay value
|
|
*/
|
|
[[nodiscard]] std::chrono::steady_clock::duration
|
|
getDelay() const;
|
|
|
|
/**
|
|
* @brief Increase the delay value
|
|
*/
|
|
void
|
|
increaseDelay();
|
|
|
|
/**
|
|
* @brief Reset the delay value
|
|
*/
|
|
void
|
|
reset();
|
|
|
|
protected:
|
|
/**
|
|
* @return The next computed delay value
|
|
*/
|
|
[[nodiscard]] virtual std::chrono::steady_clock::duration
|
|
nextDelay() const = 0;
|
|
};
|
|
using RetryStrategyPtr = std::unique_ptr<RetryStrategy>;
|
|
|
|
/**
|
|
* @brief A retry mechanism
|
|
*/
|
|
class Retry {
|
|
RetryStrategyPtr strategy_;
|
|
boost::asio::steady_timer timer_;
|
|
size_t attemptNumber_ = 0;
|
|
std::shared_ptr<std::atomic_bool> canceled_{std::make_shared<std::atomic_bool>(false)};
|
|
|
|
public:
|
|
/**
|
|
* @brief Construct a new Retry object
|
|
*
|
|
* @param strategy The retry strategy to use
|
|
* @param strand The strand to use for async operations
|
|
*/
|
|
Retry(
|
|
RetryStrategyPtr strategy,
|
|
boost::asio::strand<boost::asio::io_context::executor_type> strand
|
|
);
|
|
|
|
/**
|
|
* @brief Destroy the Retry object
|
|
*/
|
|
~Retry();
|
|
|
|
/**
|
|
* @brief Schedule a retry
|
|
*
|
|
* @tparam Fn The type of the callable to execute
|
|
* @param func The callable to execute
|
|
*/
|
|
template <typename Fn>
|
|
void
|
|
retry(Fn&& func)
|
|
{
|
|
*canceled_ = false;
|
|
timer_.expires_after(strategy_->getDelay());
|
|
strategy_->increaseDelay();
|
|
timer_.async_wait([this,
|
|
canceled = canceled_,
|
|
func = std::forward<Fn>(func)](boost::system::error_code const& ec) {
|
|
if (ec == boost::asio::error::operation_aborted or *canceled) {
|
|
return;
|
|
}
|
|
++attemptNumber_;
|
|
func();
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @brief Cancel scheduled retry if any
|
|
*/
|
|
void
|
|
cancel();
|
|
|
|
/**
|
|
* @return The current attempt number
|
|
*/
|
|
[[nodiscard]] size_t
|
|
attemptNumber() const;
|
|
|
|
/**
|
|
* @return The current delay value
|
|
*/
|
|
[[nodiscard]] std::chrono::steady_clock::duration
|
|
delayValue() const;
|
|
|
|
/**
|
|
* @brief Reset the delay value and attempt number
|
|
*/
|
|
void
|
|
reset();
|
|
};
|
|
|
|
/**
|
|
* @brief A retry strategy that retries while exponentially increasing the delay between attempts
|
|
*/
|
|
class ExponentialBackoffStrategy : public RetryStrategy {
|
|
std::chrono::steady_clock::duration maxDelay_;
|
|
|
|
public:
|
|
/**
|
|
* @brief Construct a new Exponential Backoff Strategy object
|
|
*
|
|
* @param delay The initial delay value
|
|
* @param maxDelay The maximum delay value
|
|
*/
|
|
ExponentialBackoffStrategy(
|
|
std::chrono::steady_clock::duration delay,
|
|
std::chrono::steady_clock::duration maxDelay
|
|
);
|
|
|
|
private:
|
|
[[nodiscard]] std::chrono::steady_clock::duration
|
|
nextDelay() const override;
|
|
};
|
|
|
|
/**
|
|
* @brief Create a retry mechanism with exponential backoff strategy
|
|
*
|
|
* @param delay The initial delay value
|
|
* @param maxDelay The maximum delay value
|
|
* @param strand The strand to use for async operations
|
|
* @return The retry object
|
|
*/
|
|
Retry
|
|
makeRetryExponentialBackoff(
|
|
std::chrono::steady_clock::duration delay,
|
|
std::chrono::steady_clock::duration maxDelay,
|
|
boost::asio::strand<boost::asio::io_context::executor_type> strand
|
|
);
|
|
|
|
} // namespace util
|