Summary: Basic test cases: - Manifest is lost or corrupt - Manifest refers to too many or too few SST files - SST file is corrupt - Unflushed data is present when RepairDB is called Depends on D55065 for its CreateFile() function in file_utils Test Plan: Ran the tests. Reviewers: IslamAbdelRahman, yhchiang, yoshinorim, sdong Reviewed By: sdong Subscribers: leveldb, andrewkr, dhruba Differential Revision: https://reviews.facebook.net/D55485main
parent
fbea4dc660
commit
e182f03c1e
@ -0,0 +1,177 @@ |
||||
// Copyright (c) 2016-present, Facebook, Inc. All rights reserved.
|
||||
// This source code is licensed under the BSD-style license found in the
|
||||
// LICENSE file in the root directory of this source tree. An additional grant
|
||||
// of patent rights can be found in the PATENTS file in the same directory.
|
||||
|
||||
#include <algorithm> |
||||
#include <string> |
||||
#include <vector> |
||||
|
||||
#include "db/db_impl.h" |
||||
#include "db/db_test_util.h" |
||||
#include "rocksdb/db.h" |
||||
#include "rocksdb/transaction_log.h" |
||||
#include "util/file_util.h" |
||||
|
||||
namespace rocksdb { |
||||
|
||||
class RepairTest : public DBTestBase { |
||||
public: |
||||
RepairTest() : DBTestBase("/repair_test") {} |
||||
|
||||
std::string GetFirstSstPath() { |
||||
uint64_t manifest_size; |
||||
std::vector<std::string> files; |
||||
db_->GetLiveFiles(files, &manifest_size); |
||||
auto sst_iter = |
||||
std::find_if(files.begin(), files.end(), [](const std::string& file) { |
||||
uint64_t number; |
||||
FileType type; |
||||
bool ok = ParseFileName(file, &number, &type); |
||||
return ok && type == kTableFile; |
||||
}); |
||||
return sst_iter == files.end() ? "" : dbname_ + *sst_iter; |
||||
} |
||||
}; |
||||
|
||||
TEST_F(RepairTest, LostManifest) { |
||||
// Add a couple SST files, delete the manifest, and verify RepairDB() saves
|
||||
// the day.
|
||||
Put("key", "val"); |
||||
Flush(); |
||||
Put("key2", "val2"); |
||||
Flush(); |
||||
// Need to get path before Close() deletes db_, but delete it after Close() to
|
||||
// ensure Close() didn't change the manifest.
|
||||
std::string manifest_path = |
||||
DescriptorFileName(dbname_, dbfull()->TEST_Current_Manifest_FileNo()); |
||||
|
||||
Close(); |
||||
ASSERT_OK(env_->FileExists(manifest_path)); |
||||
ASSERT_OK(env_->DeleteFile(manifest_path)); |
||||
RepairDB(dbname_, CurrentOptions()); |
||||
Reopen(CurrentOptions()); |
||||
|
||||
ASSERT_EQ(Get("key"), "val"); |
||||
ASSERT_EQ(Get("key2"), "val2"); |
||||
} |
||||
|
||||
TEST_F(RepairTest, CorruptManifest) { |
||||
// Manifest is in an invalid format. Expect a full recovery.
|
||||
Put("key", "val"); |
||||
Flush(); |
||||
Put("key2", "val2"); |
||||
Flush(); |
||||
// Need to get path before Close() deletes db_, but overwrite it after Close()
|
||||
// to ensure Close() didn't change the manifest.
|
||||
std::string manifest_path = |
||||
DescriptorFileName(dbname_, dbfull()->TEST_Current_Manifest_FileNo()); |
||||
|
||||
Close(); |
||||
ASSERT_OK(env_->FileExists(manifest_path)); |
||||
CreateFile(env_, manifest_path, "blah"); |
||||
RepairDB(dbname_, CurrentOptions()); |
||||
Reopen(CurrentOptions()); |
||||
|
||||
ASSERT_EQ(Get("key"), "val"); |
||||
ASSERT_EQ(Get("key2"), "val2"); |
||||
} |
||||
|
||||
TEST_F(RepairTest, IncompleteManifest) { |
||||
// In this case, the manifest is valid but does not reference all of the SST
|
||||
// files. Expect a full recovery.
|
||||
Put("key", "val"); |
||||
Flush(); |
||||
std::string orig_manifest_path = |
||||
DescriptorFileName(dbname_, dbfull()->TEST_Current_Manifest_FileNo()); |
||||
CopyFile(orig_manifest_path, orig_manifest_path + ".tmp"); |
||||
Put("key2", "val2"); |
||||
Flush(); |
||||
// Need to get path before Close() deletes db_, but overwrite it after Close()
|
||||
// to ensure Close() didn't change the manifest.
|
||||
std::string new_manifest_path = |
||||
DescriptorFileName(dbname_, dbfull()->TEST_Current_Manifest_FileNo()); |
||||
|
||||
Close(); |
||||
ASSERT_OK(env_->FileExists(new_manifest_path)); |
||||
// Replace the manifest with one that is only aware of the first SST file.
|
||||
CopyFile(orig_manifest_path + ".tmp", new_manifest_path); |
||||
RepairDB(dbname_, CurrentOptions()); |
||||
Reopen(CurrentOptions()); |
||||
|
||||
ASSERT_EQ(Get("key"), "val"); |
||||
ASSERT_EQ(Get("key2"), "val2"); |
||||
} |
||||
|
||||
TEST_F(RepairTest, LostSst) { |
||||
// Delete one of the SST files but preserve the manifest that refers to it,
|
||||
// then verify the DB is still usable for the intact SST.
|
||||
Put("key", "val"); |
||||
Flush(); |
||||
Put("key2", "val2"); |
||||
Flush(); |
||||
auto sst_path = GetFirstSstPath(); |
||||
ASSERT_FALSE(sst_path.empty()); |
||||
ASSERT_OK(env_->DeleteFile(sst_path)); |
||||
|
||||
Close(); |
||||
RepairDB(dbname_, CurrentOptions()); |
||||
Reopen(CurrentOptions()); |
||||
|
||||
// Exactly one of the key-value pairs should be in the DB now.
|
||||
ASSERT_TRUE((Get("key") == "val") != (Get("key2") == "val2")); |
||||
} |
||||
|
||||
TEST_F(RepairTest, CorruptSst) { |
||||
// Corrupt one of the SST files but preserve the manifest that refers to it,
|
||||
// then verify the DB is still usable for the intact SST.
|
||||
Put("key", "val"); |
||||
Flush(); |
||||
Put("key2", "val2"); |
||||
Flush(); |
||||
auto sst_path = GetFirstSstPath(); |
||||
ASSERT_FALSE(sst_path.empty()); |
||||
CreateFile(env_, sst_path, "blah"); |
||||
|
||||
Close(); |
||||
RepairDB(dbname_, CurrentOptions()); |
||||
Reopen(CurrentOptions()); |
||||
|
||||
// Exactly one of the key-value pairs should be in the DB now.
|
||||
ASSERT_TRUE((Get("key") == "val") != (Get("key2") == "val2")); |
||||
} |
||||
|
||||
TEST_F(RepairTest, UnflushedSst) { |
||||
// This test case invokes repair while some data is unflushed, then verifies
|
||||
// that data is in the db.
|
||||
Put("key", "val"); |
||||
VectorLogPtr wal_files; |
||||
ASSERT_OK(dbfull()->GetSortedWalFiles(wal_files)); |
||||
ASSERT_EQ(wal_files.size(), 1); |
||||
uint64_t total_ssts_size; |
||||
GetAllSSTFiles(&total_ssts_size); |
||||
ASSERT_EQ(total_ssts_size, 0); |
||||
// Need to get path before Close() deletes db_, but delete it after Close() to
|
||||
// ensure Close() didn't change the manifest.
|
||||
std::string manifest_path = |
||||
DescriptorFileName(dbname_, dbfull()->TEST_Current_Manifest_FileNo()); |
||||
|
||||
Close(); |
||||
ASSERT_OK(env_->FileExists(manifest_path)); |
||||
ASSERT_OK(env_->DeleteFile(manifest_path)); |
||||
RepairDB(dbname_, CurrentOptions()); |
||||
Reopen(CurrentOptions()); |
||||
|
||||
ASSERT_OK(dbfull()->GetSortedWalFiles(wal_files)); |
||||
ASSERT_EQ(wal_files.size(), 0); |
||||
GetAllSSTFiles(&total_ssts_size); |
||||
ASSERT_GT(total_ssts_size, 0); |
||||
ASSERT_EQ(Get("key"), "val"); |
||||
} |
||||
|
||||
} // namespace rocksdb
|
||||
|
||||
int main(int argc, char** argv) { |
||||
::testing::InitGoogleTest(&argc, argv); |
||||
return RUN_ALL_TESTS(); |
||||
} |
Loading…
Reference in new issue