- // Copyright (c) 2011 The LevelDB Authors. All rights reserved.
- // Use of this source code is governed by a BSD-style license that can be
- // found in the LICENSE file. See the AUTHORS file for names of contributors.
-
- #include "db/builder.h"
-
- #include "db/dbformat.h"
- #include "db/filename.h"
- #include "db/table_cache.h"
- #include "db/version_edit.h"
- #include "leveldb/db.h"
- #include "leveldb/env.h"
- #include "leveldb/iterator.h"
-
- namespace leveldb {
-
- Status BuildTable(const std::string& dbname, Env* env, const Options& options,
- TableCache* table_cache, Iterator* iter, FileMetaData* meta) {
- Status s;
- meta->file_size = 0;
- iter->SeekToFirst();
-
- std::string fname = TableFileName(dbname, meta->number);
- if (iter->Valid()) {
- WritableFile* file;
- s = env->NewWritableFile(fname, &file);
- if (!s.ok()) {
- return s;
- }
-
- TableBuilder* builder = new TableBuilder(options, file);
- meta->smallest.DecodeFrom(iter->key());
- Slice key=iter->key();
- // 小合并时检查ttl
- time_t now = time(nullptr); // 获得当前时间
- uint64_t now_time=static_cast<uint64_t>(now);
- for (; iter->Valid(); iter->Next()) {
- uint64_t ttl=*(uint64_t*)(iter->value().data()+iter->value().size()-sizeof(uint64_t)); // 将 TTL 从 new_data 的末尾取出
- // 如果 TTL 超过当前时间,说明数据已经过期
- if(ttl < static_cast<uint64_t>(now_time)){
- continue;
- }
-
- key = iter->key();
- builder->Add(key, iter->value());
- }
- if (!key.empty()) {
- meta->largest.DecodeFrom(key);
- }
-
- // Finish and check for builder errors
- s = builder->Finish();
- if (s.ok()) {
- meta->file_size = builder->FileSize();
- assert(meta->file_size > 0);
- }
- delete builder;
-
- // Finish and check for file errors
- if (s.ok()) {
- s = file->Sync();
- }
- if (s.ok()) {
- s = file->Close();
- }
- delete file;
- file = nullptr;
-
- if (s.ok()) {
- // Verify that the table is usable
- Iterator* it = table_cache->NewIterator(ReadOptions(), meta->number,
- meta->file_size);
- s = it->status();
- delete it;
- }
- }
-
- // Check for input iterator errors
- if (!iter->status().ok()) {
- s = iter->status();
- }
-
- if (s.ok() && meta->file_size > 0) {
- // Keep it
- } else {
- env->RemoveFile(fname);
- }
- return s;
- }
-
- } // namespace leveldb
|