524 lines
12 KiB
C++
524 lines
12 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.
|
|
*/
|
|
|
|
#ifndef OCEANBASE_UNITTEST_MOCK_PARTITION_MGR_H_
|
|
#define OCEANBASE_UNITTEST_MOCK_PARTITION_MGR_H_
|
|
#include "mock_ob_partition_log_service.h"
|
|
#include "../storage/mockcontainer/mock_ob_partition.h"
|
|
#include "../storage/mockcontainer/mock_ob_partition_service.h"
|
|
|
|
namespace oceanbase {
|
|
using namespace clog;
|
|
namespace storage {
|
|
class MockObPartition : public storage::MockObIPartition {
|
|
public:
|
|
MockObPartition() : valid_(false)
|
|
{}
|
|
|
|
public:
|
|
int init(int32_t seed)
|
|
{
|
|
partition_key_.init(seed, seed, 1024);
|
|
return common::OB_SUCCESS;
|
|
}
|
|
clog::ObIPartitionLogService* get_log_service()
|
|
{
|
|
return &mock_pls_;
|
|
}
|
|
void set_valid(bool valid)
|
|
{
|
|
valid_ = valid;
|
|
return;
|
|
}
|
|
bool is_valid() const
|
|
{
|
|
return valid_;
|
|
}
|
|
const common::ObPartitionKey& get_partition_key() const
|
|
{
|
|
return partition_key_;
|
|
}
|
|
virtual int get_safe_publish_version(int64_t& publish_version)
|
|
{
|
|
UNUSED(publish_version);
|
|
return 0;
|
|
}
|
|
|
|
private:
|
|
clog::MockPartitionLogService mock_pls_;
|
|
common::ObPartitionKey partition_key_;
|
|
bool valid_;
|
|
};
|
|
|
|
class MockObPartitionService : public storage::MockObIPartitionService {
|
|
public:
|
|
MockObPartitionService() : list_(NULL)
|
|
{}
|
|
|
|
public:
|
|
struct MockPartitionNode {
|
|
MockObPartition partition_;
|
|
MockPartitionNode* next_;
|
|
MockPartitionNode() : partition_(), next_(NULL)
|
|
{}
|
|
~MockPartitionNode()
|
|
{
|
|
partition_.destroy();
|
|
next_ = NULL;
|
|
}
|
|
};
|
|
class MockObPartitionIter : public storage::ObIPartitionIterator {
|
|
public:
|
|
MockObPartitionIter() : mgr_(NULL), curr_node_(NULL)
|
|
{}
|
|
MockObPartitionIter(ObPartitionService* mgr)
|
|
: mgr_(reinterpret_cast<MockObPartitionService*>(mgr)), curr_node_(NULL)
|
|
{}
|
|
virtual ~MockObPartitionIter()
|
|
{
|
|
mgr_ = NULL;
|
|
curr_node_ = NULL;
|
|
}
|
|
int get_next(storage::ObIPartitionGroup*& partition)
|
|
{
|
|
int ret = OB_SUCCESS;
|
|
MockPartitionNode* next_node = NULL;
|
|
if (NULL == mgr_) {
|
|
ret = OB_INVALID_ARGUMENT;
|
|
CLOG_LOG(WARN, "mgr is NULL");
|
|
} else if (OB_SUCCESS != (ret = mgr_->get_next(curr_node_, next_node))) {
|
|
CLOG_LOG(WARN, "fail to get next node", "ret", ret);
|
|
} else if (NULL == next_node) {
|
|
ret = OB_ITER_END;
|
|
} else {
|
|
partition = &next_node->partition_;
|
|
curr_node_ = next_node;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
private:
|
|
MockObPartitionService* mgr_;
|
|
MockPartitionNode* curr_node_;
|
|
};
|
|
|
|
public:
|
|
int create_partition(int32_t seed)
|
|
// int create_partition(common::ObMemberList &list, const ObPartitionKey &key, const int64_t replica_num)
|
|
{
|
|
int ret = OB_SUCCESS;
|
|
MockPartitionNode* node = NULL;
|
|
if (NULL == (node = static_cast<oceanbase::storage::MockObPartitionService::MockPartitionNode*>(
|
|
ob_malloc(sizeof(MockPartitionNode), ObModIds::OB_UPS_LOG)))) {
|
|
ret = OB_ALLOCATE_MEMORY_FAILED;
|
|
CLOG_LOG(WARN, "alloc MockPartitionNode fail");
|
|
} else {
|
|
new (node) MockPartitionNode();
|
|
node->partition_.init(seed);
|
|
if (NULL == list_) {
|
|
list_ = node;
|
|
} else {
|
|
node->next_ = list_;
|
|
list_ = node;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
int get_next(MockPartitionNode* curr_node, MockPartitionNode*& next_node)
|
|
{
|
|
int ret = OB_SUCCESS;
|
|
if (NULL == curr_node) {
|
|
next_node = list_;
|
|
} else {
|
|
next_node = curr_node->next_;
|
|
}
|
|
return ret;
|
|
}
|
|
ObIPartitionIterator* alloc_scan_iter()
|
|
{
|
|
MockObPartitionService::MockObPartitionIter* iter = NULL;
|
|
if (NULL == (iter = static_cast<oceanbase::storage::MockObPartitionService::MockObPartitionIter*>(
|
|
ob_malloc(sizeof(MockObPartitionService::MockObPartitionIter), ObModIds::OB_UPS_LOG)))) {
|
|
CLOG_LOG(WARN, "OB_ALLOCATE_MEMORY_FAILED");
|
|
} else {
|
|
new (iter) MockObPartitionService::MockObPartitionIter(this);
|
|
}
|
|
return iter;
|
|
}
|
|
int revert_scan_iter(ObIPartitionIterator* iter)
|
|
{
|
|
ob_free(iter);
|
|
return common::OB_SUCCESS;
|
|
}
|
|
int get_partition(const common::ObPartitionKey& pkey, storage::ObIPartitionGroup*& partition)
|
|
{
|
|
int ret = OB_ENTRY_NOT_EXIST;
|
|
MockPartitionNode* curr_node = list_;
|
|
while (NULL != curr_node) {
|
|
if (curr_node->partition_.get_partition_key() == pkey) {
|
|
ret = OB_SUCCESS;
|
|
partition = &curr_node->partition_;
|
|
break;
|
|
}
|
|
curr_node = curr_node->next_;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
private:
|
|
MockPartitionNode* list_;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(MockObPartitionService);
|
|
};
|
|
|
|
/*class MockPartition: public ObIPartition
|
|
{
|
|
public:
|
|
int init(int64_t seed)
|
|
{
|
|
seed_ = seed;
|
|
log_cursor_.file_id_ = seed;
|
|
log_cursor_.offset_ = static_cast<int32_t>(seed);
|
|
partition_key_.init(seed, seed);
|
|
append_log_count_ = 0;
|
|
return common::OB_SUCCESS;
|
|
}
|
|
//virtual int get_log_cursor(ObLogCursor &cursor)
|
|
//{
|
|
// cursor.deep_copy(log_cursor_);
|
|
// return common::OB_SUCCESS;
|
|
//}
|
|
virtual const common::ObPartitionKey &get_partition_key() const
|
|
{
|
|
return partition_key_;
|
|
}
|
|
virtual int append_disk_log(const ObLogEntry &log_entry, const ObLogCursor &log_cursor)
|
|
{
|
|
UNUSED(log_entry);
|
|
UNUSED(log_cursor);
|
|
int ret = common::OB_SUCCESS;
|
|
if (append_log_count_ <= 100) {
|
|
ret = common::OB_ERROR_OUT_OF_RANGE;
|
|
++append_log_count_;
|
|
}
|
|
return ret;
|
|
}
|
|
virtual int switch_log() {return common::OB_SUCCESS;}
|
|
virtual int set_scan_disk_log_finished()
|
|
{
|
|
int ret = common::OB_SUCCESS;
|
|
return ret;
|
|
}
|
|
virtual int switch_state()
|
|
{
|
|
int ret = common::OB_SUCCESS;
|
|
if (seed_ % 3 == 1) {
|
|
ret = common::OB_ERROR;
|
|
}
|
|
return ret;
|
|
}
|
|
virtual bool check_mc_timeout()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
virtual int check_stale_member(common::ObServer &server)
|
|
{
|
|
UNUSED(server);
|
|
int ret = common::OB_SUCCESS;
|
|
if (seed_ % 3 == 0) {
|
|
ret = common::OB_NOT_MASTER;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
virtual int remove_member(const common::ObServer &server)
|
|
{
|
|
UNUSED(server);
|
|
return common::OB_SUCCESS;
|
|
}
|
|
|
|
virtual uint64_t get_curr_file_min_log_id()
|
|
{
|
|
uint64_t ret = common::OB_INVALID_ID;
|
|
if (seed_ != 2) {
|
|
ret = static_cast<uint64_t>(seed_);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
virtual uint64_t get_curr_file_max_log_id()
|
|
{
|
|
uint64_t ret = common::OB_INVALID_ID;
|
|
if (seed_ != 2) {
|
|
ret = static_cast<uint64_t>(seed_) * 2;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
virtual bool is_valid() const
|
|
{
|
|
bool bool_ret = true;
|
|
if (4 == seed_) {
|
|
bool_ret = false;
|
|
}
|
|
return bool_ret;
|
|
}
|
|
|
|
virtual ObIPartitionLogService *get_log_service()
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
virtual int flush_cb(const ObLogType type,
|
|
const uint64_t log_id,
|
|
const int64_t mc_timestamp,
|
|
const int64_t proposal_id,
|
|
const ObLogCursor &log_cursor)
|
|
{
|
|
UNUSED(type);
|
|
UNUSED(log_id);
|
|
UNUSED(mc_timestamp);
|
|
UNUSED(proposal_id);
|
|
UNUSED(log_cursor);
|
|
return common::OB_SUCCESS;
|
|
}
|
|
|
|
private:
|
|
int64_t seed_;
|
|
int64_t append_log_count_;
|
|
ObLogCursor log_cursor_;
|
|
common::ObPartitionKey partition_key_;
|
|
};
|
|
|
|
class MockPartitionIter: public ObIPartitionIter
|
|
{
|
|
public:
|
|
MockPartitionIter(ObIPartitionMgr *partition_mgr): ObIPartitionIter(partition_mgr), num_(0)
|
|
{
|
|
partition1.init(1);
|
|
partition2.init(2);
|
|
partition3.init(3);
|
|
partition4.init(4);
|
|
}
|
|
virtual int get_next(ObIPartitionGroup *&partition)
|
|
{
|
|
int ret = common::OB_SUCCESS;
|
|
if (num_ == 0) {
|
|
partition = &partition2;
|
|
num_ ++;
|
|
} else if (num_ == 1) {
|
|
partition = &partition1;
|
|
num_ ++;
|
|
} else if (num_ == 2) {
|
|
partition = &partition3;
|
|
num_ ++;
|
|
} else if (num_ == 3) {
|
|
partition = &partition4;
|
|
num_ ++;
|
|
} else {
|
|
ret = common::OB_ITER_END;
|
|
}
|
|
return ret;
|
|
}
|
|
private:
|
|
int num_;
|
|
MockPartition partition1;
|
|
MockPartition partition2;
|
|
MockPartition partition3;
|
|
MockPartition partition4;
|
|
};
|
|
|
|
class MockMorePartitionIter: public ObIPartitionIter
|
|
{
|
|
public:
|
|
MockMorePartitionIter(ObIPartitionMgr *partition_mgr) : ObIPartitionIter(partition_mgr), num_(0)
|
|
{
|
|
for (int64_t i = 0; i < PARTITION_COUNT; ++i) {
|
|
partitions[i].init(i);
|
|
}
|
|
}
|
|
virtual int get_next(ObIPartitionGroup *&partition)
|
|
{
|
|
int ret = common::OB_SUCCESS;
|
|
if (num_ < PARTITION_COUNT) {
|
|
partition = &partitions[num_];
|
|
num_++;
|
|
} else {
|
|
ret = common::OB_ITER_END;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
private:
|
|
static const int PARTITION_COUNT = 20;
|
|
int num_;
|
|
MockPartition partitions[PARTITION_COUNT];
|
|
};
|
|
|
|
class MockNullPartitionIter: public MockPartitionIter
|
|
{
|
|
public:
|
|
MockNullPartitionIter(ObIPartitionMgr *partition_mgr): MockPartitionIter(partition_mgr)
|
|
{
|
|
}
|
|
public:
|
|
virtual int get_next(ObIPartitionGroup **partition)
|
|
{
|
|
*partition = NULL;
|
|
return common::OB_SUCCESS;
|
|
}
|
|
};
|
|
|
|
class MockPartitionMgr: public ObIPartitionMgr
|
|
{
|
|
public:
|
|
MockPartitionMgr()
|
|
{
|
|
partition1.init(1);
|
|
partition2.init(2);
|
|
partition3.init(3);
|
|
partition3.init(4);
|
|
}
|
|
public:
|
|
virtual int get_partition(const common::ObPartitionKey &partition_key,
|
|
ObIPartitionGroup *&partition)
|
|
{
|
|
int ret = common::OB_SUCCESS;
|
|
if (partition_key.table_id_ == 1) {
|
|
partition = &partition1;
|
|
} else if (partition_key.table_id_ == 2) {
|
|
partition = &partition2;
|
|
} else if (partition_key.table_id_ == 3) {
|
|
partition = &partition3;
|
|
} else if (partition_key.table_id_ == 4) {
|
|
partition = &partition4;
|
|
} else {
|
|
ret = common::OB_ENTRY_NOT_EXIST;
|
|
}
|
|
return ret;
|
|
}
|
|
virtual void revert(ObIPartitionGroup *partition)
|
|
{
|
|
UNUSED(partition);
|
|
}
|
|
virtual ObIPartitionIter *alloc_scan_iter()
|
|
{
|
|
ObIPartitionIter *partition_iter = new MockPartitionIter(this);
|
|
return partition_iter;
|
|
}
|
|
virtual int revert_scan_iter(ObIPartitionIter *iter)
|
|
{
|
|
delete iter;
|
|
return common::OB_SUCCESS;
|
|
}
|
|
private:
|
|
MockPartition partition1;
|
|
MockPartition partition2;
|
|
MockPartition partition3;
|
|
MockPartition partition4;
|
|
};
|
|
|
|
class MockMorePartitionMgr: public ObIPartitionMgr
|
|
{
|
|
public:
|
|
MockMorePartitionMgr()
|
|
{
|
|
}
|
|
public:
|
|
virtual int get_partition(const uint64_t table_id,
|
|
const int64_t partition_idx,
|
|
ObIPartitionGroup **partition)
|
|
{
|
|
UNUSED(table_id);
|
|
UNUSED(partition_idx);
|
|
UNUSED(partition);
|
|
return common::OB_SUCCESS;
|
|
}
|
|
virtual void revert(const ObIPartitionGroup *partition)
|
|
{
|
|
UNUSED(partition);
|
|
}
|
|
virtual ObIPartitionIter *alloc_scan_iter()
|
|
{
|
|
ObIPartitionIter *partition_iter = new MockMorePartitionIter(this);
|
|
return partition_iter;
|
|
}
|
|
virtual int revert_scan_iter(ObIPartitionIter *iter)
|
|
{
|
|
delete iter;
|
|
return common::OB_SUCCESS;
|
|
}
|
|
};
|
|
|
|
class MockNullPartitionMgr: public MockPartitionMgr
|
|
{
|
|
public:
|
|
virtual ObIPartitionIter *alloc_scan_iter()
|
|
{
|
|
return NULL;
|
|
}
|
|
};
|
|
|
|
class MockNull2PartitionMgr: public MockPartitionMgr
|
|
{
|
|
public:
|
|
virtual ObIPartitionIter *alloc_scan_iter()
|
|
{
|
|
return new MockNullPartitionIter(this);
|
|
}
|
|
};
|
|
|
|
class MockNull3PartitionMgr: public MockPartitionMgr
|
|
{
|
|
public:
|
|
MockNull3PartitionMgr(): count_(0)
|
|
{
|
|
}
|
|
public:
|
|
virtual ObIPartitionIter *alloc_scan_iter()
|
|
{
|
|
if (0 == count_) {
|
|
++count_;
|
|
return new MockPartitionIter(this);
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
private:
|
|
int64_t count_;
|
|
};
|
|
|
|
class MockNull4PartitionMgr: public MockPartitionMgr
|
|
{
|
|
public:
|
|
MockNull4PartitionMgr(): count_(0)
|
|
{
|
|
}
|
|
public:
|
|
virtual ObIPartitionIter *alloc_scan_iter()
|
|
{
|
|
if (0 == count_) {
|
|
++count_;
|
|
return new MockPartitionIter(this);
|
|
} else {
|
|
return new MockNullPartitionIter(this);
|
|
}
|
|
}
|
|
private:
|
|
int64_t count_;
|
|
};*/
|
|
} // namespace storage
|
|
} // namespace oceanbase
|
|
|
|
#endif // OCEANBASE_UNITTEST_MOCK_PARTITION_MGR_H_
|