Fixed a bug in EventListener::OnCompactionCompleted().

Summary:
Fixed a bug in EventListener::OnCompactionCompleted() that returns
incorrect list of input / output file names.

Test Plan: Extend existing test in listener_test.cc

Reviewers: sdong, rven, igor

Reviewed By: igor

Subscribers: dhruba, leveldb

Differential Revision: https://reviews.facebook.net/D38349
This commit is contained in:
Yueh-Hsuan Chiang 2015-05-12 16:10:23 -07:00
parent dbd95b7532
commit 14431e971d
3 changed files with 25 additions and 6 deletions

View File

@ -630,18 +630,21 @@ void ColumnFamilyData::NotifyOnCompactionCompleted(
DB* db, Compaction* c, const Status& status) { DB* db, Compaction* c, const Status& status) {
#ifndef ROCKSDB_LITE #ifndef ROCKSDB_LITE
auto listeners = ioptions()->listeners; auto listeners = ioptions()->listeners;
ASSERT_GT(listeners.size(), 0U);
CompactionJobInfo info; CompactionJobInfo info;
info.cf_name = c->column_family_data()->GetName(); info.cf_name = c->column_family_data()->GetName();
info.status = status; info.status = status;
info.output_level = c->output_level(); info.output_level = c->output_level();
for (const auto fmd : *c->inputs(c->level())) { for (size_t i = 0; i < c->num_input_levels(); ++i) {
info.input_files.push_back( for (const auto fmd : *c->inputs(i)) {
TableFileName(options_.db_paths, info.input_files.push_back(
fmd->fd.GetNumber(), TableFileName(options_.db_paths,
fmd->fd.GetPathId())); fmd->fd.GetNumber(),
fmd->fd.GetPathId()));
}
} }
for (const auto newf : c->edit()->GetNewFiles()) { for (const auto newf : c->edit()->GetNewFiles()) {
info.input_files.push_back( info.output_files.push_back(
TableFileName(options_.db_paths, TableFileName(options_.db_paths,
newf.second.fd.GetNumber(), newf.second.fd.GetNumber(),
newf.second.fd.GetPathId())); newf.second.fd.GetPathId()));

View File

@ -152,10 +152,14 @@ class EventListenerTest : public testing::Test {
class TestCompactionListener : public EventListener { class TestCompactionListener : public EventListener {
public: public:
void OnCompactionCompleted(DB *db, const CompactionJobInfo& ci) override { void OnCompactionCompleted(DB *db, const CompactionJobInfo& ci) override {
std::lock_guard<std::mutex> lock(mutex_);
compacted_dbs_.push_back(db); compacted_dbs_.push_back(db);
ASSERT_GT(ci.input_files.size(), 0U);
ASSERT_GT(ci.output_files.size(), 0U);
} }
std::vector<DB*> compacted_dbs_; std::vector<DB*> compacted_dbs_;
std::mutex mutex_;
}; };
TEST_F(EventListenerTest, OnSingleDBCompactionTest) { TEST_F(EventListenerTest, OnSingleDBCompactionTest) {

View File

@ -45,6 +45,18 @@ struct CompactionJobInfo {
// actual thread that involves in that specific event. For example, it // actual thread that involves in that specific event. For example, it
// is the RocksDB background flush thread that does the actual flush to // is the RocksDB background flush thread that does the actual flush to
// call EventListener::OnFlushCompleted(). // call EventListener::OnFlushCompleted().
//
// [Locking] All EventListener callbacks are designed to be called without
// the current thread holding any DB mutex. This is to prevent potential
// deadlock and performance issue when using EventListener callback
// in a complex way. However, all EventListener call-back functions
// should not run for an extended period of time before the function
// returns, otherwise RocksDB may be blocked. For example, it is not
// suggested to do DB::CompactFiles() (as it may run for a long while)
// or issue many of DB::Put() (as Put may be blocked in certain cases)
// in the same thread in the EventListener callback. However, doing
// DB::CompactFiles() and DB::Put() in a thread other than the
// EventListener callback thread is considered safe.
class EventListener { class EventListener {
public: public:
// A call-back function to RocksDB which will be called whenever a // A call-back function to RocksDB which will be called whenever a