Summary: This is just a simple test that passes two files though a compaction. It shows the framework so that people can continue building new compaction *unit* tests. In the future we might want to move some Compaction* tests from DBTest here. For example, CompactBetweenSnapshot seems a good candidate. Hopefully this test can be simpler when we mock out VersionSet. Test Plan: this is a test Reviewers: ljin, rven, yhchiang, sdong Reviewed By: sdong Subscribers: dhruba, leveldb Differential Revision: https://reviews.facebook.net/D28449main
							parent
							
								
									7fe247080f
								
							
						
					
					
						commit
						9be338cf9d
					
				| @ -0,0 +1,176 @@ | |||||||
|  | //  Copyright (c) 2013, 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 <map> | ||||||
|  | #include <string> | ||||||
|  | 
 | ||||||
|  | #include "db/compaction_job.h" | ||||||
|  | #include "db/column_family.h" | ||||||
|  | #include "db/version_set.h" | ||||||
|  | #include "rocksdb/cache.h" | ||||||
|  | #include "rocksdb/options.h" | ||||||
|  | #include "rocksdb/db.h" | ||||||
|  | #include "util/testharness.h" | ||||||
|  | #include "util/testutil.h" | ||||||
|  | #include "table/mock_table.h" | ||||||
|  | 
 | ||||||
|  | namespace rocksdb { | ||||||
|  | 
 | ||||||
|  | // TODO(icanadi) Make it simpler once we mock out VersionSet
 | ||||||
|  | class CompactionJobTest { | ||||||
|  |  public: | ||||||
|  |   CompactionJobTest() | ||||||
|  |       : env_(Env::Default()), | ||||||
|  |         dbname_(test::TmpDir() + "/compaction_job_test"), | ||||||
|  |         table_cache_(NewLRUCache(50000, 16, 8)), | ||||||
|  |         versions_(new VersionSet(dbname_, &db_options_, env_options_, | ||||||
|  |                                  table_cache_.get(), &write_controller_)), | ||||||
|  |         shutting_down_(false), | ||||||
|  |         mock_table_factory_(new mock::MockTableFactory()) { | ||||||
|  |     ASSERT_OK(env_->CreateDirIfMissing(dbname_)); | ||||||
|  |     db_options_.db_paths.emplace_back(dbname_, | ||||||
|  |                                       std::numeric_limits<uint64_t>::max()); | ||||||
|  |     NewDB(); | ||||||
|  |     std::vector<ColumnFamilyDescriptor> column_families; | ||||||
|  |     cf_options_.table_factory = mock_table_factory_; | ||||||
|  |     column_families.emplace_back(kDefaultColumnFamilyName, cf_options_); | ||||||
|  | 
 | ||||||
|  |     mutable_cf_options_.RefreshDerivedOptions(ImmutableCFOptions(Options())); | ||||||
|  | 
 | ||||||
|  |     ASSERT_OK(versions_->Recover(column_families, false)); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   std::string GenerateFileName(uint64_t file_number) { | ||||||
|  |     FileMetaData meta; | ||||||
|  |     std::vector<DbPath> db_paths; | ||||||
|  |     db_paths.emplace_back(dbname_, std::numeric_limits<uint64_t>::max()); | ||||||
|  |     meta.fd = FileDescriptor(file_number, 0, 0); | ||||||
|  |     return TableFileName(db_paths, meta.fd.GetNumber(), meta.fd.GetPathId()); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // returns expected result after compaction
 | ||||||
|  |   mock::MockFileContents CreateTwoFiles() { | ||||||
|  |     mock::MockFileContents expected_results; | ||||||
|  |     const int kKeysPerFile = 10000; | ||||||
|  |     SequenceNumber sequence_number = 0; | ||||||
|  |     for (int i = 0; i < 2; ++i) { | ||||||
|  |       mock::MockFileContents contents; | ||||||
|  |       SequenceNumber smallest_seqno, largest_seqno; | ||||||
|  |       InternalKey smallest, largest; | ||||||
|  |       for (int k = 0; k < kKeysPerFile; ++k) { | ||||||
|  |         auto key = std::to_string(i * (kKeysPerFile / 2) + k); | ||||||
|  |         auto value = std::to_string(i * kKeysPerFile + k); | ||||||
|  |         InternalKey internal_key(key, ++sequence_number, kTypeValue); | ||||||
|  |         if (k == 0) { | ||||||
|  |           smallest = internal_key; | ||||||
|  |           smallest_seqno = sequence_number; | ||||||
|  |         } else if (k == kKeysPerFile - 1) { | ||||||
|  |           largest = internal_key; | ||||||
|  |           largest_seqno = sequence_number; | ||||||
|  |         } | ||||||
|  |         std::pair<std::string, std::string> key_value( | ||||||
|  |             {internal_key.Encode().ToString(), value}); | ||||||
|  |         contents.insert(key_value); | ||||||
|  |         if (i == 1 || k < kKeysPerFile / 2) { | ||||||
|  |           expected_results.insert(key_value); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       uint64_t file_number = versions_->NewFileNumber(); | ||||||
|  |       ASSERT_OK(mock_table_factory_->CreateMockTable( | ||||||
|  |           env_, GenerateFileName(file_number), std::move(contents))); | ||||||
|  | 
 | ||||||
|  |       VersionEdit edit; | ||||||
|  |       edit.AddFile(0, file_number, 0, 10, smallest, largest, smallest_seqno, | ||||||
|  |                    largest_seqno); | ||||||
|  | 
 | ||||||
|  |       mutex_.Lock(); | ||||||
|  |       versions_->LogAndApply(versions_->GetColumnFamilySet()->GetDefault(), | ||||||
|  |                              mutable_cf_options_, &edit, &mutex_); | ||||||
|  |       mutex_.Unlock(); | ||||||
|  |     } | ||||||
|  |     versions_->SetLastSequence(sequence_number); | ||||||
|  |     return expected_results; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   void NewDB() { | ||||||
|  |     VersionEdit new_db; | ||||||
|  |     new_db.SetLogNumber(0); | ||||||
|  |     new_db.SetNextFile(2); | ||||||
|  |     new_db.SetLastSequence(0); | ||||||
|  | 
 | ||||||
|  |     const std::string manifest = DescriptorFileName(dbname_, 1); | ||||||
|  |     unique_ptr<WritableFile> file; | ||||||
|  |     Status s = env_->NewWritableFile( | ||||||
|  |         manifest, &file, env_->OptimizeForManifestWrite(env_options_)); | ||||||
|  |     ASSERT_OK(s); | ||||||
|  |     { | ||||||
|  |       log::Writer log(std::move(file)); | ||||||
|  |       std::string record; | ||||||
|  |       new_db.EncodeTo(&record); | ||||||
|  |       s = log.AddRecord(record); | ||||||
|  |     } | ||||||
|  |     ASSERT_OK(s); | ||||||
|  |     // Make "CURRENT" file that points to the new manifest file.
 | ||||||
|  |     s = SetCurrentFile(env_, dbname_, 1, nullptr); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Env* env_; | ||||||
|  |   std::string dbname_; | ||||||
|  |   EnvOptions env_options_; | ||||||
|  |   MutableCFOptions mutable_cf_options_; | ||||||
|  |   std::shared_ptr<Cache> table_cache_; | ||||||
|  |   WriteController write_controller_; | ||||||
|  |   DBOptions db_options_; | ||||||
|  |   ColumnFamilyOptions cf_options_; | ||||||
|  |   std::unique_ptr<VersionSet> versions_; | ||||||
|  |   port::Mutex mutex_; | ||||||
|  |   std::atomic<bool> shutting_down_; | ||||||
|  |   std::shared_ptr<mock::MockTableFactory> mock_table_factory_; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | TEST(CompactionJobTest, Simple) { | ||||||
|  |   auto cfd = versions_->GetColumnFamilySet()->GetDefault(); | ||||||
|  | 
 | ||||||
|  |   auto expected_results = CreateTwoFiles(); | ||||||
|  | 
 | ||||||
|  |   auto files = cfd->current()->storage_info()->LevelFiles(0); | ||||||
|  |   ASSERT_EQ(2U, files.size()); | ||||||
|  | 
 | ||||||
|  |   std::unique_ptr<Compaction> compaction(Compaction::TEST_NewCompaction( | ||||||
|  |       7, 0, 1, 1024 * 1024, 10, 0, kNoCompression)); | ||||||
|  |   compaction->SetInputVersion(cfd->current()); | ||||||
|  | 
 | ||||||
|  |   auto compaction_input_files = compaction->TEST_GetInputFiles(0); | ||||||
|  |   compaction_input_files->level = 0; | ||||||
|  |   compaction_input_files->files.push_back(files[0]); | ||||||
|  |   compaction_input_files->files.push_back(files[1]); | ||||||
|  | 
 | ||||||
|  |   SnapshotList snapshots; | ||||||
|  |   int yield_callback_called = 0; | ||||||
|  |   std::function<uint64_t()> yield_callback = [&]() { | ||||||
|  |     yield_callback_called++; | ||||||
|  |     return 0; | ||||||
|  |   }; | ||||||
|  |   LogBuffer log_buffer(InfoLogLevel::INFO_LEVEL, db_options_.info_log.get()); | ||||||
|  |   mutex_.Lock(); | ||||||
|  |   CompactionJob compaction_job( | ||||||
|  |       compaction.get(), db_options_, *cfd->GetLatestMutableCFOptions(), | ||||||
|  |       env_options_, versions_.get(), &shutting_down_, &log_buffer, nullptr, | ||||||
|  |       nullptr, &snapshots, true, table_cache_, std::move(yield_callback)); | ||||||
|  |   compaction_job.Prepare(); | ||||||
|  |   mutex_.Unlock(); | ||||||
|  |   ASSERT_OK(compaction_job.Run()); | ||||||
|  |   mutex_.Lock(); | ||||||
|  |   compaction_job.Install(Status::OK(), &mutex_); | ||||||
|  |   mutex_.Unlock(); | ||||||
|  | 
 | ||||||
|  |   mock_table_factory_->AssertLatestFile(expected_results); | ||||||
|  |   ASSERT_EQ(yield_callback_called, 20000); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | }  // namespace rocksdb
 | ||||||
|  | 
 | ||||||
|  | int main(int argc, char** argv) { return rocksdb::test::RunAllTests(); } | ||||||
					Loading…
					
					
				
		Reference in new issue
	
	 Igor Canadi
						Igor Canadi