Add NoSpace subcode to IOError (#1320)

Add a sub code to distinguish "out of space" errors from regular I/O errors
main
Edouard A 8 years ago committed by Siying Dong
parent 67036c0406
commit 66a91e2607
  1. 1
      db/db_io_failure_test.cc
  2. 2
      db/db_test_util.h
  3. 22
      include/rocksdb/status.h
  4. 8
      port/win/io_win.h
  5. 4
      util/env_hdfs.cc
  6. 5
      util/io_posix.h
  7. 5
      util/status.cc
  8. 3
      util/status_message.cc

@ -111,6 +111,7 @@ TEST_F(DBIOFailureTest, NoSpaceCompactRange) {
Status s = dbfull()->TEST_CompactRange(0, nullptr, nullptr, nullptr, Status s = dbfull()->TEST_CompactRange(0, nullptr, nullptr, nullptr,
true /* disallow trivial move */); true /* disallow trivial move */);
ASSERT_TRUE(s.IsIOError()); ASSERT_TRUE(s.IsIOError());
ASSERT_TRUE(s.IsNoSpace());
env_->no_space_.store(false, std::memory_order_release); env_->no_space_.store(false, std::memory_order_release);
} while (ChangeCompactOptions()); } while (ChangeCompactOptions());

@ -221,7 +221,7 @@ class SpecialEnv : public EnvWrapper {
// Drop writes on the floor // Drop writes on the floor
return Status::OK(); return Status::OK();
} else if (env_->no_space_.load(std::memory_order_acquire)) { } else if (env_->no_space_.load(std::memory_order_acquire)) {
return Status::IOError("No space left on device"); return Status::NoSpace("No space left on device");
} else { } else {
env_->bytes_written_ += data.size(); env_->bytes_written_ += data.size();
return base_->Append(data); return base_->Append(data);

@ -58,7 +58,7 @@ class Status {
kAborted = 10, kAborted = 10,
kBusy = 11, kBusy = 11,
kExpired = 12, kExpired = 12,
kTryAgain = 13, kTryAgain = 13
}; };
Code code() const { return code_; } Code code() const { return code_; }
@ -68,6 +68,7 @@ class Status {
kMutexTimeout = 1, kMutexTimeout = 1,
kLockTimeout = 2, kLockTimeout = 2,
kLockLimit = 3, kLockLimit = 3,
kNoSpace = 4,
kMaxSubCode kMaxSubCode
}; };
@ -157,6 +158,11 @@ class Status {
return Status(kTryAgain, msg, msg2); return Status(kTryAgain, msg, msg2);
} }
static Status NoSpace() { return Status(kIOError, kNoSpace); }
static Status NoSpace(const Slice& msg, const Slice& msg2 = Slice()) {
return Status(kIOError, kNoSpace, msg, msg2);
}
// Returns true iff the status indicates success. // Returns true iff the status indicates success.
bool ok() const { return code() == kOk; } bool ok() const { return code() == kOk; }
@ -200,6 +206,15 @@ class Status {
// re-attempted. // re-attempted.
bool IsTryAgain() const { return code() == kTryAgain; } bool IsTryAgain() const { return code() == kTryAgain; }
// Returns true iff the status indicates a NoSpace error
// This is caused by an I/O error returning the specific "out of space"
// error condition. Stricto sensu, an NoSpace error is an I/O error
// with a specific subcode, enabling users to take the appropriate action
// if needed
bool IsNoSpace() const {
return (code() == kIOError) && (subcode() == kNoSpace);
}
// Return a string representation of this status suitable for printing. // Return a string representation of this status suitable for printing.
// Returns the string "OK" for success. // Returns the string "OK" for success.
std::string ToString() const; std::string ToString() const;
@ -219,7 +234,10 @@ class Status {
explicit Status(Code _code, SubCode _subcode = kNone) explicit Status(Code _code, SubCode _subcode = kNone)
: code_(_code), subcode_(_subcode), state_(nullptr) {} : code_(_code), subcode_(_subcode), state_(nullptr) {}
Status(Code _code, const Slice& msg, const Slice& msg2); Status(Code _code, SubCode _subcode, const Slice& msg, const Slice& msg2);
Status(Code _code, const Slice& msg, const Slice& msg2)
: Status(_code, kNone, msg, msg2) {}
static const char* CopyState(const char* s); static const char* CopyState(const char* s);
}; };

@ -26,7 +26,9 @@ namespace port {
std::string GetWindowsErrSz(DWORD err); std::string GetWindowsErrSz(DWORD err);
inline Status IOErrorFromWindowsError(const std::string& context, DWORD err) { inline Status IOErrorFromWindowsError(const std::string& context, DWORD err) {
return Status::IOError(context, GetWindowsErrSz(err)); return (err == ERROR_HANDLE_DISK_FULL) ?
Status::NoSpace(context, GetWindowsErrSz(err)) :
Status::IOError(context, GetWindowsErrSz(err));
} }
inline Status IOErrorFromLastWindowsError(const std::string& context) { inline Status IOErrorFromLastWindowsError(const std::string& context) {
@ -34,7 +36,9 @@ inline Status IOErrorFromLastWindowsError(const std::string& context) {
} }
inline Status IOError(const std::string& context, int err_number) { inline Status IOError(const std::string& context, int err_number) {
return Status::IOError(context, strerror(err_number)); return (err_number == ENOSPC) ?
Status::NoSpace(context, strerror(err_number)) :
Status::IOError(context, strerror(err_number));
} }
// Note the below two do not set errno because they are used only here in this // Note the below two do not set errno because they are used only here in this

@ -36,7 +36,9 @@ namespace {
// Log error message // Log error message
static Status IOError(const std::string& context, int err_number) { static Status IOError(const std::string& context, int err_number) {
return Status::IOError(context, strerror(err_number)); return (err_number == ENOSPC) ?
Status::NoSpace(context, strerror(err_number)) :
Status::IOError(context, strerror(err_number));
} }
// assume that there is one global logger for now. It is not thread-safe, // assume that there is one global logger for now. It is not thread-safe,

@ -9,6 +9,7 @@
#pragma once #pragma once
#include <unistd.h> #include <unistd.h>
#include <atomic> #include <atomic>
#include <errno.h>
#include "rocksdb/env.h" #include "rocksdb/env.h"
// For non linux platform, the following macros are used only as place // For non linux platform, the following macros are used only as place
@ -24,7 +25,9 @@
namespace rocksdb { namespace rocksdb {
static Status IOError(const std::string& context, int err_number) { static Status IOError(const std::string& context, int err_number) {
return Status::IOError(context, strerror(err_number)); return (err_number == ENOSPC) ?
Status::NoSpace(context, strerror(err_number)) :
Status::IOError(context, strerror(err_number));
} }
class PosixHelper { class PosixHelper {

@ -21,9 +21,10 @@ const char* Status::CopyState(const char* state) {
return result; return result;
} }
Status::Status(Code _code, const Slice& msg, const Slice& msg2) Status::Status(Code _code, SubCode _subcode, const Slice& msg, const Slice& msg2)
: code_(_code), subcode_(kNone) { : code_(_code), subcode_(_subcode) {
assert(code_ != kOk); assert(code_ != kOk);
assert(subcode_ != kMaxSubCode);
const uint32_t len1 = static_cast<uint32_t>(msg.size()); const uint32_t len1 = static_cast<uint32_t>(msg.size());
const uint32_t len2 = static_cast<uint32_t>(msg2.size()); const uint32_t len2 = static_cast<uint32_t>(msg2.size());
const uint32_t size = len1 + (len2 ? (2 + len2) : 0); const uint32_t size = len1 + (len2 ? (2 + len2) : 0);

@ -11,7 +11,8 @@ const char* Status::msgs[] = {
"", // kNone "", // kNone
"Timeout Acquiring Mutex", // kMutexTimeout "Timeout Acquiring Mutex", // kMutexTimeout
"Timeout waiting to lock key", // kLockTimeout "Timeout waiting to lock key", // kLockTimeout
"Failed to acquire lock due to max_num_locks limit" // kLockLimit "Failed to acquire lock due to max_num_locks limit", // kLockLimit
"No space left on device" // kNoSpace
}; };
} // namespace rocksdb } // namespace rocksdb

Loading…
Cancel
Save