oceanbase
469 строк · 14.1 Кб
1/**
2* Copyright (c) 2021 OceanBase
3* OceanBase CE is licensed under Mulan PubL v2.
4* You can use this software according to the terms and conditions of the Mulan PubL v2.
5* You may obtain a copy of Mulan PubL v2 at:
6* http://license.coscl.org.cn/MulanPubL-2.0
7* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
8* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
9* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
10* See the Mulan PubL v2 for more details.
11*/
12
13#include <cstdio>
14#include <gtest/gtest.h>
15#include <signal.h>
16#define private public
17#include "lib/allocator/ob_qsync.h"
18#include "logservice/ob_ls_adapter.h"
19#include "env/ob_simple_log_cluster_env.h"
20#undef private
21
22const std::string TEST_NAME = "apply_func";
23using namespace oceanbase::common;
24using namespace oceanbase;
25
26namespace oceanbase
27{
28using namespace logservice;
29namespace unittest
30{
31class MockLSAdapter : public ObLSAdapter
32{
33public:
34MockLSAdapter() {
35ObLSAdapter();
36success_count_ = 0;
37failure_count_ = 0;
38}
39
40int wait_append_sync(const share::ObLSID &ls_id)
41{
42WaitQuiescent(ls_qs_);
43CLOG_LOG(INFO, "WaitQuiescent", K(ls_id));
44return OB_SUCCESS;
45}
46void critical_guard()
47{
48CriticalGuard(ls_qs_);
49}
50void inc_success_count()
51{
52ATOMIC_INC(&success_count_);
53}
54void inc_failure_count()
55{
56ATOMIC_INC(&failure_count_);
57}
58int replay(ObLogReplayTask *replay_task){
59UNUSED(replay_task);
60return OB_SUCCESS;
61}
62int64_t success_count_;
63int64_t failure_count_;
64common::ObQSync ls_qs_;
65};
66
67class TestObSimpleLogApplyFunc : public ObSimpleLogClusterTestEnv
68{
69public:
70TestObSimpleLogApplyFunc() : ObSimpleLogClusterTestEnv()
71{}
72};
73
74class MockAppendCb : public AppendCb
75{
76public:
77int on_success()
78{
79ls_adapter_->inc_success_count();
80CLOG_LOG(INFO, "on_success", K(log_id_));
81return OB_SUCCESS;
82}
83
84int on_failure()
85{
86ls_adapter_->inc_failure_count();
87CLOG_LOG(INFO, "on_failure", K(log_id_));
88return OB_SUCCESS;
89}
90
91void init(const int64_t log_id,
92MockLSAdapter *ls_adapter)
93{
94log_id_ = log_id;
95ls_adapter_ = ls_adapter;
96}
97int64_t log_id_;
98MockLSAdapter *ls_adapter_;
99};
100
101int64_t ObSimpleLogClusterTestBase::member_cnt_ = 3;
102int64_t ObSimpleLogClusterTestBase::node_cnt_ = 3;
103std::string ObSimpleLogClusterTestBase::test_name_ = TEST_NAME;
104bool ObSimpleLogClusterTestBase::need_add_arb_server_ = false;
105
106TEST_F(TestObSimpleLogApplyFunc, apply)
107{
108const int64_t task_count = 50;
109const int64_t truncate_count = 25;
110const int64_t id = ATOMIC_AAF(&palf_id_, 1);
111ObLSID ls_id(id);
112int64_t leader_idx = 0;
113LSN basic_lsn(0);
114PalfHandleImplGuard leader;
115MockAppendCb *cb_array[task_count];
116LSN unused_apply_end_lsn;
117bool is_apply_done = false;
118CLOG_LOG(INFO, "test apply begin", K(id));
119EXPECT_EQ(OB_SUCCESS, create_paxos_group(id, leader_idx, leader));
120MockLSAdapter ls_adapter;
121ls_adapter.init((ObLSService *)(0x1));
122ObLogApplyService ap_sv;
123for (int i = 0; i < task_count; i++)
124{
125cb_array[i] = new MockAppendCb();
126cb_array[i]->init(i + 1, &ls_adapter);
127}
128PalfEnv *palf_env;
129EXPECT_EQ(OB_SUCCESS, get_palf_env(leader_idx, palf_env));
130EXPECT_EQ(OB_SUCCESS, ap_sv.init(palf_env, &ls_adapter));
131EXPECT_EQ(OB_SUCCESS, ap_sv.start());
132EXPECT_EQ(OB_SUCCESS, ap_sv.add_ls(ls_id));
133EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_leader(ls_id, 1));
134const int64_t idx_1 = (leader_idx + 1) % get_node_cnt();
135const int64_t idx_2 = (leader_idx + 2) % get_node_cnt();
136
137//旧主少数派写日志
138block_net(leader_idx, idx_1);
139block_net(leader_idx, idx_2);
140do {
141ObApplyStatus *apply_status = NULL;
142ObApplyStatusGuard guard;
143ap_sv.get_apply_status(ls_id, guard);
144apply_status = guard.get_apply_status();
145LSN lsn;
146share::SCN scn;
147int ret = OB_SUCCESS;
148for (int i = 0; i < truncate_count; i++)
149{
150{
151ls_adapter.critical_guard();
152CLOG_LOG(INFO, "submit log start", K(i));
153share::SCN ref_scn;
154ref_scn.convert_for_logservice(i);
155ret = submit_log(leader, ref_scn, lsn, scn);
156if (OB_SUCC(ret)) {
157EXPECT_EQ(true, lsn.is_valid());
158cb_array[i]->__set_lsn(lsn);
159cb_array[i]->__set_scn(scn);
160EXPECT_EQ(OB_SUCCESS, apply_status->push_append_cb(cb_array[i]));
161} else {
162break;
163}
164}
165CLOG_LOG(INFO, "submit log finish", K(i), K(lsn), K(scn));
166}
167} while (0);
168share::SCN min_scn;
169EXPECT_EQ(OB_SUCCESS, ap_sv.get_max_applied_scn(ls_id, min_scn));
170EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_follower(ls_id));
171
172//切主, truncate旧主日志,预期所有cb都调用on_failure
173sleep(15);
174while (!is_apply_done)
175{
176ap_sv.is_apply_done(ls_id, is_apply_done, unused_apply_end_lsn);
177usleep(100);
178}
179
180//切回旧主写日志,预期所有cb都调用on_success
181unblock_net(leader_idx, idx_1);
182unblock_net(leader_idx, idx_2);
183int64_t new_leader_idx = 0;
184PalfHandleImplGuard new_leader;
185EXPECT_EQ(OB_SUCCESS, get_leader(id, new_leader, new_leader_idx));
186EXPECT_NE(new_leader_idx, leader_idx);
187//等待membership同步
188sleep(2);
189leader.reset();
190CLOG_LOG(INFO, "new leader", K(new_leader_idx), K(leader_idx));
191EXPECT_EQ(OB_SUCCESS, switch_leader(id, leader_idx, leader));
192leader.reset();
193sleep(2);
194EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_leader(ls_id, 3));
195EXPECT_EQ(OB_SUCCESS, ap_sv.get_max_applied_scn(ls_id, min_scn));
196EXPECT_EQ(OB_SUCCESS, get_leader(id, leader, new_leader_idx));
197CLOG_LOG(INFO, "check switch leader", K(new_leader_idx), K(leader_idx));
198EXPECT_EQ(new_leader_idx, leader_idx);
199share::SCN palf_end_scn;
200do {
201ObApplyStatus *apply_status = NULL;
202ObApplyStatusGuard guard;
203ap_sv.get_apply_status(ls_id, guard);
204apply_status = guard.get_apply_status();
205LSN lsn;
206share::SCN scn;
207for (int i = truncate_count; i < task_count; i++)
208{
209EXPECT_EQ(OB_SUCCESS, ap_sv.get_max_applied_scn(ls_id, scn));
210{
211int64_t ref_ts = i - truncate_count;
212share::SCN ref_scn;
213ref_scn.convert_for_logservice(ref_ts);
214ls_adapter.critical_guard();
215CLOG_LOG(INFO, "submit log start", K(i));
216EXPECT_EQ(OB_SUCCESS, submit_log(leader, ref_scn, lsn, scn));
217EXPECT_EQ(true, lsn.is_valid());
218cb_array[i]->__set_lsn(lsn);
219cb_array[i]->__set_scn(scn);
220palf_end_scn = scn;
221EXPECT_EQ(OB_SUCCESS, apply_status->push_append_cb(cb_array[i]));
222}
223CLOG_LOG(INFO, "submit log finish", K(i), K(lsn), K(scn));
224EXPECT_EQ(OB_SUCCESS, ap_sv.get_max_applied_scn(ls_id, scn));
225}
226} while (0);
227CLOG_LOG(INFO, "truncate write finish", K(id));
228
229// remove
230is_apply_done = false;
231while (!is_apply_done)
232{
233ap_sv.is_apply_done(ls_id, is_apply_done, unused_apply_end_lsn);
234usleep(100);
235}
236share::SCN scn;
237EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_follower(ls_id));
238EXPECT_EQ(OB_SUCCESS, ap_sv.get_max_applied_scn(ls_id, scn));
239EXPECT_EQ(scn.get_val_for_logservice(), palf_end_scn.get_val_for_logservice());
240EXPECT_EQ(truncate_count, ls_adapter.failure_count_);
241EXPECT_EQ(task_count- truncate_count, ls_adapter.success_count_);
242for (int i = 0; i < task_count; i++)
243{
244delete cb_array[i];
245}
246EXPECT_EQ(OB_SUCCESS, ap_sv.remove_ls(ls_id));
247ap_sv.stop();
248ap_sv.wait();
249ap_sv.destroy();
250CLOG_LOG(INFO, "test apply finish", K(id));
251}
252
253TEST_F(TestObSimpleLogApplyFunc, get_max_decided_scn)
254{
255const int64_t task_count = 10;
256const int64_t id = ATOMIC_AAF(&palf_id_, 1);
257ObLSID ls_id(id);
258int64_t leader_idx = 0;
259LSN basic_lsn(0);
260PalfHandleImplGuard leader;
261MockAppendCb *cb_array[task_count];
262CLOG_LOG(INFO, "test get_max_decided_scn begin", K(id));
263EXPECT_EQ(OB_SUCCESS, create_paxos_group(id, leader_idx, leader));
264MockLSAdapter ls_adapter;
265ls_adapter.init((ObLSService *)(0x1));
266ObLogApplyService ap_sv;
267//submit log first
268for (int i = 0; i < task_count; i++)
269{
270cb_array[i] = new MockAppendCb();
271cb_array[i]->init(i + 1, &ls_adapter);
272}
273PalfEnv *palf_env;
274ObTenantEnv::set_tenant(get_cluster()[leader_idx]->get_tenant_base());
275EXPECT_EQ(OB_SUCCESS, get_palf_env(leader_idx, palf_env));
276EXPECT_EQ(OB_SUCCESS, ap_sv.init(palf_env, &ls_adapter));
277EXPECT_EQ(OB_SUCCESS, ap_sv.start());
278EXPECT_EQ(OB_SUCCESS, ap_sv.add_ls(ls_id));
279EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_leader(ls_id, 1));
280const int64_t idx_1 = (leader_idx + 1) % get_node_cnt();
281const int64_t idx_2 = (leader_idx + 2) % get_node_cnt();
282share::SCN ref_scn;
283int i = 0;
284LSN lsn;
285share::SCN scn;
286
287do {
288ObApplyStatus *apply_status = NULL;
289ObApplyStatusGuard guard;
290ap_sv.get_apply_status(ls_id, guard);
291apply_status = guard.get_apply_status();
292ASSERT_TRUE(NULL != apply_status);
293int ret = OB_SUCCESS;
294for (i = 0; i < task_count; i++)
295{
296{
297ls_adapter.critical_guard();
298CLOG_LOG(INFO, "submit log start", K(i));
299ref_scn.convert_for_logservice(i);
300ret = submit_log(leader, ref_scn, lsn, scn);
301if (OB_SUCC(ret)) {
302EXPECT_EQ(true, lsn.is_valid());
303cb_array[i]->__set_lsn(lsn);
304cb_array[i]->__set_scn(scn);
305EXPECT_EQ(OB_SUCCESS, apply_status->push_append_cb(cb_array[i]));
306} else {
307break;
308}
309}
310CLOG_LOG(INFO, "submit log finish", K(i), K(lsn), K(scn));
311}
312} while (0);
313
314LSN max_lsn = leader.palf_handle_impl_->sw_.get_max_lsn();
315SCN max_scn = leader.palf_handle_impl_->sw_.get_max_scn();
316
317//wait apply done
318wait_lsn_until_slide(max_lsn, leader);
319EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_follower(ls_id));
320LSN apply_end_lsn;
321bool is_apply_done = false;
322while (!is_apply_done)
323{
324ap_sv.is_apply_done(ls_id, is_apply_done, apply_end_lsn);
325usleep(100);
326}
327EXPECT_EQ(max_lsn, apply_end_lsn);
328
329//switch leader and submit log
330PalfHandleImplGuard new_leader;
331EXPECT_EQ(OB_SUCCESS, switch_leader(id, idx_1, new_leader));
332
333PalfHandleImplGuard new_get_leader;
334int64_t new_leader_idx = 0;
335EXPECT_EQ(OB_SUCCESS, get_leader(id, new_get_leader, new_leader_idx));
336EXPECT_NE(new_leader_idx, leader_idx);
337
338int ret = OB_SUCCESS;
339++i;
340ref_scn.convert_for_logservice(i);
341ret = submit_log(new_leader, ref_scn, lsn, scn);
342EXPECT_EQ(OB_SUCCESS, ret);
343++i;
344ref_scn.convert_for_logservice(i);
345ret = submit_log(new_leader, ref_scn, lsn, scn);
346EXPECT_EQ(OB_SUCCESS, ret);
347
348LSN end_lsn;
349do {
350end_lsn = leader.palf_handle_impl_->get_end_lsn();
351if (end_lsn <= max_lsn) {
352CLOG_LOG(INFO, "wait end_lsn", K(end_lsn), K(max_lsn));
353sleep (1);
354}
355} while (end_lsn <= max_lsn);
356share::SCN max_decided_scn;
357EXPECT_EQ(OB_SUCCESS, ap_sv.get_max_applied_scn(ls_id, max_decided_scn));
358CLOG_LOG(INFO, "check here", K(max_scn), K(max_decided_scn), K(end_lsn), K(max_lsn), K(apply_end_lsn));
359EXPECT_EQ(max_decided_scn, max_scn);
360
361for (int i = 0; i < task_count; i++)
362{
363delete cb_array[i];
364}
365
366ObTenantEnv::set_tenant(get_cluster()[leader_idx]->get_tenant_base());
367EXPECT_EQ(OB_SUCCESS, ap_sv.remove_ls(ls_id));
368ap_sv.stop();
369ap_sv.wait();
370ap_sv.destroy();
371CLOG_LOG(INFO, "test get_max_decided_scn finish", K(id));
372}
373
374TEST_F(TestObSimpleLogApplyFunc, get_max_decided_scn_no_more_log_after_switch_to_follower)
375{
376const int64_t task_count = 10;
377const int64_t id = ATOMIC_AAF(&palf_id_, 1);
378ObLSID ls_id(id);
379int64_t leader_idx = 0;
380LSN basic_lsn(0);
381PalfHandleImplGuard leader;
382MockAppendCb *cb_array[task_count];
383CLOG_LOG(INFO, "test get_max_decided_scn_no_more_log_after_switch_to_follower begin", K(id));
384EXPECT_EQ(OB_SUCCESS, create_paxos_group(id, leader_idx, leader));
385MockLSAdapter ls_adapter;
386ls_adapter.init((ObLSService *)(0x1));
387ObLogApplyService ap_sv;
388//submit log first
389for (int i = 0; i < task_count; i++)
390{
391cb_array[i] = new MockAppendCb();
392cb_array[i]->init(i + 1, &ls_adapter);
393}
394PalfEnv *palf_env;
395ObTenantEnv::set_tenant(get_cluster()[leader_idx]->get_tenant_base());
396EXPECT_EQ(OB_SUCCESS, get_palf_env(leader_idx, palf_env));
397EXPECT_EQ(OB_SUCCESS, ap_sv.init(palf_env, &ls_adapter));
398EXPECT_EQ(OB_SUCCESS, ap_sv.start());
399EXPECT_EQ(OB_SUCCESS, ap_sv.add_ls(ls_id));
400EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_leader(ls_id, 1));
401const int64_t idx_1 = (leader_idx + 1) % get_node_cnt();
402const int64_t idx_2 = (leader_idx + 2) % get_node_cnt();
403share::SCN ref_scn;
404int i = 0;
405LSN lsn;
406share::SCN scn;
407
408do {
409ObApplyStatus *apply_status = NULL;
410ObApplyStatusGuard guard;
411ap_sv.get_apply_status(ls_id, guard);
412apply_status = guard.get_apply_status();
413ASSERT_TRUE(NULL != apply_status);
414int ret = OB_SUCCESS;
415for (i = 0; i < task_count; i++)
416{
417{
418ls_adapter.critical_guard();
419CLOG_LOG(INFO, "submit log start", K(i));
420ref_scn.convert_for_logservice(i);
421ret = submit_log(leader, ref_scn, lsn, scn);
422if (OB_SUCC(ret)) {
423EXPECT_EQ(true, lsn.is_valid());
424cb_array[i]->__set_lsn(lsn);
425cb_array[i]->__set_scn(scn);
426EXPECT_EQ(OB_SUCCESS, apply_status->push_append_cb(cb_array[i]));
427} else {
428break;
429}
430}
431CLOG_LOG(INFO, "submit log finish", K(i), K(lsn), K(scn));
432}
433} while (0);
434
435LSN max_lsn = leader.palf_handle_impl_->sw_.get_max_lsn();
436SCN max_scn = leader.palf_handle_impl_->sw_.get_max_scn();
437
438//wait apply done
439wait_lsn_until_slide(max_lsn, leader);
440EXPECT_EQ(OB_SUCCESS, ap_sv.switch_to_follower(ls_id));
441LSN apply_end_lsn;
442bool is_apply_done = false;
443while (!is_apply_done)
444{
445ap_sv.is_apply_done(ls_id, is_apply_done, apply_end_lsn);
446usleep(100);
447}
448EXPECT_EQ(max_lsn, apply_end_lsn);
449
450share::SCN max_decided_scn;
451EXPECT_EQ(OB_SUCCESS, ap_sv.get_max_applied_scn(ls_id, max_decided_scn));
452CLOG_LOG(INFO, "check here", K(max_scn), K(max_decided_scn), K(max_lsn), K(apply_end_lsn));
453EXPECT_EQ(max_decided_scn, max_scn);
454
455ObTenantEnv::set_tenant(get_cluster()[leader_idx]->get_tenant_base());
456EXPECT_EQ(OB_SUCCESS, ap_sv.remove_ls(ls_id));
457ap_sv.stop();
458ap_sv.wait();
459ap_sv.destroy();
460CLOG_LOG(INFO, "test get_max_decided_scn_with_no_more_log finish", K(id));
461}
462
463} // unitest
464} // oceanbase
465
466int main(int argc, char **argv)
467{
468RUN_SIMPLE_LOG_CLUSTER_TEST(TEST_NAME);
469}
470