|
|
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
|
|
|
// 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).
|
|
|
|
|
|
|
|
#ifndef ROCKSDB_LITE
|
|
|
|
|
|
|
|
#ifndef __STDC_FORMAT_MACROS
|
|
|
|
#define __STDC_FORMAT_MACROS
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "utilities/transactions/transaction_test.h"
|
|
|
|
#include "utilities/transactions/write_unprepared_txn.h"
|
|
|
|
#include "utilities/transactions/write_unprepared_txn_db.h"
|
|
|
|
|
|
|
|
namespace rocksdb {
|
|
|
|
|
|
|
|
class WriteUnpreparedTransactionTestBase : public TransactionTestBase {
|
|
|
|
public:
|
|
|
|
WriteUnpreparedTransactionTestBase(bool use_stackable_db,
|
|
|
|
bool two_write_queue,
|
|
|
|
TxnDBWritePolicy write_policy)
|
|
|
|
: TransactionTestBase(use_stackable_db, two_write_queue, write_policy){}
|
|
|
|
};
|
|
|
|
|
|
|
|
class WriteUnpreparedTransactionTest
|
|
|
|
: public WriteUnpreparedTransactionTestBase,
|
|
|
|
virtual public ::testing::WithParamInterface<
|
|
|
|
std::tuple<bool, bool, TxnDBWritePolicy>> {
|
|
|
|
public:
|
|
|
|
WriteUnpreparedTransactionTest()
|
|
|
|
: WriteUnpreparedTransactionTestBase(std::get<0>(GetParam()),
|
|
|
|
std::get<1>(GetParam()),
|
|
|
|
std::get<2>(GetParam())){}
|
|
|
|
};
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
|
|
WriteUnpreparedTransactionTest, WriteUnpreparedTransactionTest,
|
|
|
|
::testing::Values(std::make_tuple(false, false, WRITE_UNPREPARED),
|
|
|
|
std::make_tuple(false, true, WRITE_UNPREPARED)));
|
|
|
|
|
|
|
|
TEST_P(WriteUnpreparedTransactionTest, ReadYourOwnWrite) {
|
|
|
|
auto verify_state = [](Iterator* iter, const std::string& key,
|
|
|
|
const std::string& value) {
|
|
|
|
ASSERT_TRUE(iter->Valid());
|
|
|
|
ASSERT_OK(iter->status());
|
|
|
|
ASSERT_EQ(key, iter->key().ToString());
|
|
|
|
ASSERT_EQ(value, iter->value().ToString());
|
|
|
|
};
|
|
|
|
|
|
|
|
options.disable_auto_compactions = true;
|
|
|
|
ReOpen();
|
|
|
|
|
|
|
|
// The following tests checks whether reading your own write for
|
|
|
|
// a transaction works for write unprepared, when there are uncommitted
|
|
|
|
// values written into DB.
|
|
|
|
//
|
|
|
|
// Although the values written by DB::Put are technically committed, we add
|
|
|
|
// their seq num to unprep_seqs_ to pretend that they were written into DB
|
|
|
|
// as part of an unprepared batch, and then check if they are visible to the
|
|
|
|
// transaction.
|
|
|
|
auto snapshot0 = db->GetSnapshot();
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "a", "v1"));
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "b", "v2"));
|
|
|
|
auto snapshot2 = db->GetSnapshot();
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "a", "v3"));
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "b", "v4"));
|
|
|
|
auto snapshot4 = db->GetSnapshot();
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "a", "v5"));
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "b", "v6"));
|
|
|
|
auto snapshot6 = db->GetSnapshot();
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "a", "v7"));
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "b", "v8"));
|
|
|
|
auto snapshot8 = db->GetSnapshot();
|
|
|
|
|
|
|
|
TransactionOptions txn_options;
|
|
|
|
WriteOptions write_options;
|
|
|
|
Transaction* txn = db->BeginTransaction(write_options, txn_options);
|
|
|
|
WriteUnpreparedTxn* wup_txn = dynamic_cast<WriteUnpreparedTxn*>(txn);
|
|
|
|
|
|
|
|
ReadOptions roptions;
|
|
|
|
roptions.snapshot = snapshot0;
|
|
|
|
|
|
|
|
auto iter = txn->GetIterator(roptions);
|
|
|
|
|
|
|
|
// Test Get().
|
|
|
|
std::string value;
|
|
|
|
wup_txn->unprep_seqs_[snapshot2->GetSequenceNumber() + 1] =
|
|
|
|
snapshot4->GetSequenceNumber() - snapshot2->GetSequenceNumber();
|
|
|
|
|
|
|
|
ASSERT_OK(txn->Get(roptions, Slice("a"), &value));
|
|
|
|
ASSERT_EQ(value, "v3");
|
|
|
|
|
|
|
|
ASSERT_OK(txn->Get(roptions, Slice("b"), &value));
|
|
|
|
ASSERT_EQ(value, "v4");
|
|
|
|
|
|
|
|
wup_txn->unprep_seqs_[snapshot6->GetSequenceNumber() + 1] =
|
|
|
|
snapshot8->GetSequenceNumber() - snapshot6->GetSequenceNumber();
|
|
|
|
|
|
|
|
ASSERT_OK(txn->Get(roptions, Slice("a"), &value));
|
|
|
|
ASSERT_EQ(value, "v7");
|
|
|
|
|
|
|
|
ASSERT_OK(txn->Get(roptions, Slice("b"), &value));
|
|
|
|
ASSERT_EQ(value, "v8");
|
|
|
|
|
|
|
|
wup_txn->unprep_seqs_.clear();
|
|
|
|
|
|
|
|
// Test Next().
|
|
|
|
wup_txn->unprep_seqs_[snapshot2->GetSequenceNumber() + 1] =
|
|
|
|
snapshot4->GetSequenceNumber() - snapshot2->GetSequenceNumber();
|
|
|
|
|
|
|
|
iter->Seek("a");
|
|
|
|
verify_state(iter, "a", "v3");
|
|
|
|
|
|
|
|
iter->Next();
|
|
|
|
verify_state(iter, "b", "v4");
|
|
|
|
|
|
|
|
iter->SeekToFirst();
|
|
|
|
verify_state(iter, "a", "v3");
|
|
|
|
|
|
|
|
iter->Next();
|
|
|
|
verify_state(iter, "b", "v4");
|
|
|
|
|
|
|
|
wup_txn->unprep_seqs_[snapshot6->GetSequenceNumber() + 1] =
|
|
|
|
snapshot8->GetSequenceNumber() - snapshot6->GetSequenceNumber();
|
|
|
|
|
|
|
|
iter->Seek("a");
|
|
|
|
verify_state(iter, "a", "v7");
|
|
|
|
|
|
|
|
iter->Next();
|
|
|
|
verify_state(iter, "b", "v8");
|
|
|
|
|
|
|
|
iter->SeekToFirst();
|
|
|
|
verify_state(iter, "a", "v7");
|
|
|
|
|
|
|
|
iter->Next();
|
|
|
|
verify_state(iter, "b", "v8");
|
|
|
|
|
|
|
|
wup_txn->unprep_seqs_.clear();
|
|
|
|
|
|
|
|
// Test Prev(). For Prev(), we need to adjust the snapshot to match what is
|
|
|
|
// possible in WriteUnpreparedTxn.
|
|
|
|
//
|
|
|
|
// Because of row locks and ValidateSnapshot, there cannot be any committed
|
|
|
|
// entries after snapshot, but before the first prepared key.
|
|
|
|
delete iter;
|
|
|
|
roptions.snapshot = snapshot2;
|
|
|
|
iter = txn->GetIterator(roptions);
|
|
|
|
wup_txn->unprep_seqs_[snapshot2->GetSequenceNumber() + 1] =
|
|
|
|
snapshot4->GetSequenceNumber() - snapshot2->GetSequenceNumber();
|
|
|
|
|
|
|
|
iter->SeekForPrev("b");
|
|
|
|
verify_state(iter, "b", "v4");
|
|
|
|
|
|
|
|
iter->Prev();
|
|
|
|
verify_state(iter, "a", "v3");
|
|
|
|
|
|
|
|
iter->SeekToLast();
|
|
|
|
verify_state(iter, "b", "v4");
|
|
|
|
|
|
|
|
iter->Prev();
|
|
|
|
verify_state(iter, "a", "v3");
|
|
|
|
|
|
|
|
delete iter;
|
|
|
|
roptions.snapshot = snapshot6;
|
|
|
|
iter = txn->GetIterator(roptions);
|
|
|
|
wup_txn->unprep_seqs_[snapshot6->GetSequenceNumber() + 1] =
|
|
|
|
snapshot8->GetSequenceNumber() - snapshot6->GetSequenceNumber();
|
|
|
|
|
|
|
|
iter->SeekForPrev("b");
|
|
|
|
verify_state(iter, "b", "v8");
|
|
|
|
|
|
|
|
iter->Prev();
|
|
|
|
verify_state(iter, "a", "v7");
|
|
|
|
|
|
|
|
iter->SeekToLast();
|
|
|
|
verify_state(iter, "b", "v8");
|
|
|
|
|
|
|
|
iter->Prev();
|
|
|
|
verify_state(iter, "a", "v7");
|
|
|
|
|
|
|
|
db->ReleaseSnapshot(snapshot0);
|
|
|
|
db->ReleaseSnapshot(snapshot2);
|
|
|
|
db->ReleaseSnapshot(snapshot4);
|
|
|
|
db->ReleaseSnapshot(snapshot6);
|
|
|
|
db->ReleaseSnapshot(snapshot8);
|
|
|
|
delete iter;
|
|
|
|
delete txn;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(WriteUnpreparedTransactionTest, RecoveryRollbackUnprepared) {
|
|
|
|
WriteOptions write_options;
|
|
|
|
write_options.disableWAL = false;
|
|
|
|
uint64_t seq_used = kMaxSequenceNumber;
|
|
|
|
uint64_t log_number;
|
|
|
|
WriteBatch batch;
|
|
|
|
std::vector<Transaction*> prepared_trans;
|
|
|
|
WriteUnpreparedTxnDB* wup_db;
|
|
|
|
options.disable_auto_compactions = true;
|
|
|
|
|
|
|
|
// Try unprepared batches with empty database.
|
|
|
|
for (int num_batches = 0; num_batches < 10; num_batches++) {
|
|
|
|
// Reset database.
|
|
|
|
prepared_trans.clear();
|
|
|
|
ReOpen();
|
|
|
|
wup_db = dynamic_cast<WriteUnpreparedTxnDB*>(db);
|
|
|
|
|
|
|
|
// Write num_batches unprepared batches into the WAL.
|
|
|
|
for (int i = 0; i < num_batches; i++) {
|
|
|
|
batch.Clear();
|
|
|
|
// TODO(lth): Instead of manually calling WriteImpl with a write batch,
|
|
|
|
// use methods on Transaction instead once it is implemented.
|
|
|
|
ASSERT_OK(WriteBatchInternal::InsertNoop(&batch));
|
|
|
|
ASSERT_OK(WriteBatchInternal::Put(&batch,
|
|
|
|
db->DefaultColumnFamily()->GetID(),
|
|
|
|
"k" + ToString(i), "value"));
|
|
|
|
// MarkEndPrepare will change the Noop marker into an unprepared marker.
|
|
|
|
ASSERT_OK(WriteBatchInternal::MarkEndPrepare(
|
|
|
|
&batch, Slice("xid1"), /* write after commit */ false,
|
|
|
|
/* unprepared batch */ true));
|
|
|
|
ASSERT_OK(wup_db->db_impl_->WriteImpl(
|
|
|
|
write_options, &batch, /*callback*/ nullptr, &log_number,
|
|
|
|
/*log ref*/ 0, /* disable memtable */ true, &seq_used,
|
|
|
|
/* prepare_batch_cnt_ */ 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Crash and run recovery code paths.
|
|
|
|
wup_db->db_impl_->FlushWAL(true);
|
|
|
|
wup_db->TEST_Crash();
|
|
|
|
ReOpenNoDelete();
|
|
|
|
wup_db = dynamic_cast<WriteUnpreparedTxnDB*>(db);
|
|
|
|
|
|
|
|
db->GetAllPreparedTransactions(&prepared_trans);
|
|
|
|
ASSERT_EQ(prepared_trans.size(), 0);
|
|
|
|
|
|
|
|
// Check that DB is empty.
|
|
|
|
Iterator* iter = db->NewIterator(ReadOptions());
|
|
|
|
iter->SeekToFirst();
|
|
|
|
ASSERT_FALSE(iter->Valid());
|
|
|
|
delete iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try unprepared batches with non-empty database.
|
|
|
|
for (int num_batches = 1; num_batches < 10; num_batches++) {
|
|
|
|
// Reset database.
|
|
|
|
prepared_trans.clear();
|
|
|
|
ReOpen();
|
|
|
|
wup_db = dynamic_cast<WriteUnpreparedTxnDB*>(db);
|
|
|
|
for (int i = 0; i < num_batches; i++) {
|
|
|
|
ASSERT_OK(db->Put(WriteOptions(), "k" + ToString(i),
|
|
|
|
"before value " + ToString(i)));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write num_batches unprepared batches into the WAL.
|
|
|
|
for (int i = 0; i < num_batches; i++) {
|
|
|
|
batch.Clear();
|
|
|
|
// TODO(lth): Instead of manually calling WriteImpl with a write batch,
|
|
|
|
// use methods on Transaction instead once it is implemented.
|
|
|
|
ASSERT_OK(WriteBatchInternal::InsertNoop(&batch));
|
|
|
|
ASSERT_OK(WriteBatchInternal::Put(&batch,
|
|
|
|
db->DefaultColumnFamily()->GetID(),
|
|
|
|
"k" + ToString(i), "value"));
|
|
|
|
// MarkEndPrepare will change the Noop marker into an unprepared marker.
|
|
|
|
ASSERT_OK(WriteBatchInternal::MarkEndPrepare(
|
|
|
|
&batch, Slice("xid1"), /* write after commit */ false,
|
|
|
|
/* unprepared batch */ true));
|
|
|
|
ASSERT_OK(wup_db->db_impl_->WriteImpl(
|
|
|
|
write_options, &batch, /*callback*/ nullptr, &log_number,
|
|
|
|
/*log ref*/ 0, /* disable memtable */ true, &seq_used,
|
|
|
|
/* prepare_batch_cnt_ */ 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Crash and run recovery code paths.
|
|
|
|
wup_db->db_impl_->FlushWAL(true);
|
|
|
|
wup_db->TEST_Crash();
|
|
|
|
ReOpenNoDelete();
|
|
|
|
wup_db = dynamic_cast<WriteUnpreparedTxnDB*>(db);
|
|
|
|
|
|
|
|
db->GetAllPreparedTransactions(&prepared_trans);
|
|
|
|
ASSERT_EQ(prepared_trans.size(), 0);
|
|
|
|
|
|
|
|
// Check that DB has before values.
|
|
|
|
Iterator* iter = db->NewIterator(ReadOptions());
|
|
|
|
iter->SeekToFirst();
|
|
|
|
for (int i = 0; i < num_batches; i++) {
|
|
|
|
ASSERT_TRUE(iter->Valid());
|
|
|
|
ASSERT_EQ(iter->key().ToString(), "k" + ToString(i));
|
|
|
|
ASSERT_EQ(iter->value().ToString(), "before value " + ToString(i));
|
|
|
|
iter->Next();
|
|
|
|
}
|
|
|
|
ASSERT_FALSE(iter->Valid());
|
|
|
|
delete iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace rocksdb
|
|
|
|
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
int main(int /*argc*/, char** /*argv*/) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"SKIPPED as Transactions are not supported in ROCKSDB_LITE\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // ROCKSDB_LITE
|