2016-02-09 15:12:00 -08: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).
|
2014-05-08 17:25:13 -07:00
|
|
|
|
2014-11-12 13:05:12 -08:00
|
|
|
#ifndef ROCKSDB_LITE
|
2014-05-08 17:25:13 -07:00
|
|
|
#ifndef GFLAGS
|
|
|
|
#include <cstdio>
|
|
|
|
int main() {
|
|
|
|
fprintf(stderr, "Please install gflags to run rocksdb tools\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
|
2012-12-11 11:57:35 -08:00
|
|
|
#include <cstdio>
|
2014-10-27 15:12:20 -07:00
|
|
|
#include <atomic>
|
2012-12-11 11:57:35 -08:00
|
|
|
|
2013-10-24 07:43:14 -07:00
|
|
|
#include <gflags/gflags.h>
|
|
|
|
|
2012-12-11 11:57:35 -08:00
|
|
|
#include "db/write_batch_internal.h"
|
2013-08-23 08:38:13 -07:00
|
|
|
#include "rocksdb/db.h"
|
|
|
|
#include "rocksdb/types.h"
|
2012-12-11 11:57:35 -08:00
|
|
|
#include "util/testutil.h"
|
|
|
|
|
|
|
|
// Run a thread to perform Put's.
|
|
|
|
// Another thread uses GetUpdatesSince API to keep getting the updates.
|
|
|
|
// options :
|
|
|
|
// --num_inserts = the num of inserts the first thread should perform.
|
|
|
|
// --wal_ttl = the wal ttl for the run.
|
|
|
|
|
2013-10-03 21:49:15 -07:00
|
|
|
using namespace rocksdb;
|
2012-12-11 11:57:35 -08:00
|
|
|
|
2014-05-08 17:25:13 -07:00
|
|
|
using GFLAGS::ParseCommandLineFlags;
|
|
|
|
using GFLAGS::SetUsageMessage;
|
|
|
|
|
2012-12-11 11:57:35 -08:00
|
|
|
struct DataPumpThread {
|
|
|
|
size_t no_records;
|
|
|
|
DB* db; // Assumption DB is Open'ed already.
|
|
|
|
};
|
|
|
|
|
|
|
|
static std::string RandomString(Random* rnd, int len) {
|
|
|
|
std::string r;
|
|
|
|
test::RandomString(rnd, len, &r);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void DataPumpThreadBody(void* arg) {
|
|
|
|
DataPumpThread* t = reinterpret_cast<DataPumpThread*>(arg);
|
|
|
|
DB* db = t->db;
|
|
|
|
Random rnd(301);
|
|
|
|
size_t i = 0;
|
2013-08-06 12:54:37 -07:00
|
|
|
while(i++ < t->no_records) {
|
|
|
|
if(!db->Put(WriteOptions(), Slice(RandomString(&rnd, 500)),
|
|
|
|
Slice(RandomString(&rnd, 500))).ok()) {
|
|
|
|
fprintf(stderr, "Error in put\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2012-12-11 11:57:35 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ReplicationThread {
|
2014-10-27 14:50:21 -07:00
|
|
|
std::atomic<bool> stop;
|
2012-12-11 11:57:35 -08:00
|
|
|
DB* db;
|
|
|
|
volatile size_t no_read;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void ReplicationThreadBody(void* arg) {
|
|
|
|
ReplicationThread* t = reinterpret_cast<ReplicationThread*>(arg);
|
|
|
|
DB* db = t->db;
|
2013-01-24 10:54:26 -08:00
|
|
|
unique_ptr<TransactionLogIterator> iter;
|
2013-08-06 12:54:37 -07:00
|
|
|
SequenceNumber currentSeqNum = 1;
|
2014-10-27 14:50:21 -07:00
|
|
|
while (!t->stop.load(std::memory_order_acquire)) {
|
2013-08-06 12:54:37 -07:00
|
|
|
iter.reset();
|
|
|
|
Status s;
|
|
|
|
while(!db->GetUpdatesSince(currentSeqNum, &iter).ok()) {
|
2014-10-27 14:50:21 -07:00
|
|
|
if (t->stop.load(std::memory_order_acquire)) {
|
2013-08-06 12:54:37 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr, "Refreshing iterator\n");
|
2014-03-12 09:37:43 -07:00
|
|
|
for(;iter->Valid(); iter->Next(), t->no_read++, currentSeqNum++) {
|
2013-08-06 12:54:37 -07:00
|
|
|
BatchResult res = iter->GetBatch();
|
|
|
|
if (res.sequence != currentSeqNum) {
|
2013-11-16 23:44:39 -08:00
|
|
|
fprintf(stderr,
|
|
|
|
"Missed a seq no. b/w %ld and %ld\n",
|
|
|
|
(long)currentSeqNum,
|
|
|
|
(long)res.sequence);
|
2013-08-06 12:54:37 -07:00
|
|
|
exit(1);
|
2012-12-11 11:57:35 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-24 07:43:14 -07:00
|
|
|
DEFINE_uint64(num_inserts, 1000, "the num of inserts the first thread should"
|
|
|
|
" perform.");
|
2013-11-06 18:46:28 -08:00
|
|
|
DEFINE_uint64(wal_ttl_seconds, 1000, "the wal ttl for the run(in seconds)");
|
|
|
|
DEFINE_uint64(wal_size_limit_MB, 10, "the wal size limit for the run"
|
|
|
|
"(in MB)");
|
2012-12-11 11:57:35 -08:00
|
|
|
|
2013-10-24 07:43:14 -07:00
|
|
|
int main(int argc, const char** argv) {
|
2014-05-08 17:25:13 -07:00
|
|
|
SetUsageMessage(
|
|
|
|
std::string("\nUSAGE:\n") + std::string(argv[0]) +
|
|
|
|
" --num_inserts=<num_inserts> --wal_ttl_seconds=<WAL_ttl_seconds>" +
|
|
|
|
" --wal_size_limit_MB=<WAL_size_limit_MB>");
|
|
|
|
ParseCommandLineFlags(&argc, const_cast<char***>(&argv), true);
|
2012-12-11 11:57:35 -08:00
|
|
|
|
|
|
|
Env* env = Env::Default();
|
|
|
|
std::string default_db_path;
|
|
|
|
env->GetTestDirectory(&default_db_path);
|
|
|
|
default_db_path += "db_repl_stress";
|
|
|
|
Options options;
|
|
|
|
options.create_if_missing = true;
|
2013-11-06 18:46:28 -08:00
|
|
|
options.WAL_ttl_seconds = FLAGS_wal_ttl_seconds;
|
|
|
|
options.WAL_size_limit_MB = FLAGS_wal_size_limit_MB;
|
2012-12-11 11:57:35 -08:00
|
|
|
DB* db;
|
2013-08-06 12:54:37 -07:00
|
|
|
DestroyDB(default_db_path, options);
|
2012-12-11 11:57:35 -08:00
|
|
|
|
|
|
|
Status s = DB::Open(options, default_db_path, &db);
|
|
|
|
|
|
|
|
if (!s.ok()) {
|
|
|
|
fprintf(stderr, "Could not open DB due to %s\n", s.ToString().c_str());
|
2013-08-06 12:54:37 -07:00
|
|
|
exit(1);
|
2012-12-11 11:57:35 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
DataPumpThread dataPump;
|
|
|
|
dataPump.no_records = FLAGS_num_inserts;
|
|
|
|
dataPump.db = db;
|
|
|
|
env->StartThread(DataPumpThreadBody, &dataPump);
|
|
|
|
|
|
|
|
ReplicationThread replThread;
|
|
|
|
replThread.db = db;
|
|
|
|
replThread.no_read = 0;
|
2014-10-27 14:50:21 -07:00
|
|
|
replThread.stop.store(false, std::memory_order_release);
|
2012-12-11 11:57:35 -08:00
|
|
|
|
|
|
|
env->StartThread(ReplicationThreadBody, &replThread);
|
2013-08-06 12:54:37 -07:00
|
|
|
while(replThread.no_read < FLAGS_num_inserts);
|
2014-10-27 14:50:21 -07:00
|
|
|
replThread.stop.store(true, std::memory_order_release);
|
2013-08-06 12:54:37 -07:00
|
|
|
if (replThread.no_read < dataPump.no_records) {
|
2012-12-11 11:57:35 -08:00
|
|
|
// no. read should be => than inserted.
|
2015-07-13 12:11:05 -07:00
|
|
|
fprintf(stderr,
|
|
|
|
"No. of Record's written and read not same\nRead : %" ROCKSDB_PRIszt
|
|
|
|
" Written : %" ROCKSDB_PRIszt "\n",
|
|
|
|
replThread.no_read, dataPump.no_records);
|
2012-12-11 11:57:35 -08:00
|
|
|
exit(1);
|
|
|
|
}
|
2013-08-06 12:54:37 -07:00
|
|
|
fprintf(stderr, "Successful!\n");
|
2012-12-11 11:57:35 -08:00
|
|
|
exit(0);
|
|
|
|
}
|
2014-05-08 17:25:13 -07:00
|
|
|
|
|
|
|
#endif // GFLAGS
|
2014-11-12 13:05:12 -08:00
|
|
|
|
|
|
|
#else // ROCKSDB_LITE
|
|
|
|
#include <stdio.h>
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
fprintf(stderr, "Not supported in lite mode.\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#endif // ROCKSDB_LITE
|