#pragma once #include "retry_lib.h" #include "wait_proxy.h" #include #include #include #include namespace NYT::NDetail { //////////////////////////////////////////////////////////////////////////////// template TResult RequestWithRetry( IRequestRetryPolicyPtr retryPolicy, std::function func) { bool useSameMutationId = false; TMutationId mutationId; while (true) { try { retryPolicy->NotifyNewAttempt(); if constexpr (std::is_same_v) { func(mutationId); return; } else { return func(mutationId); } } catch (const TErrorResponse& e) { YT_LOG_ERROR("Retry failed %v - %v", e.GetError().GetMessage(), retryPolicy->GetAttemptDescription()); useSameMutationId = e.IsTransportError(); if (!IsRetriable(e)) { throw; } auto maybeRetryTimeout = retryPolicy->OnRetriableError(e); if (maybeRetryTimeout) { TWaitProxy::Get()->Sleep(*maybeRetryTimeout); } else { throw; } } catch (const std::exception& e) { YT_LOG_ERROR("Retry failed %v - %v", e.what(), retryPolicy->GetAttemptDescription()); useSameMutationId = true; if (!IsRetriable(e)) { throw; } auto maybeRetryTimeout = retryPolicy->OnGenericError(e); if (maybeRetryTimeout) { TWaitProxy::Get()->Sleep(*maybeRetryTimeout); } else { throw; } } if (!useSameMutationId) { mutationId = {}; } } } //////////////////////////////////////////////////////////////////////////////// } // namespace NYT::NDetail