The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#) https://grpc.io/
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

187 lines
7.0 KiB

/*
*
* Copyright 2016 gRPC 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 "src/core/lib/backoff/backoff.h"
#include <algorithm>
#include <gtest/gtest.h>
#include <grpc/grpc.h>
#include <grpc/support/log.h>
#include "src/core/lib/gprpp/time.h"
#include "test/core/util/test_config.h"
namespace grpc {
namespace testing {
namespace {
using grpc_core::BackOff;
TEST(BackOffTest, ConstantBackOff) {
const auto initial_backoff = grpc_core::Duration::Milliseconds(200);
const double multiplier = 1.0;
const double jitter = 0.0;
const auto max_backoff = grpc_core::Duration::Seconds(1);
grpc_core::ExecCtx exec_ctx;
BackOff::Options options;
options.set_initial_backoff(initial_backoff)
.set_multiplier(multiplier)
.set_jitter(jitter)
.set_max_backoff(max_backoff);
BackOff backoff(options);
grpc_core::Timestamp next_attempt_start_time = backoff.NextAttemptTime();
EXPECT_EQ(next_attempt_start_time - grpc_core::ExecCtx::Get()->Now(),
initial_backoff);
for (int i = 0; i < 10000; i++) {
next_attempt_start_time = backoff.NextAttemptTime();
EXPECT_EQ(next_attempt_start_time - grpc_core::ExecCtx::Get()->Now(),
initial_backoff);
}
}
TEST(BackOffTest, MinConnect) {
const auto initial_backoff = grpc_core::Duration::Milliseconds(100);
const double multiplier = 1.0;
const double jitter = 0.0;
const auto max_backoff = grpc_core::Duration::Seconds(1);
grpc_core::ExecCtx exec_ctx;
BackOff::Options options;
options.set_initial_backoff(initial_backoff)
.set_multiplier(multiplier)
.set_jitter(jitter)
.set_max_backoff(max_backoff);
BackOff backoff(options);
grpc_core::Timestamp next = backoff.NextAttemptTime();
EXPECT_EQ(next - grpc_core::ExecCtx::Get()->Now(), initial_backoff);
}
TEST(BackOffTest, NoJitterBackOff) {
const auto initial_backoff = grpc_core::Duration::Milliseconds(2);
const double multiplier = 2.0;
const double jitter = 0.0;
const auto max_backoff = grpc_core::Duration::Milliseconds(513);
BackOff::Options options;
options.set_initial_backoff(initial_backoff)
.set_multiplier(multiplier)
.set_jitter(jitter)
.set_max_backoff(max_backoff);
BackOff backoff(options);
// x_1 = 2
// x_n = 2**i + x_{i-1} ( = 2**(n+1) - 2 )
grpc_core::ExecCtx exec_ctx;
grpc_core::ExecCtx::Get()->TestOnlySetNow(
grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(0));
grpc_core::Timestamp next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(2));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(6));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(14));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(30));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(62));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(126));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(254));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next, grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(510));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next,
grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(1022));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
// Hit the maximum timeout. From this point onwards, retries will increase
// only by max timeout.
EXPECT_EQ(next,
grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(1535));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next,
grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(2048));
grpc_core::ExecCtx::Get()->TestOnlySetNow(next);
next = backoff.NextAttemptTime();
EXPECT_EQ(next,
grpc_core::Timestamp::FromMillisecondsAfterProcessEpoch(2561));
}
TEST(BackOffTest, JitterBackOff) {
const auto initial_backoff = grpc_core::Duration::Milliseconds(500);
auto current_backoff = initial_backoff;
const auto max_backoff = grpc_core::Duration::Seconds(1);
const double multiplier = 1.0;
const double jitter = 0.1;
BackOff::Options options;
options.set_initial_backoff(initial_backoff)
.set_multiplier(multiplier)
.set_jitter(jitter)
.set_max_backoff(max_backoff);
BackOff backoff(options);
grpc_core::ExecCtx exec_ctx;
grpc_core::Timestamp next = backoff.NextAttemptTime();
EXPECT_EQ(next - grpc_core::ExecCtx::Get()->Now(), initial_backoff);
auto expected_next_lower_bound = grpc_core::Duration::Milliseconds(
static_cast<double>(current_backoff.millis()) * (1 - jitter));
auto expected_next_upper_bound = grpc_core::Duration::Milliseconds(
static_cast<double>(current_backoff.millis()) * (1 + jitter));
for (int i = 0; i < 10000; i++) {
next = backoff.NextAttemptTime();
// next-now must be within (jitter*100)% of the current backoff (which
// increases by * multiplier up to max_backoff).
const grpc_core::Duration timeout_millis =
next - grpc_core::ExecCtx::Get()->Now();
EXPECT_GE(timeout_millis, expected_next_lower_bound);
EXPECT_LE(timeout_millis, expected_next_upper_bound);
current_backoff = std::min(
grpc_core::Duration::Milliseconds(
static_cast<double>(current_backoff.millis()) * multiplier),
max_backoff);
expected_next_lower_bound = grpc_core::Duration::Milliseconds(
static_cast<double>(current_backoff.millis()) * (1 - jitter));
expected_next_upper_bound = grpc_core::Duration::Milliseconds(
static_cast<double>(current_backoff.millis()) * (1 + jitter));
}
}
} // namespace
} // namespace testing
} // namespace grpc
int main(int argc, char** argv) {
::testing::InitGoogleTest(&argc, argv);
grpc::testing::TestEnvironment env(argc, argv);
grpc_init();
int ret = RUN_ALL_TESTS();
grpc_shutdown();
return ret;
}