|
|
@ -20,23 +20,23 @@ |
|
|
|
|
|
|
|
|
|
|
|
namespace leveldb { |
|
|
|
namespace leveldb { |
|
|
|
|
|
|
|
|
|
|
|
static bool SnappyCompressionSupported() {
|
|
|
|
static bool SnappyCompressionSupported(const CompressionOptions& options) { |
|
|
|
std::string out;
|
|
|
|
std::string out; |
|
|
|
Slice in = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa";
|
|
|
|
Slice in = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; |
|
|
|
return port::Snappy_Compress(in.data(), in.size(), &out);
|
|
|
|
return port::Snappy_Compress(options, in.data(), in.size(), &out); |
|
|
|
}
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
static bool ZlibCompressionSupported() {
|
|
|
|
static bool ZlibCompressionSupported(const CompressionOptions& options) { |
|
|
|
std::string out;
|
|
|
|
std::string out; |
|
|
|
Slice in = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa";
|
|
|
|
Slice in = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; |
|
|
|
return port::Zlib_Compress(in.data(), in.size(), &out);
|
|
|
|
return port::Zlib_Compress(options, in.data(), in.size(), &out); |
|
|
|
}
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
static bool BZip2CompressionSupported() {
|
|
|
|
static bool BZip2CompressionSupported(const CompressionOptions& options) { |
|
|
|
std::string out;
|
|
|
|
std::string out; |
|
|
|
Slice in = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa";
|
|
|
|
Slice in = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; |
|
|
|
return port::BZip2_Compress(in.data(), in.size(), &out);
|
|
|
|
return port::BZip2_Compress(options, in.data(), in.size(), &out); |
|
|
|
}
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
static std::string RandomString(Random* rnd, int len) { |
|
|
|
static std::string RandomString(Random* rnd, int len) { |
|
|
|
std::string r; |
|
|
|
std::string r; |
|
|
@ -1076,57 +1076,59 @@ TEST(DBTest, CompactionTrigger) { |
|
|
|
ASSERT_EQ(NumTableFilesAtLevel(1), 1); |
|
|
|
ASSERT_EQ(NumTableFilesAtLevel(1), 1); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void MinLevelHelper(DBTest* self, Options& options) {
|
|
|
|
void MinLevelHelper(DBTest* self, Options& options) { |
|
|
|
Random rnd(301);
|
|
|
|
Random rnd(301); |
|
|
|
|
|
|
|
|
|
|
|
for (int num = 0;
|
|
|
|
for (int num = 0; |
|
|
|
num < options.level0_file_num_compaction_trigger - 1;
|
|
|
|
num < options.level0_file_num_compaction_trigger - 1; |
|
|
|
num++)
|
|
|
|
num++) |
|
|
|
{
|
|
|
|
{ |
|
|
|
std::vector<std::string> values;
|
|
|
|
std::vector<std::string> values; |
|
|
|
// Write 120KB (12 values, each 10K)
|
|
|
|
// Write 120KB (12 values, each 10K)
|
|
|
|
for (int i = 0; i < 12; i++) {
|
|
|
|
for (int i = 0; i < 12; i++) { |
|
|
|
values.push_back(RandomString(&rnd, 10000));
|
|
|
|
values.push_back(RandomString(&rnd, 10000)); |
|
|
|
ASSERT_OK(self->Put(Key(i), values[i]));
|
|
|
|
ASSERT_OK(self->Put(Key(i), values[i])); |
|
|
|
}
|
|
|
|
} |
|
|
|
self->dbfull()->TEST_WaitForCompactMemTable();
|
|
|
|
self->dbfull()->TEST_WaitForCompactMemTable(); |
|
|
|
ASSERT_EQ(self->NumTableFilesAtLevel(0), num + 1);
|
|
|
|
ASSERT_EQ(self->NumTableFilesAtLevel(0), num + 1); |
|
|
|
}
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
//generate one more file in level-0, and should trigger level-0 compaction
|
|
|
|
//generate one more file in level-0, and should trigger level-0 compaction
|
|
|
|
std::vector<std::string> values;
|
|
|
|
std::vector<std::string> values; |
|
|
|
for (int i = 0; i < 12; i++) {
|
|
|
|
for (int i = 0; i < 12; i++) { |
|
|
|
values.push_back(RandomString(&rnd, 10000));
|
|
|
|
values.push_back(RandomString(&rnd, 10000)); |
|
|
|
ASSERT_OK(self->Put(Key(i), values[i]));
|
|
|
|
ASSERT_OK(self->Put(Key(i), values[i])); |
|
|
|
}
|
|
|
|
} |
|
|
|
self->dbfull()->TEST_WaitForCompact();
|
|
|
|
self->dbfull()->TEST_WaitForCompact(); |
|
|
|
|
|
|
|
|
|
|
|
ASSERT_EQ(self->NumTableFilesAtLevel(0), 0);
|
|
|
|
ASSERT_EQ(self->NumTableFilesAtLevel(0), 0); |
|
|
|
ASSERT_EQ(self->NumTableFilesAtLevel(1), 1);
|
|
|
|
ASSERT_EQ(self->NumTableFilesAtLevel(1), 1); |
|
|
|
}
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
TEST(DBTest, MinLevelToCompress) {
|
|
|
|
void MinLevelToCompress(CompressionType& type, Options& options, int wbits, |
|
|
|
Options options = CurrentOptions();
|
|
|
|
int lev, int strategy) { |
|
|
|
options.write_buffer_size = 100<<10; //100KB
|
|
|
|
fprintf(stderr, "Test with compression options : window_bits = %d, level = %d, strategy = %d}\n", wbits, lev, strategy); |
|
|
|
options.num_levels = 3;
|
|
|
|
options.write_buffer_size = 100<<10; //100KB
|
|
|
|
options.max_mem_compaction_level = 0;
|
|
|
|
options.num_levels = 3; |
|
|
|
options.level0_file_num_compaction_trigger = 3;
|
|
|
|
options.max_mem_compaction_level = 0; |
|
|
|
|
|
|
|
options.level0_file_num_compaction_trigger = 3; |
|
|
|
options.create_if_missing = true; |
|
|
|
options.create_if_missing = true; |
|
|
|
CompressionType type; |
|
|
|
|
|
|
|
|
|
|
|
if (SnappyCompressionSupported(CompressionOptions(wbits, lev, strategy))) { |
|
|
|
if (SnappyCompressionSupported()) {
|
|
|
|
type = kSnappyCompression; |
|
|
|
type = kSnappyCompression;
|
|
|
|
fprintf(stderr, "using snappy\n"); |
|
|
|
fprintf(stderr, "using snappy\n");
|
|
|
|
} else if (ZlibCompressionSupported( |
|
|
|
} else if (ZlibCompressionSupported()) {
|
|
|
|
CompressionOptions(wbits, lev, strategy))) { |
|
|
|
type = kZlibCompression;
|
|
|
|
type = kZlibCompression; |
|
|
|
fprintf(stderr, "using zlib\n");
|
|
|
|
fprintf(stderr, "using zlib\n"); |
|
|
|
} else if (BZip2CompressionSupported()) {
|
|
|
|
} else if (BZip2CompressionSupported( |
|
|
|
type = kBZip2Compression;
|
|
|
|
CompressionOptions(wbits, lev, strategy))) { |
|
|
|
fprintf(stderr, "using bzip2\n");
|
|
|
|
type = kBZip2Compression; |
|
|
|
} else {
|
|
|
|
fprintf(stderr, "using bzip2\n"); |
|
|
|
fprintf(stderr, "skipping test, compression disabled\n");
|
|
|
|
} else { |
|
|
|
return;
|
|
|
|
fprintf(stderr, "skipping test, compression disabled\n"); |
|
|
|
}
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
options.compression_per_level = new CompressionType[options.num_levels]; |
|
|
|
options.compression_per_level = new CompressionType[options.num_levels]; |
|
|
|
|
|
|
|
|
|
|
|
// do not compress L0
|
|
|
|
// do not compress L0
|
|
|
@ -1136,9 +1138,32 @@ TEST(DBTest, MinLevelToCompress) { |
|
|
|
for (int i = 1; i < options.num_levels; i++) { |
|
|
|
for (int i = 1; i < options.num_levels; i++) { |
|
|
|
options.compression_per_level[i] = type; |
|
|
|
options.compression_per_level[i] = type; |
|
|
|
} |
|
|
|
} |
|
|
|
Reopen(&options);
|
|
|
|
} |
|
|
|
MinLevelHelper(this, options);
|
|
|
|
TEST(DBTest, MinLevelToCompress1) { |
|
|
|
|
|
|
|
Options options = CurrentOptions(); |
|
|
|
|
|
|
|
CompressionType type; |
|
|
|
|
|
|
|
MinLevelToCompress(type, options, -14, -1, 0); |
|
|
|
|
|
|
|
Reopen(&options); |
|
|
|
|
|
|
|
MinLevelHelper(this, options); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// do not compress L0 and L1
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++) { |
|
|
|
|
|
|
|
options.compression_per_level[i] = kNoCompression; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
for (int i = 2; i < options.num_levels; i++) { |
|
|
|
|
|
|
|
options.compression_per_level[i] = type; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
DestroyAndReopen(&options); |
|
|
|
|
|
|
|
MinLevelHelper(this, options); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TEST(DBTest, MinLevelToCompress2) { |
|
|
|
|
|
|
|
Options options = CurrentOptions(); |
|
|
|
|
|
|
|
CompressionType type; |
|
|
|
|
|
|
|
MinLevelToCompress(type, options, 15, -1, 0); |
|
|
|
|
|
|
|
Reopen(&options); |
|
|
|
|
|
|
|
MinLevelHelper(this, options); |
|
|
|
|
|
|
|
|
|
|
|
// do not compress L0 and L1
|
|
|
|
// do not compress L0 and L1
|
|
|
|
for (int i = 0; i < 2; i++) { |
|
|
|
for (int i = 0; i < 2; i++) { |
|
|
|
options.compression_per_level[i] = kNoCompression; |
|
|
|
options.compression_per_level[i] = kNoCompression; |
|
|
@ -1146,9 +1171,9 @@ TEST(DBTest, MinLevelToCompress) { |
|
|
|
for (int i = 2; i < options.num_levels; i++) { |
|
|
|
for (int i = 2; i < options.num_levels; i++) { |
|
|
|
options.compression_per_level[i] = type; |
|
|
|
options.compression_per_level[i] = type; |
|
|
|
} |
|
|
|
} |
|
|
|
DestroyAndReopen(&options);
|
|
|
|
DestroyAndReopen(&options); |
|
|
|
MinLevelHelper(this, options);
|
|
|
|
MinLevelHelper(this, options); |
|
|
|
}
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
TEST(DBTest, RepeatedWritesToSameKey) { |
|
|
|
TEST(DBTest, RepeatedWritesToSameKey) { |
|
|
|
Options options = CurrentOptions(); |
|
|
|
Options options = CurrentOptions(); |
|
|
@ -1682,6 +1707,29 @@ TEST(DBTest, DBOpen_Options) { |
|
|
|
db = NULL; |
|
|
|
db = NULL; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TEST(DBTest, DBOpen_Change_NumLevels) { |
|
|
|
|
|
|
|
std::string dbname = test::TmpDir() + "/db_change_num_levels"; |
|
|
|
|
|
|
|
DestroyDB(dbname, Options()); |
|
|
|
|
|
|
|
Options opts; |
|
|
|
|
|
|
|
Status s; |
|
|
|
|
|
|
|
DB* db = NULL; |
|
|
|
|
|
|
|
opts.create_if_missing = true; |
|
|
|
|
|
|
|
s = DB::Open(opts, dbname, &db); |
|
|
|
|
|
|
|
ASSERT_OK(s); |
|
|
|
|
|
|
|
ASSERT_TRUE(db != NULL); |
|
|
|
|
|
|
|
db->Put(WriteOptions(), "a", "123"); |
|
|
|
|
|
|
|
db->Put(WriteOptions(), "b", "234"); |
|
|
|
|
|
|
|
db->CompactRange(NULL, NULL); |
|
|
|
|
|
|
|
delete db; |
|
|
|
|
|
|
|
db = NULL; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
opts.create_if_missing = false; |
|
|
|
|
|
|
|
opts.num_levels = 2; |
|
|
|
|
|
|
|
s = DB::Open(opts, dbname, &db); |
|
|
|
|
|
|
|
ASSERT_TRUE(strstr(s.ToString().c_str(), "Corruption") != NULL); |
|
|
|
|
|
|
|
ASSERT_TRUE(db == NULL); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Check that number of files does not grow when we are out of space
|
|
|
|
// Check that number of files does not grow when we are out of space
|
|
|
|
TEST(DBTest, NoSpace) { |
|
|
|
TEST(DBTest, NoSpace) { |
|
|
|
Options options = CurrentOptions(); |
|
|
|
Options options = CurrentOptions(); |
|
|
@ -1828,7 +1876,7 @@ TEST(DBTest, SnapshotFiles) { |
|
|
|
uint64_t size; |
|
|
|
uint64_t size; |
|
|
|
ASSERT_OK(env_->GetFileSize(src, &size)); |
|
|
|
ASSERT_OK(env_->GetFileSize(src, &size)); |
|
|
|
|
|
|
|
|
|
|
|
// record the number and the size of the
|
|
|
|
// record the number and the size of the
|
|
|
|
// latest manifest file
|
|
|
|
// latest manifest file
|
|
|
|
if (ParseFileName(files[i].substr(1), &number, &type)) { |
|
|
|
if (ParseFileName(files[i].substr(1), &number, &type)) { |
|
|
|
if (type == kDescriptorFile) { |
|
|
|
if (type == kDescriptorFile) { |
|
|
@ -1843,7 +1891,7 @@ TEST(DBTest, SnapshotFiles) { |
|
|
|
ASSERT_OK(env_->NewSequentialFile(src, &srcfile)); |
|
|
|
ASSERT_OK(env_->NewSequentialFile(src, &srcfile)); |
|
|
|
WritableFile* destfile; |
|
|
|
WritableFile* destfile; |
|
|
|
ASSERT_OK(env_->NewWritableFile(dest, &destfile)); |
|
|
|
ASSERT_OK(env_->NewWritableFile(dest, &destfile)); |
|
|
|
|
|
|
|
|
|
|
|
char buffer[4096]; |
|
|
|
char buffer[4096]; |
|
|
|
Slice slice; |
|
|
|
Slice slice; |
|
|
|
while (size > 0) { |
|
|
|
while (size > 0) { |
|
|
@ -1866,7 +1914,7 @@ TEST(DBTest, SnapshotFiles) { |
|
|
|
extras.push_back(RandomString(&rnd, 100000)); |
|
|
|
extras.push_back(RandomString(&rnd, 100000)); |
|
|
|
ASSERT_OK(Put(Key(i), extras[i])); |
|
|
|
ASSERT_OK(Put(Key(i), extras[i])); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// verify that data in the snapshot are correct
|
|
|
|
// verify that data in the snapshot are correct
|
|
|
|
Options opts; |
|
|
|
Options opts; |
|
|
|
DB* snapdb; |
|
|
|
DB* snapdb; |
|
|
@ -1882,7 +1930,7 @@ TEST(DBTest, SnapshotFiles) { |
|
|
|
} |
|
|
|
} |
|
|
|
delete snapdb; |
|
|
|
delete snapdb; |
|
|
|
|
|
|
|
|
|
|
|
// look at the new live files after we added an 'extra' key
|
|
|
|
// look at the new live files after we added an 'extra' key
|
|
|
|
// and after we took the first snapshot.
|
|
|
|
// and after we took the first snapshot.
|
|
|
|
uint64_t new_manifest_number = 0; |
|
|
|
uint64_t new_manifest_number = 0; |
|
|
|
uint64_t new_manifest_size = 0; |
|
|
|
uint64_t new_manifest_size = 0; |
|
|
@ -1896,7 +1944,7 @@ TEST(DBTest, SnapshotFiles) { |
|
|
|
// previous shapshot.
|
|
|
|
// previous shapshot.
|
|
|
|
for (unsigned int i = 0; i < newfiles.size(); i++) { |
|
|
|
for (unsigned int i = 0; i < newfiles.size(); i++) { |
|
|
|
std::string src = dbname_ + "/" + newfiles[i]; |
|
|
|
std::string src = dbname_ + "/" + newfiles[i]; |
|
|
|
// record the lognumber and the size of the
|
|
|
|
// record the lognumber and the size of the
|
|
|
|
// latest manifest file
|
|
|
|
// latest manifest file
|
|
|
|
if (ParseFileName(newfiles[i].substr(1), &number, &type)) { |
|
|
|
if (ParseFileName(newfiles[i].substr(1), &number, &type)) { |
|
|
|
if (type == kDescriptorFile) { |
|
|
|
if (type == kDescriptorFile) { |
|
|
@ -1911,7 +1959,7 @@ TEST(DBTest, SnapshotFiles) { |
|
|
|
} |
|
|
|
} |
|
|
|
ASSERT_EQ(manifest_number, new_manifest_number); |
|
|
|
ASSERT_EQ(manifest_number, new_manifest_number); |
|
|
|
ASSERT_GT(new_manifest_size, manifest_size); |
|
|
|
ASSERT_GT(new_manifest_size, manifest_size); |
|
|
|
|
|
|
|
|
|
|
|
// release file snapshot
|
|
|
|
// release file snapshot
|
|
|
|
dbfull()->DisableFileDeletions(); |
|
|
|
dbfull()->DisableFileDeletions(); |
|
|
|
} |
|
|
|
} |
|
|
@ -1975,7 +2023,7 @@ TEST(DBTest, ReadCompaction) { |
|
|
|
// in some level, indicating that there was a compaction
|
|
|
|
// in some level, indicating that there was a compaction
|
|
|
|
ASSERT_TRUE(NumTableFilesAtLevel(0) < l1 || |
|
|
|
ASSERT_TRUE(NumTableFilesAtLevel(0) < l1 || |
|
|
|
NumTableFilesAtLevel(1) < l2 || |
|
|
|
NumTableFilesAtLevel(1) < l2 || |
|
|
|
NumTableFilesAtLevel(2) < l3);
|
|
|
|
NumTableFilesAtLevel(2) < l3); |
|
|
|
delete options.block_cache; |
|
|
|
delete options.block_cache; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|