2017-05-31 10:45:47 -07:00
|
|
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
2017-07-15 16:03:42 -07:00
|
|
|
// This source code is licensed under both the GPLv2 (found in the
|
|
|
|
// COPYING file in the root directory) and Apache 2.0 License
|
|
|
|
// (found in the LICENSE.Apache file in the root directory).
|
2017-05-31 10:45:47 -07:00
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
#include <thread>
|
|
|
|
#include <vector>
|
|
|
|
#include "db/db_test_util.h"
|
|
|
|
#include "db/write_batch_internal.h"
|
|
|
|
#include "port/stack_trace.h"
|
|
|
|
|
|
|
|
namespace rocksdb {
|
|
|
|
|
|
|
|
// Test variations of WriteImpl.
|
|
|
|
class DBWriteTest : public DBTestBase, public testing::WithParamInterface<int> {
|
|
|
|
public:
|
|
|
|
DBWriteTest() : DBTestBase("/db_write_test") {}
|
|
|
|
|
|
|
|
void Open() { DBTestBase::Reopen(GetOptions(GetParam())); }
|
|
|
|
};
|
|
|
|
|
2017-10-28 21:56:50 -07:00
|
|
|
// It is invalid to do sync write while disabling WAL.
|
|
|
|
TEST_P(DBWriteTest, SyncAndDisableWAL) {
|
|
|
|
WriteOptions write_options;
|
|
|
|
write_options.sync = true;
|
|
|
|
write_options.disableWAL = true;
|
|
|
|
ASSERT_TRUE(dbfull()->Put(write_options, "foo", "bar").IsInvalidArgument());
|
|
|
|
WriteBatch batch;
|
|
|
|
ASSERT_OK(batch.Put("foo", "bar"));
|
|
|
|
ASSERT_TRUE(dbfull()->Write(write_options, &batch).IsInvalidArgument());
|
|
|
|
}
|
|
|
|
|
2017-05-31 10:45:47 -07:00
|
|
|
// Sequence number should be return through input write batch.
|
|
|
|
TEST_P(DBWriteTest, ReturnSeuqneceNumber) {
|
|
|
|
Random rnd(4422);
|
|
|
|
Open();
|
|
|
|
for (int i = 0; i < 100; i++) {
|
|
|
|
WriteBatch batch;
|
|
|
|
batch.Put("key" + ToString(i), test::RandomHumanReadableString(&rnd, 10));
|
|
|
|
ASSERT_OK(dbfull()->Write(WriteOptions(), &batch));
|
|
|
|
ASSERT_EQ(dbfull()->GetLatestSequenceNumber(),
|
|
|
|
WriteBatchInternal::Sequence(&batch));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(DBWriteTest, ReturnSeuqneceNumberMultiThreaded) {
|
|
|
|
constexpr size_t kThreads = 16;
|
|
|
|
constexpr size_t kNumKeys = 1000;
|
|
|
|
Open();
|
|
|
|
ASSERT_EQ(0, dbfull()->GetLatestSequenceNumber());
|
|
|
|
// Check each sequence is used once and only once.
|
|
|
|
std::vector<std::atomic_flag> flags(kNumKeys * kThreads + 1);
|
|
|
|
for (size_t i = 0; i < flags.size(); i++) {
|
|
|
|
flags[i].clear();
|
|
|
|
}
|
|
|
|
auto writer = [&](size_t id) {
|
2017-06-05 12:06:23 -07:00
|
|
|
Random rnd(4422 + static_cast<uint32_t>(id));
|
2017-05-31 10:45:47 -07:00
|
|
|
for (size_t k = 0; k < kNumKeys; k++) {
|
|
|
|
WriteBatch batch;
|
|
|
|
batch.Put("key" + ToString(id) + "-" + ToString(k),
|
|
|
|
test::RandomHumanReadableString(&rnd, 10));
|
|
|
|
ASSERT_OK(dbfull()->Write(WriteOptions(), &batch));
|
|
|
|
SequenceNumber sequence = WriteBatchInternal::Sequence(&batch);
|
|
|
|
ASSERT_GT(sequence, 0);
|
|
|
|
ASSERT_LE(sequence, kNumKeys * kThreads);
|
|
|
|
// The sequence isn't consumed by someone else.
|
|
|
|
ASSERT_FALSE(flags[sequence].test_and_set());
|
|
|
|
}
|
|
|
|
};
|
2017-06-20 10:16:24 -07:00
|
|
|
std::vector<port::Thread> threads;
|
2017-05-31 10:45:47 -07:00
|
|
|
for (size_t i = 0; i < kThreads; i++) {
|
|
|
|
threads.emplace_back(writer, i);
|
|
|
|
}
|
|
|
|
for (size_t i = 0; i < kThreads; i++) {
|
|
|
|
threads[i].join();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(DBWriteTestInstance, DBWriteTest,
|
|
|
|
testing::Values(DBTestBase::kDefault,
|
2017-06-24 14:06:43 -07:00
|
|
|
DBTestBase::kConcurrentWALWrites,
|
2017-05-31 10:45:47 -07:00
|
|
|
DBTestBase::kPipelinedWrite));
|
|
|
|
|
|
|
|
} // namespace rocksdb
|
|
|
|
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
rocksdb::port::InstallStackTraceHandler();
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
|
|
|
}
|