rocksdb/thrift/test/simpletest.cpp
Dhruba Borthakur d5503208cf Share a single cache for all the DBs served by this server.
Summary:

Test Plan:

Reviewers:

CC:

Task ID: #

Blame Rev:
2012-07-11 14:08:46 -07:00

225 lines
6 KiB
C++

/**
* Tests for Thrift server for leveldb
* @author Dhruba Borthakur (dhruba@gmail.com)
* Copyright 2012 Facebook
*/
#include <protocol/TBinaryProtocol.h>
#include <transport/TSocket.h>
#include <transport/TBufferTransports.h>
#include <DB.h>
#include <leveldb_types.h>
#include "server_options.h"
using namespace apache::thrift;
using namespace apache::thrift::protocol;
using namespace apache::thrift::transport;
using boost::shared_ptr;
using namespace Tleveldb;
extern "C" void startServer(int argc, char**argv);
extern "C" void stopServer(int port);
extern ServerOptions server_options;
static DBHandle dbhandle;
static DBClient* dbclient;
static const Text dbname = "test";
static pthread_t serverThread;
static int ARGC;
static char** ARGV;
static void cleanupDir(std::string dir) {
// remove old data, if any
int ret = unlink(dir.c_str());
char* cleanup = new char[100];
snprintf(cleanup, 100, "rm -rf %s", dir.c_str());
system(cleanup);
}
static void createDatabase() {
DBOptions options;
options.create_if_missing = true; // create
options.error_if_exists = false; // overwrite
options.write_buffer_size = (4<<20); // 4 MB
options.max_open_files = 1000;
options.block_size = 4096;
options.block_restart_interval = 16;
options.compression = kSnappyCompression;
cleanupDir(server_options.getDataDirectory(dbname));
// create the database
dbclient->Open(dbhandle, dbname, options);
}
static void testClient(int port) {
boost::shared_ptr<TSocket> socket(new TSocket("localhost", port));
boost::shared_ptr<TTransport> transport(new TBufferedTransport(socket));
boost::shared_ptr<TProtocol> protocol(new TBinaryProtocol(transport));
WriteOptions writeOptions;
// open database
dbclient = new DBClient(protocol);
transport->open();
createDatabase();
printf("Database created.\n");
// insert record into leveldb
Slice key;
key.data = "Key1";
key.size = 4;
Slice value;
value.data = "value1";
value.size = 6;
kv keyvalue;
keyvalue.key = key;
keyvalue.value = value;
int ret = dbclient->Put(dbhandle, keyvalue, writeOptions);
assert(ret == Code::kOk);
printf("Put Key1 suceeded.\n");
//read it back
ReadOptions readOptions;
ResultItem rValue;
dbclient->Get(rValue, dbhandle, key, readOptions);
assert(rValue.status == Code::kOk);
assert(value.data.compare(rValue.value.data) == 0);
assert(value.size == rValue.value.size);
printf("Get suceeded.\n");
// get a snapshot
ResultSnapshot rsnap;
dbclient->GetSnapshot(rsnap, dbhandle);
assert(rsnap.status == Code::kOk);
assert(rsnap.snapshot.snapshotid > 0);
printf("Snapshot suceeded.\n");
// insert a new record into leveldb
Slice key2;
key2.data = "Key2";
key2.size = 4;
Slice value2;
value2.data = "value2";
value2.size = 6;
keyvalue.key = key2;
keyvalue.value = value2;
ret = dbclient->Put(dbhandle, keyvalue, writeOptions);
assert(ret == Code::kOk);
printf("Put Key2 suceeded.\n");
// verify that a get done with a previous snapshot does not find Key2
readOptions.snapshot = rsnap.snapshot;
dbclient->Get(rValue, dbhandle, key2, readOptions);
assert(rValue.status == Code::kNotFound);
printf("Get with snapshot succeeded.\n");
// release snapshot
ret = dbclient->ReleaseSnapshot(dbhandle, rsnap.snapshot);
assert(ret == Code::kOk);
printf("Snapshot released.\n");
// if we try to re-release the same snapshot, it should fail
ret = dbclient->ReleaseSnapshot(dbhandle, rsnap.snapshot);
assert(ret == Code::kNotFound);
// compact whole database
Slice range;
range.size = 0;
ret = dbclient->CompactRange(dbhandle, range, range);
assert(ret == Code::kOk);
printf("Compaction trigger suceeded.\n");
// create a new iterator to scan all keys from the start
Slice target;
ResultIterator ri;
readOptions.snapshot.snapshotid = 0;
dbclient->NewIterator(ri, dbhandle, readOptions,
IteratorType::seekToFirst, target);
assert(ri.status == Code::kOk);
int foundPairs = 0;
while (true) {
ResultPair pair;
dbclient->GetNext(pair, dbhandle, ri.iterator);
if (pair.status == Code::kOk) {
foundPairs++;
} else {
break;
}
}
assert(foundPairs == 2);
ret = dbclient->DeleteIterator(dbhandle, ri.iterator);
assert(ret == Code::kOk);
printf("Iterator scan-all forward passes.\n");
// create a new iterator, position at end and scan backwards
readOptions.snapshot.snapshotid = 0;
dbclient->NewIterator(ri, dbhandle, readOptions,
IteratorType::seekToLast, target);
assert(ri.status == Code::kOk);
foundPairs = 0;
while (true) {
ResultPair pair;
dbclient->GetPrev(pair, dbhandle, ri.iterator);
if (pair.status == Code::kOk) {
foundPairs++;
} else {
break;
}
}
assert(foundPairs == 2);
ret = dbclient->DeleteIterator(dbhandle, ri.iterator);
assert(ret == Code::kOk);
printf("Iterator scan-all backward passes.\n");
// create a new iterator, position at middle
readOptions.snapshot.snapshotid = 0;
target = key;
dbclient->NewIterator(ri, dbhandle, readOptions,
IteratorType::seekToKey, target);
assert(ri.status == Code::kOk);
foundPairs = 0;
while (true) {
ResultPair pair;
dbclient->GetPrev(pair, dbhandle, ri.iterator);
if (pair.status == Code::kOk) {
foundPairs++;
} else {
break;
}
}
assert(foundPairs == 1);
ret = dbclient->DeleteIterator(dbhandle, ri.iterator);
assert(ret == Code::kOk);
printf("Iterator scan-selective backward passes.\n");
// close database
dbclient->Close(dbhandle, dbname);
transport->close();
}
static void* startTestServer(void *arg) {
printf("Server test server\n");
startServer(ARGC, ARGV);
}
int main(int argc, char **argv) {
ARGC = argc;
ARGV = argv;
// create a server
int rc = pthread_create(&serverThread, NULL, startTestServer, NULL);
printf("Server thread created.\n");
// give some time to the server to initialize itself
while (server_options.getPort() == 0) {
sleep(1);
}
// test client
testClient(server_options.getPort());
return 0;
}