Summary: Added put/get/scan/batchput/delete/approxsize Test Plan: Added pyunit script to test the newly added commands Reviewers: chip, leveldb Reviewed By: chip CC: zshao, emayanke Differential Revision: https://reviews.facebook.net/D7947main
parent
0b83a83191
commit
11ce6a060e
@ -0,0 +1,308 @@ |
|||||||
|
import os |
||||||
|
import os.path |
||||||
|
import shutil |
||||||
|
import subprocess |
||||||
|
import time |
||||||
|
import unittest |
||||||
|
import tempfile |
||||||
|
|
||||||
|
def my_check_output(*popenargs, **kwargs): |
||||||
|
""" |
||||||
|
If we had python 2.7, we should simply use subprocess.check_output. |
||||||
|
This is a stop-gap solution for python 2.6 |
||||||
|
""" |
||||||
|
if 'stdout' in kwargs: |
||||||
|
raise ValueError('stdout argument not allowed, it will be overridden.') |
||||||
|
process = subprocess.Popen(stdout=subprocess.PIPE, *popenargs, **kwargs) |
||||||
|
output, unused_err = process.communicate() |
||||||
|
retcode = process.poll() |
||||||
|
if retcode: |
||||||
|
cmd = kwargs.get("args") |
||||||
|
if cmd is None: |
||||||
|
cmd = popenargs[0] |
||||||
|
raise Exception("Exit code is not 0. It is %d. Command: %s" % |
||||||
|
(retcode, cmd)) |
||||||
|
return output |
||||||
|
|
||||||
|
|
||||||
|
class LDBTestCase(unittest.TestCase): |
||||||
|
def setUp(self): |
||||||
|
self.TMP_DIR = tempfile.mkdtemp(prefix="ldb_test_") |
||||||
|
self.DB_NAME = "testdb" |
||||||
|
|
||||||
|
def tearDown(self): |
||||||
|
assert(self.TMP_DIR.strip() != "/" |
||||||
|
and self.TMP_DIR.strip() != "/tmp" |
||||||
|
and self.TMP_DIR.strip() != "/tmp/") #Just some paranoia |
||||||
|
|
||||||
|
shutil.rmtree(self.TMP_DIR) |
||||||
|
|
||||||
|
def dbParam(self, dbName): |
||||||
|
return "--db=%s" % os.path.join(self.TMP_DIR, dbName) |
||||||
|
|
||||||
|
def assertRunOKFull(self, params, expectedOutput): |
||||||
|
""" |
||||||
|
All command-line params must be specified. |
||||||
|
Allows full flexibility in testing; for example: missing db param. |
||||||
|
|
||||||
|
""" |
||||||
|
|
||||||
|
output = my_check_output("./ldb %s |grep -v \"Created bg thread\"" % |
||||||
|
params, shell=True) |
||||||
|
self.assertEquals(output.strip(), expectedOutput.strip()); |
||||||
|
|
||||||
|
def assertRunFAILFull(self, params): |
||||||
|
""" |
||||||
|
All command-line params must be specified. |
||||||
|
Allows full flexibility in testing; for example: missing db param. |
||||||
|
|
||||||
|
""" |
||||||
|
try: |
||||||
|
my_check_output("./ldb %s |grep -v \"Created bg thread\"" % params, |
||||||
|
shell=True) |
||||||
|
except Exception, e: |
||||||
|
return |
||||||
|
self.fail( |
||||||
|
"Exception should have been raised for command with params: %s" % |
||||||
|
params) |
||||||
|
|
||||||
|
def assertRunOK(self, params, expectedOutput): |
||||||
|
""" |
||||||
|
Uses the default test db. |
||||||
|
|
||||||
|
""" |
||||||
|
self.assertRunOKFull("%s %s" % (self.dbParam(self.DB_NAME), params), |
||||||
|
expectedOutput) |
||||||
|
|
||||||
|
def assertRunFAIL(self, params): |
||||||
|
""" |
||||||
|
Uses the default test db. |
||||||
|
""" |
||||||
|
self.assertRunFAILFull("%s %s" % (self.dbParam(self.DB_NAME), params)) |
||||||
|
|
||||||
|
def testSimpleStringPutGet(self): |
||||||
|
self.assertRunFAIL("put x1 y1") |
||||||
|
self.assertRunOK("put --create_if_missing x1 y1", "OK") |
||||||
|
self.assertRunOK("get x1", "y1") |
||||||
|
self.assertRunFAIL("get x2") |
||||||
|
|
||||||
|
self.assertRunOK("put x2 y2", "OK") |
||||||
|
self.assertRunOK("get x1", "y1") |
||||||
|
self.assertRunOK("get x2", "y2") |
||||||
|
self.assertRunFAIL("get x3") |
||||||
|
|
||||||
|
self.assertRunOK("scan --from=x1 --to=z", "x1 : y1\nx2 : y2") |
||||||
|
self.assertRunOK("put x3 y3", "OK") |
||||||
|
|
||||||
|
self.assertRunOK("scan --from=x1 --to=z", "x1 : y1\nx2 : y2\nx3 : y3") |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3") |
||||||
|
self.assertRunOK("scan --from=x", "x1 : y1\nx2 : y2\nx3 : y3") |
||||||
|
|
||||||
|
self.assertRunOK("scan --to=x2", "x1 : y1") |
||||||
|
self.assertRunOK("scan --from=x1 --to=z --max_keys=1", "x1 : y1") |
||||||
|
self.assertRunOK("scan --from=x1 --to=z --max_keys=2", |
||||||
|
"x1 : y1\nx2 : y2") |
||||||
|
|
||||||
|
self.assertRunOK("scan --from=x1 --to=z --max_keys=3", |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3") |
||||||
|
self.assertRunOK("scan --from=x1 --to=z --max_keys=4", |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3") |
||||||
|
self.assertRunOK("scan --from=x1 --to=x2", "x1 : y1") |
||||||
|
self.assertRunOK("scan --from=x2 --to=x4", "x2 : y2\nx3 : y3") |
||||||
|
self.assertRunFAIL("scan --from=x4 --to=z") # No results => FAIL |
||||||
|
self.assertRunFAIL("scan --from=x1 --to=z --max_keys=foo") |
||||||
|
|
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3") |
||||||
|
|
||||||
|
self.assertRunOK("delete x1", "OK") |
||||||
|
self.assertRunOK("scan", "x2 : y2\nx3 : y3") |
||||||
|
|
||||||
|
self.assertRunOK("delete NonExistentKey", "OK") |
||||||
|
# It is wierd that GET and SCAN raise exception for |
||||||
|
# non-existent key, while delete does not |
||||||
|
|
||||||
|
def dumpDb(self, params, dumpFile): |
||||||
|
return 0 == os.system("./ldb dump %s > %s" % (params, dumpFile)) |
||||||
|
|
||||||
|
def loadDb(self, params, dumpFile): |
||||||
|
return 0 == os.system("cat %s | ./ldb load %s" % (dumpFile, params)) |
||||||
|
|
||||||
|
def testStringBatchPut(self): |
||||||
|
self.assertRunOK("batchput x1 y1 --create_if_missing", "OK") |
||||||
|
self.assertRunOK("scan", "x1 : y1") |
||||||
|
self.assertRunOK("batchput x2 y2 x3 y3 \"x4 abc\" \"y4 xyz\"", "OK") |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 abc : y4 xyz") |
||||||
|
self.assertRunFAIL("batchput") |
||||||
|
self.assertRunFAIL("batchput k1") |
||||||
|
self.assertRunFAIL("batchput k1 v1 k2") |
||||||
|
|
||||||
|
|
||||||
|
def testHexPutGet(self): |
||||||
|
self.assertRunOK("put a1 b1 --create_if_missing", "OK") |
||||||
|
self.assertRunOK("scan", "a1 : b1") |
||||||
|
self.assertRunOK("scan --hex", "0x6131 : 0x6231") |
||||||
|
self.assertRunFAIL("put --hex 6132 6232") |
||||||
|
self.assertRunOK("put --hex 0x6132 0x6232", "OK") |
||||||
|
self.assertRunOK("scan --hex", "0x6131 : 0x6231\n0x6132 : 0x6232") |
||||||
|
self.assertRunOK("scan", "a1 : b1\na2 : b2") |
||||||
|
self.assertRunOK("get a1", "b1") |
||||||
|
self.assertRunOK("get --hex 0x6131", "0x6231") |
||||||
|
self.assertRunOK("get a2", "b2") |
||||||
|
self.assertRunOK("get --hex 0x6132", "0x6232") |
||||||
|
self.assertRunOK("get --key_hex 0x6132", "b2") |
||||||
|
self.assertRunOK("get --key_hex --value_hex 0x6132", "0x6232") |
||||||
|
self.assertRunOK("get --value_hex a2", "0x6232") |
||||||
|
self.assertRunOK("scan --key_hex --value_hex", |
||||||
|
"0x6131 : 0x6231\n0x6132 : 0x6232") |
||||||
|
self.assertRunOK("scan --hex --from=0x6131 --to=0x6133", |
||||||
|
"0x6131 : 0x6231\n0x6132 : 0x6232") |
||||||
|
self.assertRunOK("scan --hex --from=0x6131 --to=0x6132", |
||||||
|
"0x6131 : 0x6231") |
||||||
|
self.assertRunOK("scan --key_hex", "0x6131 : b1\n0x6132 : b2") |
||||||
|
self.assertRunOK("scan --value_hex", "a1 : 0x6231\na2 : 0x6232") |
||||||
|
self.assertRunOK("batchput --hex 0x6133 0x6233 0x6134 0x6234", "OK") |
||||||
|
self.assertRunOK("scan", "a1 : b1\na2 : b2\na3 : b3\na4 : b4") |
||||||
|
self.assertRunOK("delete --hex 0x6133", "OK") |
||||||
|
self.assertRunOK("scan", "a1 : b1\na2 : b2\na4 : b4") |
||||||
|
|
||||||
|
|
||||||
|
def testInvalidCmdLines(self): |
||||||
|
# db not specified |
||||||
|
self.assertRunFAILFull("put 0x6133 0x6233 --hex --create_if_missing") |
||||||
|
# No param called he |
||||||
|
self.assertRunFAIL("put 0x6133 0x6233 --he --create_if_missing") |
||||||
|
# max_keys is not applicable for put |
||||||
|
self.assertRunFAIL("put 0x6133 0x6233 --max_keys=1 --create_if_missing") |
||||||
|
# hex has invalid boolean value |
||||||
|
self.assertRunFAIL("put 0x6133 0x6233 --hex=Boo --create_if_missing") |
||||||
|
|
||||||
|
|
||||||
|
def testDumpLoad(self): |
||||||
|
self.assertRunOK("batchput --create_if_missing x1 y1 x2 y2 x3 y3 x4 y4", |
||||||
|
"OK") |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
origDbPath = os.path.join(self.TMP_DIR, self.DB_NAME) |
||||||
|
|
||||||
|
# Dump and load without any additional params specified |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump1") |
||||||
|
loadedDbPath = os.path.join(self.TMP_DIR, "loaded_from_dump1") |
||||||
|
self.assertTrue(self.dumpDb("--db=%s" % origDbPath, dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb( |
||||||
|
"--db=%s --create_if_missing" % loadedDbPath, dumpFilePath)) |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
# Dump and load in hex |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump2") |
||||||
|
loadedDbPath = os.path.join(self.TMP_DIR, "loaded_from_dump2") |
||||||
|
self.assertTrue(self.dumpDb("--db=%s --hex" % origDbPath, dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb( |
||||||
|
"--db=%s --hex --create_if_missing" % loadedDbPath, dumpFilePath)) |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
# Dump only a portion of the key range |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump3") |
||||||
|
loadedDbPath = os.path.join(self.TMP_DIR, "loaded_from_dump3") |
||||||
|
self.assertTrue(self.dumpDb( |
||||||
|
"--db=%s --from=x1 --to=x3" % origDbPath, dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb( |
||||||
|
"--db=%s --create_if_missing" % loadedDbPath, dumpFilePath)) |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, "x1 : y1\nx2 : y2") |
||||||
|
|
||||||
|
# Dump upto max_keys rows |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump4") |
||||||
|
loadedDbPath = os.path.join(self.TMP_DIR, "loaded_from_dump4") |
||||||
|
self.assertTrue(self.dumpDb( |
||||||
|
"--db=%s --max_keys=3" % origDbPath, dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb( |
||||||
|
"--db=%s --create_if_missing" % loadedDbPath, dumpFilePath)) |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3") |
||||||
|
|
||||||
|
# Load into an existing db, create_if_missing is not specified |
||||||
|
self.assertTrue(self.dumpDb("--db=%s" % origDbPath, dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb("--db=%s" % loadedDbPath, dumpFilePath)) |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
# Dump and load with WAL disabled |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump5") |
||||||
|
loadedDbPath = os.path.join(self.TMP_DIR, "loaded_from_dump5") |
||||||
|
self.assertTrue(self.dumpDb("--db=%s" % origDbPath, dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb( |
||||||
|
"--db=%s --disable_wal --create_if_missing" % loadedDbPath, |
||||||
|
dumpFilePath)) |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
# Dump and load with lots of extra params specified |
||||||
|
extraParams = " ".join(["--bloom_bits=14", "--compression_type=bzip2", |
||||||
|
"--block_size=1024", "--auto_compaction=true", |
||||||
|
"--write_buffer_size=4194304", |
||||||
|
"--file_size=2097152"]) |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump6") |
||||||
|
loadedDbPath = os.path.join(self.TMP_DIR, "loaded_from_dump6") |
||||||
|
self.assertTrue(self.dumpDb( |
||||||
|
"--db=%s %s" % (origDbPath, extraParams), dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb( |
||||||
|
"--db=%s %s --create_if_missing" % (loadedDbPath, extraParams), |
||||||
|
dumpFilePath)) |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, |
||||||
|
"x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
# Dump with count_only |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump7") |
||||||
|
loadedDbPath = os.path.join(self.TMP_DIR, "loaded_from_dump7") |
||||||
|
self.assertTrue(self.dumpDb( |
||||||
|
"--db=%s --count_only" % origDbPath, dumpFilePath)) |
||||||
|
self.assertTrue(self.loadDb( |
||||||
|
"--db=%s --create_if_missing" % loadedDbPath, dumpFilePath)) |
||||||
|
# DB should have atleast one value for scan to work |
||||||
|
self.assertRunOKFull("put --db=%s k1 v1" % loadedDbPath, "OK") |
||||||
|
self.assertRunOKFull("scan --db=%s" % loadedDbPath, "k1 : v1") |
||||||
|
|
||||||
|
# Dump command fails because of typo in params |
||||||
|
dumpFilePath = os.path.join(self.TMP_DIR, "dump8") |
||||||
|
self.assertFalse(self.dumpDb( |
||||||
|
"--db=%s --create_if_missin" % origDbPath, dumpFilePath)) |
||||||
|
|
||||||
|
|
||||||
|
def testMiscAdminTask(self): |
||||||
|
# These tests need to be improved; for example with asserts about |
||||||
|
# whether compaction or level reduction actually took place. |
||||||
|
self.assertRunOK("batchput --create_if_missing x1 y1 x2 y2 x3 y3 x4 y4", |
||||||
|
"OK") |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
origDbPath = os.path.join(self.TMP_DIR, self.DB_NAME) |
||||||
|
|
||||||
|
self.assertTrue(0 == os.system("./ldb compact --db=%s" % origDbPath)) |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
self.assertTrue(0 == os.system( |
||||||
|
"./ldb reduce_levels --db=%s --new_levels=2" % origDbPath)) |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
self.assertTrue(0 == os.system( |
||||||
|
"./ldb reduce_levels --db=%s --new_levels=3" % origDbPath)) |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
self.assertTrue(0 == os.system( |
||||||
|
"./ldb compact --db=%s --from=x1 --to=x3" % origDbPath)) |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
self.assertTrue(0 == os.system( |
||||||
|
"./ldb compact --db=%s --hex --from=0x6131 --to=0x6134" % |
||||||
|
origDbPath)) |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
#TODO(dilip): Not sure what should be passed to WAL.Currently corrupted. |
||||||
|
self.assertTrue(0 == os.system( |
||||||
|
"./ldb dump_wal --db=%s --walfile=%s --header" % ( |
||||||
|
origDbPath, os.path.join(origDbPath, "LOG")))) |
||||||
|
self.assertRunOK("scan", "x1 : y1\nx2 : y2\nx3 : y3\nx4 : y4") |
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__": |
||||||
|
unittest.main() |
||||||
|
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,74 @@ |
|||||||
|
#ifndef LEVELDB_UTIL_LDB_CMD_EXECUTE_RESULT_H_ |
||||||
|
#define LEVELDB_UTIL_LDB_CMD_EXECUTE_RESULT_H_ |
||||||
|
|
||||||
|
namespace leveldb { |
||||||
|
|
||||||
|
class LDBCommandExecuteResult { |
||||||
|
public: |
||||||
|
enum State { |
||||||
|
EXEC_NOT_STARTED = 0, EXEC_SUCCEED = 1, EXEC_FAILED = 2, |
||||||
|
}; |
||||||
|
|
||||||
|
LDBCommandExecuteResult() { |
||||||
|
state_ = EXEC_NOT_STARTED; |
||||||
|
message_ = ""; |
||||||
|
} |
||||||
|
|
||||||
|
LDBCommandExecuteResult(State state, std::string& msg) { |
||||||
|
state_ = state; |
||||||
|
message_ = msg; |
||||||
|
} |
||||||
|
|
||||||
|
std::string ToString() { |
||||||
|
std::string ret; |
||||||
|
switch (state_) { |
||||||
|
case EXEC_SUCCEED: |
||||||
|
break; |
||||||
|
case EXEC_FAILED: |
||||||
|
ret.append("Failed: "); |
||||||
|
break; |
||||||
|
case EXEC_NOT_STARTED: |
||||||
|
ret.append("Not started: "); |
||||||
|
} |
||||||
|
if (!message_.empty()) { |
||||||
|
ret.append(message_); |
||||||
|
} |
||||||
|
return ret; |
||||||
|
} |
||||||
|
|
||||||
|
void Reset() { |
||||||
|
state_ = EXEC_NOT_STARTED; |
||||||
|
message_ = ""; |
||||||
|
} |
||||||
|
|
||||||
|
bool IsSucceed() { |
||||||
|
return state_ == EXEC_SUCCEED; |
||||||
|
} |
||||||
|
|
||||||
|
bool IsNotStarted() { |
||||||
|
return state_ == EXEC_NOT_STARTED; |
||||||
|
} |
||||||
|
|
||||||
|
bool IsFailed() { |
||||||
|
return state_ == EXEC_FAILED; |
||||||
|
} |
||||||
|
|
||||||
|
static LDBCommandExecuteResult SUCCEED(std::string msg) { |
||||||
|
return LDBCommandExecuteResult(EXEC_SUCCEED, msg); |
||||||
|
} |
||||||
|
|
||||||
|
static LDBCommandExecuteResult FAILED(std::string msg) { |
||||||
|
return LDBCommandExecuteResult(EXEC_FAILED, msg); |
||||||
|
} |
||||||
|
|
||||||
|
private: |
||||||
|
State state_; |
||||||
|
std::string message_; |
||||||
|
|
||||||
|
bool operator==(const LDBCommandExecuteResult&); |
||||||
|
bool operator!=(const LDBCommandExecuteResult&); |
||||||
|
}; |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
#endif |
Loading…
Reference in new issue