176 lines
4.8 KiB
C++
176 lines
4.8 KiB
C++
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
||
|
// This source code is licensed under both the GPLv2 (found in the
|
||
|
// COPYING file in the root directory) and Apache 2.0 License
|
||
|
// (found in the LICENSE.Apache file in the root directory).
|
||
|
|
||
|
#include "db/wal_edit.h"
|
||
|
|
||
|
#include "rocksdb/slice.h"
|
||
|
#include "rocksdb/status.h"
|
||
|
#include "util/coding.h"
|
||
|
|
||
|
namespace ROCKSDB_NAMESPACE {
|
||
|
|
||
|
void WalAddition::EncodeTo(std::string* dst) const {
|
||
|
PutVarint64(dst, number_);
|
||
|
|
||
|
if (metadata_.HasSize()) {
|
||
|
PutVarint32(dst, static_cast<uint32_t>(WalAdditionTag::kSize));
|
||
|
PutVarint64(dst, metadata_.GetSizeInBytes());
|
||
|
}
|
||
|
|
||
|
PutVarint32(dst, static_cast<uint32_t>(WalAdditionTag::kTerminate));
|
||
|
}
|
||
|
|
||
|
Status WalAddition::DecodeFrom(Slice* src) {
|
||
|
constexpr char class_name[] = "WalAddition";
|
||
|
|
||
|
if (!GetVarint64(src, &number_)) {
|
||
|
return Status::Corruption(class_name, "Error decoding WAL log number");
|
||
|
}
|
||
|
|
||
|
while (true) {
|
||
|
uint32_t tag_value = 0;
|
||
|
if (!GetVarint32(src, &tag_value)) {
|
||
|
return Status::Corruption(class_name, "Error decoding tag");
|
||
|
}
|
||
|
WalAdditionTag tag = static_cast<WalAdditionTag>(tag_value);
|
||
|
switch (tag) {
|
||
|
case WalAdditionTag::kSize: {
|
||
|
uint64_t size = 0;
|
||
|
if (!GetVarint64(src, &size)) {
|
||
|
return Status::Corruption(class_name, "Error decoding WAL file size");
|
||
|
}
|
||
|
metadata_.SetSizeInBytes(size);
|
||
|
break;
|
||
|
}
|
||
|
// TODO: process future tags such as checksum.
|
||
|
case WalAdditionTag::kTerminate:
|
||
|
return Status::OK();
|
||
|
default: {
|
||
|
std::stringstream ss;
|
||
|
ss << "Unknown tag " << tag_value;
|
||
|
return Status::Corruption(class_name, ss.str());
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
JSONWriter& operator<<(JSONWriter& jw, const WalAddition& wal) {
|
||
|
jw << "LogNumber" << wal.GetLogNumber() << "SizeInBytes"
|
||
|
<< wal.GetMetadata().GetSizeInBytes();
|
||
|
return jw;
|
||
|
}
|
||
|
|
||
|
std::ostream& operator<<(std::ostream& os, const WalAddition& wal) {
|
||
|
os << "log_number: " << wal.GetLogNumber()
|
||
|
<< " size_in_bytes: " << wal.GetMetadata().GetSizeInBytes();
|
||
|
return os;
|
||
|
}
|
||
|
|
||
|
std::string WalAddition::DebugString() const {
|
||
|
std::ostringstream oss;
|
||
|
oss << *this;
|
||
|
return oss.str();
|
||
|
}
|
||
|
|
||
|
void WalDeletion::EncodeTo(std::string* dst) const {
|
||
|
PutVarint64(dst, number_);
|
||
|
}
|
||
|
|
||
|
Status WalDeletion::DecodeFrom(Slice* src) {
|
||
|
constexpr char class_name[] = "WalDeletion";
|
||
|
|
||
|
if (!GetVarint64(src, &number_)) {
|
||
|
return Status::Corruption(class_name, "Error decoding WAL log number");
|
||
|
}
|
||
|
|
||
|
return Status::OK();
|
||
|
}
|
||
|
|
||
|
JSONWriter& operator<<(JSONWriter& jw, const WalDeletion& wal) {
|
||
|
jw << "LogNumber" << wal.GetLogNumber();
|
||
|
return jw;
|
||
|
}
|
||
|
|
||
|
std::ostream& operator<<(std::ostream& os, const WalDeletion& wal) {
|
||
|
os << "log_number: " << wal.GetLogNumber();
|
||
|
return os;
|
||
|
}
|
||
|
|
||
|
std::string WalDeletion::DebugString() const {
|
||
|
std::ostringstream oss;
|
||
|
oss << *this;
|
||
|
return oss.str();
|
||
|
}
|
||
|
|
||
|
Status WalSet::AddWal(const WalAddition& wal) {
|
||
|
auto it = wals_.lower_bound(wal.GetLogNumber());
|
||
|
if (wal.GetMetadata().HasSize()) {
|
||
|
// The WAL must exist without size.
|
||
|
if (it == wals_.end() || it->first != wal.GetLogNumber()) {
|
||
|
std::stringstream ss;
|
||
|
ss << "WAL " << wal.GetLogNumber() << " is not created before closing";
|
||
|
return Status::Corruption("WalSet", ss.str());
|
||
|
}
|
||
|
if (it->second.HasSize()) {
|
||
|
std::stringstream ss;
|
||
|
ss << "WAL " << wal.GetLogNumber() << " is closed more than once";
|
||
|
return Status::Corruption("WalSet", ss.str());
|
||
|
}
|
||
|
it->second = wal.GetMetadata();
|
||
|
} else {
|
||
|
// The WAL must not exist beforehand.
|
||
|
if (it != wals_.end() && it->first == wal.GetLogNumber()) {
|
||
|
std::stringstream ss;
|
||
|
ss << "WAL " << wal.GetLogNumber() << " is created more than once";
|
||
|
return Status::Corruption("WalSet", ss.str());
|
||
|
}
|
||
|
wals_[wal.GetLogNumber()] = wal.GetMetadata();
|
||
|
}
|
||
|
return Status::OK();
|
||
|
}
|
||
|
|
||
|
Status WalSet::AddWals(const WalAdditions& wals) {
|
||
|
Status s;
|
||
|
for (const WalAddition& wal : wals) {
|
||
|
s = AddWal(wal);
|
||
|
if (!s.ok()) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
Status WalSet::DeleteWal(const WalDeletion& wal) {
|
||
|
auto it = wals_.lower_bound(wal.GetLogNumber());
|
||
|
// The WAL must exist and has been closed.
|
||
|
if (it == wals_.end() || it->first != wal.GetLogNumber()) {
|
||
|
std::stringstream ss;
|
||
|
ss << "WAL " << wal.GetLogNumber() << " must exist before deletion";
|
||
|
return Status::Corruption("WalSet", ss.str());
|
||
|
}
|
||
|
if (!it->second.HasSize()) {
|
||
|
std::stringstream ss;
|
||
|
ss << "WAL " << wal.GetLogNumber() << " must be closed before deletion";
|
||
|
return Status::Corruption("WalSet", ss.str());
|
||
|
}
|
||
|
wals_.erase(it);
|
||
|
return Status::OK();
|
||
|
}
|
||
|
|
||
|
Status WalSet::DeleteWals(const WalDeletions& wals) {
|
||
|
Status s;
|
||
|
for (const WalDeletion& wal : wals) {
|
||
|
s = DeleteWal(wal);
|
||
|
if (!s.ok()) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
void WalSet::Reset() { wals_.clear(); }
|
||
|
|
||
|
} // namespace ROCKSDB_NAMESPACE
|