Files
oceanbase/unittest/share/backup/test_multi_backup_dest_util.cpp
2021-12-06 10:57:09 +08:00

454 lines
15 KiB
C++

/**
* Copyright (c) 2021 OceanBase
* OceanBase CE is licensed under Mulan PubL v2.
* You can use this software according to the terms and conditions of the Mulan PubL v2.
* You may obtain a copy of Mulan PubL v2 at:
* http://license.coscl.org.cn/MulanPubL-2.0
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
* See the Mulan PubL v2 for more details.
*/
#define USING_LOG_PREFIX SHARE
#include <gtest/gtest.h>
#define private public
#include "share/backup/ob_backup_struct.h"
#include "share/backup/ob_multi_backup_dest_util.h"
using namespace oceanbase;
using namespace oceanbase::common;
using namespace oceanbase::share;
void mock_backup_set(const int64_t backup_set_id, const int64_t prev_full_backup_set_id,
const int64_t prev_inc_backup_set_id, const int64_t snapshot_version, const int64_t start_replay_log_ts,
const share::ObBackupType::BackupType type, share::ObBackupSetFileInfo& set_info)
{
set_info.reset();
set_info.backup_set_id_ = backup_set_id;
set_info.prev_full_backup_set_id_ = prev_full_backup_set_id;
set_info.prev_inc_backup_set_id_ = prev_inc_backup_set_id;
set_info.snapshot_version_ = snapshot_version;
set_info.start_replay_log_ts_ = start_replay_log_ts;
set_info.backup_type_.type_ = type;
}
void mock_backup_piece(const int64_t round_id, const int64_t piece_id, const int64_t start_ts,
const int64_t checkpoint_ts, const int64_t max_ts, const share::ObBackupPieceStatus::STATUS& status,
share::ObBackupPieceInfo& piece)
{
piece.reset();
piece.key_.round_id_ = round_id; // check all same round
piece.key_.backup_piece_id_ = piece_id; // check piece is continuous
piece.start_ts_ = start_ts;
piece.checkpoint_ts_ = checkpoint_ts;
piece.max_ts_ = max_ts;
piece.status_ = status; // check all but last is frozen
piece.start_piece_id_ = 1; // check start piece id
}
int mock_only_full_backup_set(const int64_t full_count, common::ObArray<share::ObBackupSetFileInfo>& set_list)
{
int ret = OB_SUCCESS;
set_list.reset();
for (int64_t i = 1; OB_SUCC(ret) && i <= full_count; ++i) {
ObBackupSetFileInfo info;
mock_backup_set(i, 0, 0, 100, 100, ObBackupType::FULL_BACKUP, info);
if (OB_FAIL(set_list.push_back(info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_only_inc_backup_set_list(const int64_t inc_count, common::ObArray<share::ObBackupSetFileInfo>& set_list)
{
int ret = OB_SUCCESS;
set_list.reset();
for (int64_t i = 2; OB_SUCC(ret) && i < 2 + inc_count; ++i) {
ObBackupSetFileInfo info;
mock_backup_set(i, 1, 1, 100, 100, ObBackupType::INCREMENTAL_BACKUP, info);
if (OB_FAIL(set_list.push_back(info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_normal_full_and_inc_backup_set_list(common::ObArray<share::ObBackupSetFileInfo>& set_list)
{
int ret = OB_SUCCESS;
set_list.reset();
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupSetFileInfo info;
if (1 == i) {
mock_backup_set(1, 0, 0, 100, 100, ObBackupType::FULL_BACKUP, info);
} else if (2 == i) {
mock_backup_set(2, 1, 1, 200, 200, ObBackupType::INCREMENTAL_BACKUP, info);
} else if (3 == i) {
mock_backup_set(3, 1, 2, 300, 300, ObBackupType::INCREMENTAL_BACKUP, info);
}
if (OB_FAIL(set_list.push_back(info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_duplicate_inc_backup_set_list(common::ObArray<share::ObBackupSetFileInfo>& set_list)
{
int ret = OB_SUCCESS;
set_list.reset();
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupSetFileInfo info;
if (1 == i) {
mock_backup_set(1, 0, 0, 100, 100, ObBackupType::FULL_BACKUP, info);
} else if (2 == i) {
mock_backup_set(2, 1, 1, 200, 200, ObBackupType::INCREMENTAL_BACKUP, info);
} else if (3 == i) {
mock_backup_set(2, 1, 1, 300, 300, ObBackupType::INCREMENTAL_BACKUP, info);
}
if (OB_FAIL(set_list.push_back(info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_full_and_inc_discontinuous_backup_set_list_with_failed(common::ObArray<share::ObBackupSetFileInfo>& set_list)
{
int ret = OB_SUCCESS;
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupSetFileInfo info;
if (1 == i) {
mock_backup_set(1, 0, 0, 100, 100, ObBackupType::FULL_BACKUP, info);
} else if (2 == i) {
mock_backup_set(2, 1, 1, 200, 200, ObBackupType::INCREMENTAL_BACKUP, info);
info.result_ = OB_IO_ERROR;
} else if (3 == i) {
mock_backup_set(3, 1, 1, 300, 300, ObBackupType::INCREMENTAL_BACKUP, info);
}
if (OB_FAIL(set_list.push_back(info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_full_and_inc_discontinuous_backup_set_list(common::ObArray<share::ObBackupSetFileInfo>& set_list)
{
int ret = OB_SUCCESS;
set_list.reset();
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupSetFileInfo info;
if (1 == i) {
mock_backup_set(1, 0, 0, 100, 100, ObBackupType::FULL_BACKUP, info);
} else if (2 == i) {
continue;
} else if (3 == i) {
mock_backup_set(3, 1, 2, 300, 300, ObBackupType::INCREMENTAL_BACKUP, info);
}
if (OB_FAIL(set_list.push_back(info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_normal_round_piece_list(common::ObArray<share::ObBackupPieceInfo>& piece_list)
{
int ret = OB_SUCCESS;
piece_list.reset();
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupPieceInfo piece_info;
if (1 == i) {
mock_backup_piece(1, i, 100, 200, 300, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
} else if (2 == i) {
mock_backup_piece(1, i, 300, 400, 500, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
} else if (3 == i) {
mock_backup_piece(1, i, 500, 700, 1000, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
}
if (OB_FAIL(piece_list.push_back(piece_info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_discontinuous_piece_list(common::ObArray<share::ObBackupPieceInfo>& piece_list)
{
int ret = OB_SUCCESS;
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupPieceInfo piece_info;
piece_info.start_piece_id_ = 1;
if (1 == i) {
mock_backup_piece(1, i, 100, 200, 300, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
} else if (2 == i) {
continue;
} else if (3 == i) {
mock_backup_piece(1, i, 600, 1000, 1100, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
}
if (OB_FAIL(piece_list.push_back(piece_info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_duplicate_piece_list(common::ObArray<share::ObBackupPieceInfo>& piece_list)
{
int ret = OB_SUCCESS;
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupPieceInfo piece_info;
mock_backup_piece(1, 1, 100, 200, 300, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
if (OB_FAIL(piece_list.push_back(piece_info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_mid_freezing_piece_list(common::ObArray<share::ObBackupPieceInfo>& piece_list)
{
int ret = OB_SUCCESS;
piece_list.reset();
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupPieceInfo piece_info;
if (1 == i) {
mock_backup_piece(1, 1, 100, 200, 300, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
} else if (2 == i) {
mock_backup_piece(1, 2, 300, 500, 600, ObBackupPieceStatus::BACKUP_PIECE_FREEZING, piece_info);
} else if (3 == i) {
mock_backup_piece(1, 3, 600, 1000, 1100, ObBackupPieceStatus::BACKUP_PIECE_ACTIVE, piece_info);
}
if (OB_FAIL(piece_list.push_back(piece_info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
int mock_different_round_piece_list(common::ObArray<share::ObBackupPieceInfo>& piece_list)
{
int ret = OB_SUCCESS;
piece_list.reset();
for (int64_t i = 1; OB_SUCC(ret) && i <= 3; ++i) {
ObBackupPieceInfo piece_info;
if (1 == i) {
mock_backup_piece(1, 1, 100, 200, 300, ObBackupPieceStatus::BACKUP_PIECE_FROZEN, piece_info);
} else if (2 == i) {
mock_backup_piece(2, 2, 300, 500, 600, ObBackupPieceStatus::BACKUP_PIECE_FREEZING, piece_info);
} else if (3 == i) {
mock_backup_piece(3, 3, 600, 1000, 1100, ObBackupPieceStatus::BACKUP_PIECE_ACTIVE, piece_info);
}
if (OB_FAIL(piece_list.push_back(piece_info))) {
LOG_WARN("failed to push back", KR(ret));
}
}
return ret;
}
// case1:
// F I I
// 1 2 3
TEST(ObMultiBackupDestUtil, case1)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 600; // change restore timestamp
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_normal_full_and_inc_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_normal_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_SUCCESS, ret);
}
// I I
// 1 2 3
TEST(ObMultiBackupDestUtil, case2)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 1000;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_only_inc_backup_set_list(2, set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_normal_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 3: missing one incremental backup set
// F X I
// 1 2 3
TEST(ObMultiBackupDestUtil, case3)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 1000;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_full_and_inc_discontinuous_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_normal_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 4: one incremntal backup set is failed
// F X I
// 1 2 3
TEST(ObMultiBackupDestUtil, case4)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 1000;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_full_and_inc_discontinuous_backup_set_list_with_failed(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_normal_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 5: piece id not continuous
// F
// 1 3
TEST(ObMultiBackupDestUtil, case5)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 1000;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_only_full_backup_set(1, set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_discontinuous_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 6: duplicate inc backup set id
TEST(ObMultiBackupDestUtil, case6)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 700;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_duplicate_inc_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_normal_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 7: duplicate backup piece id
TEST(ObMultiBackupDestUtil, case7)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 700;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_normal_full_and_inc_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_duplicate_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 8: has freezing backup piece
TEST(ObMultiBackupDestUtil, case8)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 700;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_normal_full_and_inc_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_mid_freezing_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 9: piece from different round
TEST(ObMultiBackupDestUtil, case9)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 700;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_normal_full_and_inc_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_different_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 10: restore timestamp too small
TEST(ObMultiBackupDestUtil, case10)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 299;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_normal_full_and_inc_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_normal_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
// case 11: restore timestamp too large
TEST(ObMultiBackupDestUtil, case11)
{
int ret = OB_SUCCESS;
const int64_t restore_timestamp = 1000;
bool is_complete = false;
ObArray<ObBackupSetFileInfo> set_list;
ObArray<ObBackupPieceInfo> piece_list;
ret = mock_normal_full_and_inc_backup_set_list(set_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = mock_normal_round_piece_list(piece_list);
ASSERT_EQ(OB_SUCCESS, ret);
ret = ObMultiBackupDestUtil::check_multi_path_is_complete(restore_timestamp, set_list, piece_list, is_complete);
ASSERT_EQ(OB_INVALID_ARGUMENT, ret);
}
int main(int argc, char** argv)
{
OB_LOGGER.set_log_level("INFO");
testing::InitGoogleTest(&argc, argv);
return RUN_ALL_TESTS();
}