| // Copyright 2016 The Bazel Authors. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #include "src/main/cpp/util/bazel_log_handler.h" |
| |
| #include <chrono> // NOLINT -- for windows portability |
| #include <cstdio> |
| #include <cstdlib> |
| #include <ctime> |
| #include <iostream> |
| #include <sstream> |
| #include <string> |
| #include <utility> |
| |
| #include "src/main/cpp/util/logging.h" |
| |
| namespace blaze_util { |
| |
| BazelLogHandler::BazelLogHandler() |
| : debug_stream_(nullptr), |
| debug_stream_set_(false), |
| detail_(LOGGINGDETAIL_DEBUG), |
| debug_buffer_stream_(new std::stringstream()) {} |
| |
| BazelLogHandler::~BazelLogHandler() { |
| if (detail_ == LOGGINGDETAIL_DEBUG) { |
| // If SetLoggingOutputStream was never called, dump the buffer to stderr, |
| // otherwise, flush the stream. |
| if (debug_stream_ != nullptr) { |
| debug_stream_->flush(); |
| } else if (debug_buffer_stream_ != nullptr) { |
| std::cerr << debug_buffer_stream_->rdbuf(); |
| } else { |
| std::cerr << "Illegal state - neither a logfile nor a logbuffer " |
| << "existed at program end." << '\n'; |
| } |
| } |
| } |
| |
| // Messages intended for the user (level USER, along with WARNINGs an ERRORs) |
| // should be printed even if debug level logging was not requested. |
| void PrintUserLevelMessageToStream(std::ostream* stream, LogLevel level, |
| const std::string& message) { |
| if (level == LOGLEVEL_USER) { |
| (*stream) << message << '\n'; |
| } else if (level > LOGLEVEL_USER) { |
| (*stream) << LogLevelName(level) << ": " << message << '\n'; |
| } |
| // If level < USER, this is an INFO message. It's useful for debugging but |
| // should not be printed to the user unless the user has asked for debugging |
| // output. We ignore it here. |
| } |
| |
| static std::string Timestamp() { |
| auto now = std::chrono::system_clock::now(); |
| time_t s = std::chrono::system_clock::to_time_t(now); |
| auto ms = std::chrono::duration_cast<std::chrono::milliseconds>( |
| now.time_since_epoch()); |
| struct tm tmbuf = {}; |
| #ifdef _WIN32 |
| tmbuf = *localtime(&s); // NOLINT -- threadsafe on windows |
| #else |
| localtime_r(&s, &tmbuf); |
| #endif |
| char buf[16]; |
| int r = strftime(buf, sizeof buf - 5, "%H:%M:%S", &tmbuf); |
| r += snprintf(buf + r, +5, ".%03d", static_cast<int>(ms.count() % 1000)); |
| return std::string(buf, r); |
| } |
| |
| // For debug logs, print all logs, both debug logging and USER logs and above, |
| // along with information about where the log message came from. |
| void PrintDebugLevelMessageToStream(std::ostream* stream, |
| const std::string& filename, int line, |
| LogLevel level, |
| const std::string& message) { |
| (*stream) << "[" << LogLevelName(level) << " " << Timestamp() << " " |
| << filename << ":" << line << "] " << message << '\n'; |
| } |
| |
| void BazelLogHandler::HandleMessage(LogLevel level, const std::string& filename, |
| int line, const std::string& message, |
| int exit_code) { |
| if (detail_ != LOGGINGDETAIL_DEBUG) { |
| // If debugging was explicitly disabled, never print INFO messages, but |
| // messages of level USER and above should always be printed, as should |
| // warnings and errors. Omit the debug-level file and line number |
| // information, though. |
| LogLevel min_log_level = |
| detail_ == LOGGINGDETAIL_QUIET ? LOGLEVEL_ERROR : LOGLEVEL_USER; |
| if (level >= min_log_level) { |
| PrintUserLevelMessageToStream(&std::cerr, level, message); |
| } |
| if (level == LOGLEVEL_FATAL) { |
| std::exit(exit_code); |
| } |
| return; |
| } |
| |
| if (debug_stream_ == nullptr) { |
| // If we haven't decided whether messages should be logged to debug levels |
| // or not, buffer each version. This is redundant for USER levels and above, |
| // but is to make sure we can provide the right output to the user once we |
| // know that they do or do not want debug level information. |
| user_messages_.push_back(std::pair<LogLevel, std::string>(level, message)); |
| PrintDebugLevelMessageToStream(debug_buffer_stream_.get(), filename, line, |
| level, message); |
| } else { |
| // If an output stream has been specifically set, it is for the full suite |
| // of log messages. We don't print the user messages separately here as they |
| // are included. |
| PrintDebugLevelMessageToStream(debug_stream_, filename, line, level, |
| message); |
| } |
| |
| // If we have a fatal message, exit with the provided error code. |
| if (level == LOGLEVEL_FATAL) { |
| if (debug_stream_ != nullptr) { |
| PrintUserLevelMessageToStream(&std::cerr, level, message); |
| } |
| std::exit(exit_code); |
| } |
| } |
| |
| void BazelLogHandler::SetLoggingDetail(LoggingDetail detail, |
| std::ostream* debug_stream) { |
| // Disallow second calls to this, we only intend to support setting the output |
| // once, otherwise the buffering will not work as intended and the log will be |
| // fragmented. |
| BAZEL_CHECK(!debug_stream_set_) << "Tried to set log output a second time"; |
| debug_stream_set_ = true; |
| debug_stream_ = debug_stream; |
| detail_ = detail; |
| |
| if (detail == LOGGINGDETAIL_DEBUG) { |
| // The user asked for debug level information, which includes the user |
| // messages. We can discard the separate buffer at this point. |
| user_messages_.clear(); |
| FlushBufferToNewStreamAndSet(debug_buffer_stream_.get(), debug_stream_); |
| debug_buffer_stream_ = nullptr; |
| return; |
| } |
| |
| debug_buffer_stream_ = nullptr; |
| LogLevel min_log_level = |
| detail == LOGGINGDETAIL_QUIET ? LOGLEVEL_ERROR : LOGLEVEL_USER; |
| |
| for (const auto& log_entry : user_messages_) { |
| if (log_entry.first >= min_log_level) { |
| PrintUserLevelMessageToStream(&std::cerr, log_entry.first, |
| log_entry.second); |
| } |
| } |
| } |
| |
| void BazelLogHandler::Close() { |
| if (debug_stream_ != nullptr) { |
| debug_stream_->flush(); |
| } |
| |
| debug_stream_ = nullptr; |
| } |
| |
| void BazelLogHandler::FlushBufferToNewStreamAndSet( |
| std::stringstream* buffer, std::ostream* new_debug_stream) { |
| // Flush the buffer to the new stream, and print new log lines to it. |
| debug_stream_ = new_debug_stream; |
| // Transfer the contents of the buffer to the new stream, then remove the |
| // buffer. |
| *debug_stream_ << buffer->str(); |
| debug_stream_->flush(); |
| } |
| |
| } // namespace blaze_util |