/* * Copyright 2018 The Cartographer Authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <chrono> #include <thread> #include "cartographer_grpc/framework/retry.h" #include "glog/logging.h" namespace cartographer_grpc { namespace framework { RetryStrategy CreateRetryStrategy(RetryIndicator retry_indicator, RetryDelayCalculator retry_delay_calculator) { return [retry_indicator, retry_delay_calculator](int failed_attempts) { if (!retry_indicator(failed_attempts)) { return optional<Duration>(); } return optional<Duration>(retry_delay_calculator(failed_attempts)); }; } RetryIndicator CreateLimitedRetryIndicator(int max_attempts) { return [max_attempts](int failed_attempts) { return failed_attempts < max_attempts; }; } RetryDelayCalculator CreateBackoffDelayCalculator(Duration min_delay, float backoff_factor) { return [min_delay, backoff_factor](int failed_attempts) -> Duration { CHECK_GE(failed_attempts, 0); using cartographer::common::FromSeconds; using cartographer::common::ToSeconds; return FromSeconds(std::pow(backoff_factor, failed_attempts - 1) * ToSeconds(min_delay)); }; } RetryStrategy CreateLimitedBackoffStrategy(Duration min_delay, float backoff_factor, int max_attempts) { return CreateRetryStrategy( CreateLimitedRetryIndicator(max_attempts), CreateBackoffDelayCalculator(min_delay, backoff_factor)); } bool RetryWithStrategy(RetryStrategy retry_strategy, std::function<bool()> op, std::function<void()> reset) { optional<Duration> delay; int failed_attemps = 0; for (;;) { if (op()) { return true; } if (!retry_strategy) { return false; } delay = retry_strategy(++failed_attemps); if (!delay.has_value()) { break; } LOG(INFO) << "Retrying after " << std::chrono::duration_cast<std::chrono::milliseconds>( delay.value()) .count() << " milliseconds."; std::this_thread::sleep_for(delay.value()); if (reset) { reset(); } } return false; } } // namespace framework } // namespace cartographer_grpc