2019-08-05 18:50:55 +03:00
|
|
|
//
|
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2020
|
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
2019-08-07 14:01:22 +03:00
|
|
|
#include "td/db/binlog/Binlog.h"
|
|
|
|
#include "td/db/binlog/BinlogEvent.h"
|
|
|
|
#include "td/db/binlog/BinlogHelper.h"
|
|
|
|
#include "td/db/TQueue.h"
|
|
|
|
|
2020-08-04 21:37:47 +03:00
|
|
|
#include "td/utils/buffer.h"
|
2019-08-05 18:50:55 +03:00
|
|
|
#include "td/utils/int_types.h"
|
2019-08-07 14:01:22 +03:00
|
|
|
#include "td/utils/misc.h"
|
|
|
|
#include "td/utils/port/path.h"
|
2020-08-04 22:48:23 +03:00
|
|
|
#include "td/utils/Random.h"
|
2019-08-05 18:50:55 +03:00
|
|
|
#include "td/utils/Slice.h"
|
|
|
|
#include "td/utils/Span.h"
|
2019-08-07 14:01:22 +03:00
|
|
|
#include "td/utils/Status.h"
|
2019-08-05 18:50:55 +03:00
|
|
|
#include "td/utils/tests.h"
|
2019-08-07 14:01:22 +03:00
|
|
|
#include "td/utils/VectorQueue.h"
|
|
|
|
|
2019-08-06 17:30:15 +03:00
|
|
|
#include <map>
|
|
|
|
#include <set>
|
|
|
|
|
2019-08-05 18:50:55 +03:00
|
|
|
TEST(TQueue, hands) {
|
2020-06-12 00:46:47 +03:00
|
|
|
td::TQueue::Event events[100];
|
|
|
|
auto events_span = td::MutableSpan<td::TQueue::Event>(events, 100);
|
2019-08-05 18:50:55 +03:00
|
|
|
|
2020-06-12 00:46:47 +03:00
|
|
|
auto tqueue = td::TQueue::create();
|
2019-08-05 18:50:55 +03:00
|
|
|
auto qid = 12;
|
2019-08-07 14:01:22 +03:00
|
|
|
ASSERT_EQ(true, tqueue->get_head(qid).empty());
|
|
|
|
ASSERT_EQ(true, tqueue->get_tail(qid).empty());
|
2020-06-12 03:02:20 +03:00
|
|
|
tqueue->push(qid, "hello", 0, 0, td::TQueue::EventId());
|
2019-08-07 14:01:22 +03:00
|
|
|
auto head = tqueue->get_head(qid);
|
2020-06-12 03:02:20 +03:00
|
|
|
auto tail = tqueue->get_tail(qid);
|
|
|
|
ASSERT_EQ(head.next().ok(), tail);
|
2019-08-27 18:06:00 +03:00
|
|
|
ASSERT_EQ(1u, tqueue->get(qid, head, true, 0, events_span).move_as_ok());
|
2020-06-12 03:02:20 +03:00
|
|
|
ASSERT_EQ(1u, tqueue->get(qid, head, true, 0, events_span).move_as_ok());
|
2020-07-25 00:37:45 +03:00
|
|
|
ASSERT_EQ(1u, tqueue->get(qid, tail, false, 0, events_span).move_as_ok());
|
2020-06-12 03:02:20 +03:00
|
|
|
ASSERT_EQ(1u, tqueue->get(qid, head, true, 0, events_span).move_as_ok());
|
|
|
|
ASSERT_EQ(0u, tqueue->get(qid, tail, true, 0, events_span).move_as_ok());
|
|
|
|
ASSERT_EQ(0u, tqueue->get(qid, head, true, 0, events_span).move_as_ok());
|
2019-08-06 17:30:15 +03:00
|
|
|
}
|
|
|
|
|
2019-08-06 21:24:28 +03:00
|
|
|
class TestTQueue {
|
|
|
|
public:
|
2020-06-12 00:46:47 +03:00
|
|
|
using EventId = td::TQueue::EventId;
|
|
|
|
|
2020-06-12 03:53:04 +03:00
|
|
|
static td::CSlice binlog_path() {
|
|
|
|
return td::CSlice("tqueue_binlog");
|
2019-08-06 21:24:28 +03:00
|
|
|
}
|
2020-06-12 03:53:04 +03:00
|
|
|
|
2019-08-06 21:24:28 +03:00
|
|
|
TestTQueue() {
|
2020-06-12 00:46:47 +03:00
|
|
|
baseline_ = td::TQueue::create();
|
2019-08-07 14:01:22 +03:00
|
|
|
|
2020-06-12 03:53:04 +03:00
|
|
|
memory_ = td::TQueue::create();
|
2020-06-12 00:46:47 +03:00
|
|
|
auto memory_storage = td::make_unique<td::TQueueMemoryStorage>();
|
2019-08-06 21:24:28 +03:00
|
|
|
memory_storage_ = memory_storage.get();
|
2019-08-07 14:01:22 +03:00
|
|
|
memory_->set_callback(std::move(memory_storage));
|
2019-08-06 21:24:28 +03:00
|
|
|
|
2020-06-12 00:46:47 +03:00
|
|
|
binlog_ = td::TQueue::create();
|
|
|
|
auto tqueue_binlog = td::make_unique<td::TQueueBinlog<td::Binlog>>();
|
|
|
|
td::Binlog::destroy(binlog_path()).ensure();
|
|
|
|
auto binlog = std::make_shared<td::Binlog>();
|
|
|
|
binlog->init(binlog_path().str(), [&](const td::BinlogEvent &event) { UNREACHABLE(); }).ensure();
|
2020-06-12 03:53:04 +03:00
|
|
|
tqueue_binlog->set_binlog(std::move(binlog));
|
2019-08-07 14:01:22 +03:00
|
|
|
binlog_->set_callback(std::move(tqueue_binlog));
|
2019-08-06 21:24:28 +03:00
|
|
|
}
|
2019-08-06 17:30:15 +03:00
|
|
|
|
2020-06-12 00:46:47 +03:00
|
|
|
void restart(td::Random::Xorshift128plus &rnd, double now) {
|
2019-08-09 20:29:17 +03:00
|
|
|
if (rnd.fast(0, 10) == 0) {
|
|
|
|
baseline_->run_gc(now);
|
|
|
|
}
|
2019-08-07 18:29:47 +03:00
|
|
|
|
2019-08-07 14:01:22 +03:00
|
|
|
memory_->extract_callback().release();
|
2020-06-12 00:46:47 +03:00
|
|
|
auto memory_storage = td::unique_ptr<td::TQueueMemoryStorage>(memory_storage_);
|
|
|
|
memory_ = td::TQueue::create();
|
2019-08-07 14:01:22 +03:00
|
|
|
memory_storage->replay(*memory_);
|
|
|
|
memory_->set_callback(std::move(memory_storage));
|
2019-08-09 20:29:17 +03:00
|
|
|
if (rnd.fast(0, 10) == 0) {
|
|
|
|
memory_->run_gc(now);
|
|
|
|
}
|
2019-08-06 17:30:15 +03:00
|
|
|
|
2020-06-12 03:53:04 +03:00
|
|
|
if (rnd.fast(0, 30) != 0) {
|
2019-08-06 21:24:28 +03:00
|
|
|
return;
|
|
|
|
}
|
2019-08-06 17:30:15 +03:00
|
|
|
|
2020-06-11 23:54:56 +03:00
|
|
|
LOG(INFO) << "Restart binlog";
|
2020-06-12 00:46:47 +03:00
|
|
|
binlog_ = td::TQueue::create();
|
|
|
|
auto tqueue_binlog = td::make_unique<td::TQueueBinlog<td::Binlog>>();
|
|
|
|
auto binlog = std::make_shared<td::Binlog>();
|
2020-07-24 06:57:48 +03:00
|
|
|
binlog
|
|
|
|
->init(binlog_path().str(),
|
|
|
|
[&](const td::BinlogEvent &event) { tqueue_binlog->replay(event, *binlog_).ignore(); })
|
2019-08-06 21:24:28 +03:00
|
|
|
.ensure();
|
2020-06-12 03:53:04 +03:00
|
|
|
tqueue_binlog->set_binlog(std::move(binlog));
|
2019-08-07 14:01:22 +03:00
|
|
|
binlog_->set_callback(std::move(tqueue_binlog));
|
2020-06-12 03:53:04 +03:00
|
|
|
if (rnd.fast(0, 2) == 0) {
|
2019-08-09 20:29:17 +03:00
|
|
|
binlog_->run_gc(now);
|
|
|
|
}
|
2019-08-06 21:24:28 +03:00
|
|
|
}
|
2019-08-06 17:30:15 +03:00
|
|
|
|
2020-06-12 00:46:47 +03:00
|
|
|
EventId push(td::TQueue::QueueId queue_id, td::string data, double expires_at, EventId new_id = EventId()) {
|
|
|
|
auto a_id = baseline_->push(queue_id, data, expires_at, 0, new_id).move_as_ok();
|
|
|
|
auto b_id = memory_->push(queue_id, data, expires_at, 0, new_id).move_as_ok();
|
|
|
|
auto c_id = binlog_->push(queue_id, data, expires_at, 0, new_id).move_as_ok();
|
2019-08-06 21:24:28 +03:00
|
|
|
ASSERT_EQ(a_id, b_id);
|
|
|
|
ASSERT_EQ(a_id, c_id);
|
|
|
|
return a_id;
|
|
|
|
}
|
2019-08-06 17:30:15 +03:00
|
|
|
|
2020-06-12 00:46:47 +03:00
|
|
|
void check_head_tail(td::TQueue::QueueId qid, double now) {
|
2019-08-09 20:29:17 +03:00
|
|
|
//ASSERT_EQ(baseline_->get_head(qid), memory_->get_head(qid));
|
|
|
|
//ASSERT_EQ(baseline_->get_head(qid), binlog_->get_head(qid));
|
2019-08-07 14:01:22 +03:00
|
|
|
ASSERT_EQ(baseline_->get_tail(qid), memory_->get_tail(qid));
|
|
|
|
ASSERT_EQ(baseline_->get_tail(qid), binlog_->get_tail(qid));
|
2019-08-06 21:24:28 +03:00
|
|
|
}
|
|
|
|
|
2020-06-12 00:46:47 +03:00
|
|
|
void check_get(td::TQueue::QueueId qid, td::Random::Xorshift128plus &rnd, double now) {
|
|
|
|
td::TQueue::Event a[10];
|
|
|
|
td::MutableSpan<td::TQueue::Event> a_span(a, 10);
|
|
|
|
td::TQueue::Event b[10];
|
|
|
|
td::MutableSpan<td::TQueue::Event> b_span(b, 10);
|
|
|
|
td::TQueue::Event c[10];
|
|
|
|
td::MutableSpan<td::TQueue::Event> c_span(c, 10);
|
2019-08-06 21:24:28 +03:00
|
|
|
|
2019-08-07 14:01:22 +03:00
|
|
|
auto a_from = baseline_->get_head(qid);
|
2019-08-09 20:29:17 +03:00
|
|
|
//auto b_from = memory_->get_head(qid);
|
|
|
|
//auto c_from = binlog_->get_head(qid);
|
|
|
|
//ASSERT_EQ(a_from, b_from);
|
|
|
|
//ASSERT_EQ(a_from, c_from);
|
2019-08-06 17:30:15 +03:00
|
|
|
|
|
|
|
auto tmp = a_from.advance(rnd.fast(-10, 10));
|
|
|
|
if (tmp.is_ok()) {
|
|
|
|
a_from = tmp.move_as_ok();
|
|
|
|
}
|
2019-08-27 18:06:00 +03:00
|
|
|
baseline_->get(qid, a_from, true, now, a_span).move_as_ok();
|
|
|
|
memory_->get(qid, a_from, true, now, b_span).move_as_ok();
|
|
|
|
binlog_->get(qid, a_from, true, now, c_span).move_as_ok();
|
2019-08-07 18:13:10 +03:00
|
|
|
ASSERT_EQ(a_span.size(), b_span.size());
|
|
|
|
ASSERT_EQ(a_span.size(), c_span.size());
|
|
|
|
for (size_t i = 0; i < a_span.size(); i++) {
|
2019-08-06 21:24:28 +03:00
|
|
|
ASSERT_EQ(a_span[i].id, b_span[i].id);
|
|
|
|
ASSERT_EQ(a_span[i].id, c_span[i].id);
|
|
|
|
ASSERT_EQ(a_span[i].data, b_span[i].data);
|
|
|
|
ASSERT_EQ(a_span[i].data, c_span[i].data);
|
2019-08-06 17:30:15 +03:00
|
|
|
}
|
2019-08-06 21:24:28 +03:00
|
|
|
}
|
2019-08-06 17:30:15 +03:00
|
|
|
|
2019-08-06 21:24:28 +03:00
|
|
|
private:
|
2020-06-12 00:46:47 +03:00
|
|
|
td::unique_ptr<td::TQueue> baseline_;
|
|
|
|
td::unique_ptr<td::TQueue> memory_;
|
|
|
|
td::unique_ptr<td::TQueue> binlog_;
|
|
|
|
td::TQueueMemoryStorage *memory_storage_{nullptr};
|
2019-08-06 21:24:28 +03:00
|
|
|
};
|
2019-08-06 17:30:15 +03:00
|
|
|
|
2019-08-06 21:24:28 +03:00
|
|
|
TEST(TQueue, random) {
|
2020-06-12 00:46:47 +03:00
|
|
|
using EventId = td::TQueue::EventId;
|
|
|
|
td::Random::Xorshift128plus rnd(123);
|
2020-06-12 03:53:04 +03:00
|
|
|
auto next_queue_id = [&rnd] {
|
2019-08-07 14:01:22 +03:00
|
|
|
return rnd.fast(1, 10);
|
|
|
|
};
|
2020-06-12 03:53:04 +03:00
|
|
|
auto next_first_id = [&rnd] {
|
|
|
|
if (rnd.fast(0, 3) == 0) {
|
|
|
|
return EventId::from_int32(EventId::MAX_ID - 20).move_as_ok();
|
|
|
|
}
|
|
|
|
return EventId::from_int32(rnd.fast(1000000000, 1500000000)).move_as_ok();
|
2019-08-06 21:24:28 +03:00
|
|
|
};
|
2020-06-12 03:53:04 +03:00
|
|
|
|
2019-08-06 21:24:28 +03:00
|
|
|
TestTQueue q;
|
2019-08-09 20:29:17 +03:00
|
|
|
double now = 0;
|
2019-08-06 21:24:28 +03:00
|
|
|
auto push_event = [&] {
|
|
|
|
auto data = PSTRING() << rnd();
|
2020-06-12 03:53:04 +03:00
|
|
|
if (rnd.fast(0, 10000) == 0) {
|
|
|
|
data = td::string(1 << 19, '\0');
|
|
|
|
}
|
|
|
|
q.push(next_queue_id(), data, now + rnd.fast(-10, 10) * 10 + 5, next_first_id());
|
2019-08-09 20:29:17 +03:00
|
|
|
};
|
|
|
|
auto inc_now = [&] {
|
|
|
|
now += 10;
|
2019-08-06 21:24:28 +03:00
|
|
|
};
|
2019-08-07 14:01:22 +03:00
|
|
|
auto check_head_tail = [&] {
|
2020-06-12 03:53:04 +03:00
|
|
|
q.check_head_tail(next_queue_id(), now);
|
2019-08-07 14:01:22 +03:00
|
|
|
};
|
|
|
|
auto restart = [&] {
|
2019-09-06 18:55:19 +03:00
|
|
|
q.restart(rnd, now);
|
2019-08-07 14:01:22 +03:00
|
|
|
};
|
|
|
|
auto get = [&] {
|
2020-06-12 03:53:04 +03:00
|
|
|
q.check_get(next_queue_id(), rnd, now);
|
2019-08-07 14:01:22 +03:00
|
|
|
};
|
2020-06-12 00:46:47 +03:00
|
|
|
td::RandomSteps steps({{push_event, 100}, {check_head_tail, 10}, {get, 40}, {inc_now, 5}, {restart, 1}});
|
2020-06-11 23:54:56 +03:00
|
|
|
for (int i = 0; i < 100000; i++) {
|
2019-08-06 17:30:15 +03:00
|
|
|
steps.step(rnd);
|
|
|
|
}
|
2019-08-05 18:50:55 +03:00
|
|
|
}
|
2020-08-04 21:37:47 +03:00
|
|
|
|
|
|
|
TEST(TQueue, memory_leak) {
|
2020-08-04 22:48:23 +03:00
|
|
|
return;
|
2020-08-04 21:37:47 +03:00
|
|
|
auto tqueue = td::TQueue::create();
|
|
|
|
auto tqueue_binlog = td::make_unique<td::TQueueBinlog<td::Binlog>>();
|
|
|
|
std::string binlog_path = "test_tqueue.binlog";
|
|
|
|
td::Binlog::destroy(binlog_path).ensure();
|
|
|
|
auto binlog = std::make_shared<td::Binlog>();
|
|
|
|
binlog->init(binlog_path, [&](const td::BinlogEvent &event) { UNREACHABLE(); }).ensure();
|
|
|
|
tqueue_binlog->set_binlog(std::move(binlog));
|
|
|
|
tqueue->set_callback(std::move(tqueue_binlog));
|
|
|
|
|
|
|
|
double now = 0;
|
|
|
|
std::vector<td::TQueue::EventId> ids;
|
|
|
|
td::Random::Xorshift128plus rnd(123);
|
|
|
|
int i = 0;
|
|
|
|
while (true) {
|
|
|
|
auto id = tqueue->push(1, "a", now + 600000, 0, {}).move_as_ok();
|
|
|
|
ids.push_back(id);
|
2020-08-04 22:48:23 +03:00
|
|
|
if (ids.size() > static_cast<std::size_t>(rnd()) % 100000) {
|
|
|
|
auto it = static_cast<std::size_t>(rnd()) % ids.size();
|
2020-08-04 21:37:47 +03:00
|
|
|
std::swap(ids.back(), ids[it]);
|
|
|
|
tqueue->forget(1, ids.back());
|
|
|
|
ids.pop_back();
|
|
|
|
}
|
|
|
|
now += 1;
|
|
|
|
if (i++ % 100000 == 0) {
|
|
|
|
LOG(ERROR) << td::BufferAllocator::get_buffer_mem() << " " << tqueue->get_size(1) << " "
|
|
|
|
<< td::BufferAllocator::get_buffer_slice_size();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|