fix handling of empty string as checkpoint directory

Summary:
- made `CreateCheckpoint` properly return `InvalidArgument` when called with an empty directory. Previously it triggered an assertion failure due to a bug in the logic.
- made `ldb` set empty `checkpoint_dir` if that's what the user specifies, so that we can use it to properly test `CreateCheckpoint` in the future.

Differential Revision: D6874562

fbshipit-source-id: dcc1bd41768261d9338987fa7711444289707ed7
This commit is contained in:
Andrew Kryczka 2018-02-20 16:42:06 -08:00 committed by Facebook Github Bot
parent 5263da6396
commit 1960e73e21
4 changed files with 15 additions and 7 deletions

View File

@ -4,6 +4,7 @@
* Iterator::SeekForPrev is now a pure virtual method. This is to prevent user who implement the Iterator interface fail to implement SeekForPrev by mistake. * Iterator::SeekForPrev is now a pure virtual method. This is to prevent user who implement the Iterator interface fail to implement SeekForPrev by mistake.
* Add `include_end` option to make the range end exclusive when `include_end == false` in `DeleteFilesInRange()`. * Add `include_end` option to make the range end exclusive when `include_end == false` in `DeleteFilesInRange()`.
* Add `CompactRangeOptions::allow_write_stall`, which makes `CompactRange` start working immediately, even if it causes user writes to stall. The default value is false, meaning we add delay to `CompactRange` calls until stalling can be avoided when possible. Note this delay is not present in previous RocksDB versions. * Add `CompactRangeOptions::allow_write_stall`, which makes `CompactRange` start working immediately, even if it causes user writes to stall. The default value is false, meaning we add delay to `CompactRange` calls until stalling can be avoided when possible. Note this delay is not present in previous RocksDB versions.
* Creating checkpoint with empty directory now returns `Status::InvalidArgument`; previously, it returned `Status::IOError`.
### New Features ### New Features
* Improve the performance of iterators doing long range scans by using readahead. * Improve the performance of iterators doing long range scans by using readahead.

View File

@ -2616,10 +2616,7 @@ CheckPointCommand::CheckPointCommand(
: LDBCommand(options, flags, false /* is_read_only */, : LDBCommand(options, flags, false /* is_read_only */,
BuildCmdLineOptions({ARG_CHECKPOINT_DIR})) { BuildCmdLineOptions({ARG_CHECKPOINT_DIR})) {
auto itr = options.find(ARG_CHECKPOINT_DIR); auto itr = options.find(ARG_CHECKPOINT_DIR);
if (itr == options.end()) { if (itr != options.end()) {
exec_state_ = LDBCommandExecuteResult::Failed(
"--" + ARG_CHECKPOINT_DIR + ": missing checkpoint directory");
} else {
checkpoint_dir_ = itr->second; checkpoint_dir_ = itr->second;
} }
} }

View File

@ -62,9 +62,10 @@ Status CheckpointImpl::CreateCheckpoint(const std::string& checkpoint_dir,
size_t final_nonslash_idx = checkpoint_dir.find_last_not_of('/'); size_t final_nonslash_idx = checkpoint_dir.find_last_not_of('/');
if (final_nonslash_idx == std::string::npos) { if (final_nonslash_idx == std::string::npos) {
// npos means it's only slashes, which means it's the root directory, but it // npos means it's only slashes or empty. Non-empty means it's the root
// shouldn't be because we verified above the directory doesn't exist. // directory, but it shouldn't be because we verified above the directory
assert(false); // doesn't exist.
assert(checkpoint_dir.empty());
return Status::InvalidArgument("invalid checkpoint directory name"); return Status::InvalidArgument("invalid checkpoint directory name");
} }

View File

@ -564,6 +564,15 @@ TEST_F(CheckpointTest, CurrentFileModifiedWhileCheckpointing2PC) {
delete txdb; delete txdb;
} }
TEST_F(CheckpointTest, CheckpointInvalidDirectoryName) {
for (std::string checkpoint_dir : {"", "/", "////"}) {
Checkpoint* checkpoint;
ASSERT_OK(Checkpoint::Create(db_, &checkpoint));
ASSERT_TRUE(checkpoint->CreateCheckpoint("").IsInvalidArgument());
delete checkpoint;
}
}
} // namespace rocksdb } // namespace rocksdb
int main(int argc, char** argv) { int main(int argc, char** argv) {