mirror of
https://github.com/facebook/rocksdb.git
synced 2024-11-26 16:30:56 +00:00
555ca3e7b7
* std::chrono does not provide enough granularity for microsecs and periodically emits duplicates * the bug is manifested in log rotation logic where we get duplicate log file names and loose previous log content * msvc does not imlement COW on std::strings adjusted the test to use refs in the loops as auto does not retain ref info * adjust auto_log rotation test with Windows specific command to remove a folder. The test previously worked because we have unix utils installed in house but this may not be the case for everyone.
162 lines
4.8 KiB
C++
162 lines
4.8 KiB
C++
// 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 "util/auto_roll_logger.h"
|
|
#include "util/mutexlock.h"
|
|
|
|
using namespace std;
|
|
|
|
namespace rocksdb {
|
|
|
|
// -- AutoRollLogger
|
|
Status AutoRollLogger::ResetLogger() {
|
|
status_ = env_->NewLogger(log_fname_, &logger_);
|
|
|
|
if (!status_.ok()) {
|
|
return status_;
|
|
}
|
|
|
|
if (logger_->GetLogFileSize() == Logger::kDoNotSupportGetLogFileSize) {
|
|
status_ = Status::NotSupported(
|
|
"The underlying logger doesn't support GetLogFileSize()");
|
|
}
|
|
if (status_.ok()) {
|
|
cached_now = static_cast<uint64_t>(env_->NowMicros() * 1e-6);
|
|
ctime_ = cached_now;
|
|
cached_now_access_count = 0;
|
|
}
|
|
|
|
return status_;
|
|
}
|
|
|
|
void AutoRollLogger::RollLogFile() {
|
|
std::string old_fname = OldInfoLogFileName(
|
|
dbname_, env_->NowMicros(), db_absolute_path_, db_log_dir_);
|
|
env_->RenameFile(log_fname_, old_fname);
|
|
}
|
|
|
|
string AutoRollLogger::ValistToString(const char* format, va_list args) const {
|
|
// Any log messages longer than 1024 will get truncated.
|
|
// The user is responsible for chopping longer messages into multi line log
|
|
static const int MAXBUFFERSIZE = 1024;
|
|
char buffer[MAXBUFFERSIZE];
|
|
|
|
int count = vsnprintf(buffer, MAXBUFFERSIZE, format, args);
|
|
(void) count;
|
|
assert(count >= 0);
|
|
|
|
return buffer;
|
|
}
|
|
|
|
void AutoRollLogger::LogInternal(const char* format, ...) {
|
|
mutex_.AssertHeld();
|
|
va_list args;
|
|
va_start(args, format);
|
|
logger_->Logv(format, args);
|
|
va_end(args);
|
|
}
|
|
|
|
void AutoRollLogger::Logv(const char* format, va_list ap) {
|
|
assert(GetStatus().ok());
|
|
|
|
std::shared_ptr<Logger> logger;
|
|
{
|
|
MutexLock l(&mutex_);
|
|
if ((kLogFileTimeToRoll > 0 && LogExpired()) ||
|
|
(kMaxLogFileSize > 0 && logger_->GetLogFileSize() >= kMaxLogFileSize)) {
|
|
RollLogFile();
|
|
Status s = ResetLogger();
|
|
if (!s.ok()) {
|
|
// can't really log the error if creating a new LOG file failed
|
|
return;
|
|
}
|
|
|
|
WriteHeaderInfo();
|
|
}
|
|
|
|
// pin down the current logger_ instance before releasing the mutex.
|
|
logger = logger_;
|
|
}
|
|
|
|
// Another thread could have put a new Logger instance into logger_ by now.
|
|
// However, since logger is still hanging on to the previous instance
|
|
// (reference count is not zero), we don't have to worry about it being
|
|
// deleted while we are accessing it.
|
|
// Note that logv itself is not mutex protected to allow maximum concurrency,
|
|
// as thread safety should have been handled by the underlying logger.
|
|
logger->Logv(format, ap);
|
|
}
|
|
|
|
void AutoRollLogger::WriteHeaderInfo() {
|
|
mutex_.AssertHeld();
|
|
for (auto& header : headers_) {
|
|
LogInternal("%s", header.c_str());
|
|
}
|
|
}
|
|
|
|
void AutoRollLogger::LogHeader(const char* format, va_list args) {
|
|
// header message are to be retained in memory. Since we cannot make any
|
|
// assumptions about the data contained in va_list, we will retain them as
|
|
// strings
|
|
va_list tmp;
|
|
va_copy(tmp, args);
|
|
string data = ValistToString(format, tmp);
|
|
va_end(tmp);
|
|
|
|
MutexLock l(&mutex_);
|
|
headers_.push_back(data);
|
|
|
|
// Log the original message to the current log
|
|
logger_->Logv(format, args);
|
|
}
|
|
|
|
bool AutoRollLogger::LogExpired() {
|
|
if (cached_now_access_count >= call_NowMicros_every_N_records_) {
|
|
cached_now = static_cast<uint64_t>(env_->NowMicros() * 1e-6);
|
|
cached_now_access_count = 0;
|
|
}
|
|
|
|
++cached_now_access_count;
|
|
return cached_now >= ctime_ + kLogFileTimeToRoll;
|
|
}
|
|
|
|
Status CreateLoggerFromOptions(
|
|
const std::string& dbname,
|
|
const std::string& db_log_dir,
|
|
Env* env,
|
|
const DBOptions& options,
|
|
std::shared_ptr<Logger>* logger) {
|
|
std::string db_absolute_path;
|
|
env->GetAbsolutePath(dbname, &db_absolute_path);
|
|
std::string fname = InfoLogFileName(dbname, db_absolute_path, db_log_dir);
|
|
|
|
env->CreateDirIfMissing(dbname); // In case it does not exist
|
|
// Currently we only support roll by time-to-roll and log size
|
|
if (options.log_file_time_to_roll > 0 || options.max_log_file_size > 0) {
|
|
AutoRollLogger* result = new AutoRollLogger(
|
|
env, dbname, db_log_dir,
|
|
options.max_log_file_size,
|
|
options.log_file_time_to_roll, options.info_log_level);
|
|
Status s = result->GetStatus();
|
|
if (!s.ok()) {
|
|
delete result;
|
|
} else {
|
|
logger->reset(result);
|
|
}
|
|
return s;
|
|
} else {
|
|
// Open a log file in the same directory as the db
|
|
env->RenameFile(fname, OldInfoLogFileName(dbname, env->NowMicros(),
|
|
db_absolute_path, db_log_dir));
|
|
auto s = env->NewLogger(fname, logger);
|
|
if (logger->get() != nullptr) {
|
|
(*logger)->SetInfoLogLevel(options.info_log_level);
|
|
}
|
|
return s;
|
|
}
|
|
}
|
|
|
|
} // namespace rocksdb
|